core/homeassistant/components/esphome/config_flow.py

180 lines
6.2 KiB
Python
Raw Normal View History

"""Config flow to configure esphome component."""
from collections import OrderedDict
from typing import Optional
import voluptuous as vol
from homeassistant import config_entries
from homeassistant.helpers import ConfigType
from .entry_data import DATA_KEY, RuntimeEntryData
2019-07-31 19:25:30 +00:00
@config_entries.HANDLERS.register("esphome")
class EsphomeFlowHandler(config_entries.ConfigFlow):
"""Handle a esphome config flow."""
VERSION = 1
CONNECTION_CLASS = config_entries.CONN_CLASS_LOCAL_PUSH
def __init__(self):
"""Initialize flow."""
self._host = None # type: Optional[str]
self._port = None # type: Optional[int]
self._password = None # type: Optional[str]
2019-07-31 19:25:30 +00:00
async def async_step_user(
self, user_input: Optional[ConfigType] = None, error: Optional[str] = None
):
"""Handle a flow initialized by the user."""
if user_input is not None:
return await self._async_authenticate_or_add(user_input)
fields = OrderedDict()
2019-07-31 19:25:30 +00:00
fields[vol.Required("host", default=self._host or vol.UNDEFINED)] = str
fields[vol.Optional("port", default=self._port or 6053)] = int
errors = {}
if error is not None:
2019-07-31 19:25:30 +00:00
errors["base"] = error
return self.async_show_form(
2019-07-31 19:25:30 +00:00
step_id="user", data_schema=vol.Schema(fields), errors=errors
)
@property
def _name(self):
2019-07-31 19:25:30 +00:00
return self.context.get("name")
@_name.setter
def _name(self, value):
# pylint: disable=unsupported-assignment-operation
2019-07-31 19:25:30 +00:00
self.context["name"] = value
self.context["title_placeholders"] = {"name": self._name}
def _set_user_input(self, user_input):
if user_input is None:
return
2019-07-31 19:25:30 +00:00
self._host = user_input["host"]
self._port = user_input["port"]
async def _async_authenticate_or_add(self, user_input):
self._set_user_input(user_input)
error, device_info = await self.fetch_device_info()
if error is not None:
return await self.async_step_user(error=error)
self._name = device_info.name
# Only show authentication step if device uses password
if device_info.uses_password:
return await self.async_step_authenticate()
return self._async_get_entry()
async def async_step_discovery_confirm(self, user_input=None):
"""Handle user-confirmation of discovered node."""
if user_input is not None:
return await self._async_authenticate_or_add(None)
return self.async_show_form(
2019-07-31 19:25:30 +00:00
step_id="discovery_confirm", description_placeholders={"name": self._name}
)
async def async_step_zeroconf(self, user_input: ConfigType):
"""Handle zeroconf discovery."""
# Hostname is format: livingroom.local.
2019-07-31 19:25:30 +00:00
local_name = user_input["hostname"][:-1]
node_name = local_name[: -len(".local")]
address = user_input["properties"].get("address", local_name)
# Check if already configured
for entry in self._async_current_entries():
already_configured = False
2019-07-31 19:25:30 +00:00
if entry.data["host"] == address:
# Is this address already configured?
already_configured = True
elif entry.entry_id in self.hass.data.get(DATA_KEY, {}):
# Does a config entry with this name already exist?
data = self.hass.data[DATA_KEY][
2019-07-31 19:25:30 +00:00
entry.entry_id
] # type: RuntimeEntryData
# Node names are unique in the network
if data.device_info is not None:
already_configured = data.device_info.name == node_name
if already_configured:
2019-07-31 19:25:30 +00:00
return self.async_abort(reason="already_configured")
self._host = address
2019-07-31 19:25:30 +00:00
self._port = user_input["port"]
self._name = node_name
# Check if flow for this device already in progress
for flow in self._async_in_progress():
2019-07-31 19:25:30 +00:00
if flow["context"].get("name") == node_name:
return self.async_abort(reason="already_configured")
return await self.async_step_discovery_confirm()
def _async_get_entry(self):
return self.async_create_entry(
title=self._name,
data={
2019-07-31 19:25:30 +00:00
"host": self._host,
"port": self._port,
# The API uses protobuf, so empty string denotes absence
2019-07-31 19:25:30 +00:00
"password": self._password or "",
},
)
async def async_step_authenticate(self, user_input=None, error=None):
"""Handle getting password for authentication."""
if user_input is not None:
2019-07-31 19:25:30 +00:00
self._password = user_input["password"]
error = await self.try_login()
if error:
return await self.async_step_authenticate(error=error)
return self._async_get_entry()
errors = {}
if error is not None:
2019-07-31 19:25:30 +00:00
errors["base"] = error
return self.async_show_form(
2019-07-31 19:25:30 +00:00
step_id="authenticate",
data_schema=vol.Schema({vol.Required("password"): str}),
description_placeholders={"name": self._name},
errors=errors,
)
async def fetch_device_info(self):
"""Fetch device info from API and return any errors."""
from aioesphomeapi import APIClient, APIConnectionError
2019-07-31 19:25:30 +00:00
cli = APIClient(self.hass.loop, self._host, self._port, "")
try:
await cli.connect()
device_info = await cli.device_info()
except APIConnectionError as err:
2019-07-31 19:25:30 +00:00
if "resolving" in str(err):
return "resolve_error", None
return "connection_error", None
finally:
await cli.disconnect(force=True)
return None, device_info
async def try_login(self):
"""Try logging in to device and return any errors."""
from aioesphomeapi import APIClient, APIConnectionError
cli = APIClient(self.hass.loop, self._host, self._port, self._password)
try:
await cli.connect(login=True)
except APIConnectionError:
await cli.disconnect(force=True)
2019-07-31 19:25:30 +00:00
return "invalid_password"
return None