HA-EasyComputerManager/custom_components/easy_computer_manage/config_flow.py

109 lines
3.3 KiB
Python
Raw Normal View History

"""Config flow for Easy Computer Manage integration."""
from __future__ import annotations
import logging
from typing import Any
import voluptuous as vol
from paramiko.ssh_exception import AuthenticationException
from homeassistant import config_entries, exceptions
from homeassistant.core import HomeAssistant
from . import utils
from .const import DOMAIN
_LOGGER = logging.getLogger(__name__)
DATA_SCHEMA = vol.Schema(
{
vol.Optional("name"): str,
vol.Required("host"): str,
vol.Required("mac"): str,
vol.Required("dualboot"): bool,
vol.Required("username"): str,
vol.Required("password"): str,
vol.Optional("port", default=22): int,
}
)
class Hub:
"""Dummy for test connection"""
def __init__(self, hass: HomeAssistant, host: str, username: str, password: str, port: int) -> None:
"""Init dummy hub."""
self._host = host
self._username = username
self._password = password
self._port = port
self._hass = hass
self._name = host
self._id = host.lower()
@property
def hub_id(self) -> str:
"""ID for dummy."""
return self._id
async def test_connection(self) -> bool:
"""Test connectivity to the computer is OK."""
try:
2023-04-04 13:50:00 +02:00
return utils.test_connection(
utils.create_ssh_connection(self._host, self._username, self._password, self._port))
except AuthenticationException:
return False
async def validate_input(hass: HomeAssistant, data: dict) -> dict[str, Any]:
"""Validate the user input allows us to connect.
Data has the keys from DATA_SCHEMA with values provided by the user.
"""
if len(data["host"]) < 3:
raise InvalidHost
hub = Hub(hass, data["host"], data["username"], data["password"], data["port"])
result = await hub.test_connection()
if not result:
raise CannotConnect
return {"title": data["host"]}
class ConfigFlow(config_entries.ConfigFlow, domain=DOMAIN):
"""Handle a config flow"""
VERSION = 1
CONNECTION_CLASS = config_entries.CONN_CLASS_LOCAL_PUSH
async def async_step_user(self, user_input=None):
errors = {}
if user_input is not None:
try:
info = await validate_input(self.hass, user_input)
return self.async_create_entry(title=info["title"], data=user_input)
except AuthenticationException:
errors["host"] = "cannot_connect"
except CannotConnect:
errors["base"] = "cannot_connect"
except InvalidHost:
errors["host"] = "cannot_connect"
except Exception: # pylint: disable=broad-except
_LOGGER.exception("Unexpected exception")
errors["base"] = "unknown"
# If there is no user input or there were errors, show the form again, including any errors that were found
# with the input.
return self.async_show_form(
step_id="user", data_schema=DATA_SCHEMA, errors=errors
)
class CannotConnect(exceptions.HomeAssistantError):
"""Error to indicate we cannot connect."""
class InvalidHost(exceptions.HomeAssistantError):
"""Error to indicate there is an invalid hostname."""