2020-01-10 02:19:10 +00:00
|
|
|
"""Config flow for Samsung TV."""
|
2021-09-18 04:51:07 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2022-06-02 12:28:14 +00:00
|
|
|
from collections.abc import Mapping
|
2021-11-24 02:16:09 +00:00
|
|
|
from functools import partial
|
2020-01-10 02:19:10 +00:00
|
|
|
import socket
|
2021-09-18 04:51:07 +00:00
|
|
|
from typing import Any
|
2020-01-10 02:19:10 +00:00
|
|
|
from urllib.parse import urlparse
|
|
|
|
|
2021-07-26 14:43:05 +00:00
|
|
|
import getmac
|
2022-03-22 10:11:41 +00:00
|
|
|
from samsungtvws.encrypted.authenticator import SamsungTVEncryptedWSAsyncAuthenticator
|
2020-01-10 02:19:10 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2022-06-29 09:46:59 +00:00
|
|
|
from homeassistant import config_entries
|
2021-11-29 16:10:07 +00:00
|
|
|
from homeassistant.components import dhcp, ssdp, zeroconf
|
2020-01-10 02:19:10 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
CONF_HOST,
|
2021-05-22 15:41:18 +00:00
|
|
|
CONF_MAC,
|
2020-01-10 02:19:10 +00:00
|
|
|
CONF_METHOD,
|
2022-03-29 08:01:10 +00:00
|
|
|
CONF_MODEL,
|
2020-01-10 02:19:10 +00:00
|
|
|
CONF_NAME,
|
|
|
|
CONF_PORT,
|
2020-03-10 10:48:09 +00:00
|
|
|
CONF_TOKEN,
|
2020-01-10 02:19:10 +00:00
|
|
|
)
|
2021-05-22 15:41:18 +00:00
|
|
|
from homeassistant.core import callback
|
2022-06-29 09:46:59 +00:00
|
|
|
from homeassistant.data_entry_flow import AbortFlow, FlowResult
|
2022-03-22 10:11:41 +00:00
|
|
|
from homeassistant.helpers.aiohttp_client import async_get_clientsession
|
2021-05-22 15:41:18 +00:00
|
|
|
from homeassistant.helpers.device_registry import format_mac
|
2020-01-10 02:19:10 +00:00
|
|
|
|
2022-03-10 13:48:30 +00:00
|
|
|
from .bridge import SamsungTVBridge, async_get_device_info, mac_from_device_info
|
2020-03-10 10:48:09 +00:00
|
|
|
from .const import (
|
|
|
|
CONF_MANUFACTURER,
|
2022-03-22 10:11:41 +00:00
|
|
|
CONF_SESSION_ID,
|
2022-03-28 08:01:07 +00:00
|
|
|
CONF_SSDP_MAIN_TV_AGENT_LOCATION,
|
2022-03-27 20:30:45 +00:00
|
|
|
CONF_SSDP_RENDERING_CONTROL_LOCATION,
|
2021-05-22 15:41:18 +00:00
|
|
|
DEFAULT_MANUFACTURER,
|
2020-03-10 10:48:09 +00:00
|
|
|
DOMAIN,
|
2022-03-22 10:11:41 +00:00
|
|
|
ENCRYPTED_WEBSOCKET_PORT,
|
2021-05-22 15:41:18 +00:00
|
|
|
LEGACY_PORT,
|
2020-03-10 10:48:09 +00:00
|
|
|
LOGGER,
|
2022-03-22 10:11:41 +00:00
|
|
|
METHOD_ENCRYPTED_WEBSOCKET,
|
2020-03-10 10:48:09 +00:00
|
|
|
METHOD_LEGACY,
|
|
|
|
METHOD_WEBSOCKET,
|
|
|
|
RESULT_AUTH_MISSING,
|
2020-10-05 09:45:35 +00:00
|
|
|
RESULT_CANNOT_CONNECT,
|
2022-03-22 10:11:41 +00:00
|
|
|
RESULT_INVALID_PIN,
|
2021-05-22 15:41:18 +00:00
|
|
|
RESULT_NOT_SUPPORTED,
|
2020-03-10 10:48:09 +00:00
|
|
|
RESULT_SUCCESS,
|
2021-05-22 15:41:18 +00:00
|
|
|
RESULT_UNKNOWN_HOST,
|
2022-03-27 20:30:45 +00:00
|
|
|
SUCCESSFUL_RESULTS,
|
2022-03-28 08:01:07 +00:00
|
|
|
UPNP_SVC_MAIN_TV_AGENT,
|
|
|
|
UPNP_SVC_RENDERING_CONTROL,
|
2021-05-22 15:41:18 +00:00
|
|
|
WEBSOCKET_PORTS,
|
2020-03-10 10:48:09 +00:00
|
|
|
)
|
2020-01-10 02:19:10 +00:00
|
|
|
|
|
|
|
DATA_SCHEMA = vol.Schema({vol.Required(CONF_HOST): str, vol.Required(CONF_NAME): str})
|
2020-02-08 11:03:35 +00:00
|
|
|
|
2020-01-10 02:19:10 +00:00
|
|
|
|
2021-09-18 04:51:07 +00:00
|
|
|
def _strip_uuid(udn: str) -> str:
|
2021-05-22 15:41:18 +00:00
|
|
|
return udn[5:] if udn.startswith("uuid:") else udn
|
2020-01-10 02:19:10 +00:00
|
|
|
|
|
|
|
|
2022-03-27 20:30:45 +00:00
|
|
|
def _entry_is_complete(
|
2022-03-28 08:01:07 +00:00
|
|
|
entry: config_entries.ConfigEntry,
|
|
|
|
ssdp_rendering_control_location: str | None,
|
|
|
|
ssdp_main_tv_agent_location: str | None,
|
2022-03-27 20:30:45 +00:00
|
|
|
) -> bool:
|
|
|
|
"""Return True if the config entry information is complete.
|
|
|
|
|
|
|
|
If we do not have an ssdp location we consider it complete
|
|
|
|
as some TVs will not support SSDP/UPNP
|
|
|
|
"""
|
|
|
|
return bool(
|
|
|
|
entry.unique_id
|
|
|
|
and entry.data.get(CONF_MAC)
|
|
|
|
and (
|
|
|
|
not ssdp_rendering_control_location
|
|
|
|
or entry.data.get(CONF_SSDP_RENDERING_CONTROL_LOCATION)
|
|
|
|
)
|
2022-03-28 08:01:07 +00:00
|
|
|
and (
|
|
|
|
not ssdp_main_tv_agent_location
|
|
|
|
or entry.data.get(CONF_SSDP_MAIN_TV_AGENT_LOCATION)
|
|
|
|
)
|
2022-03-27 20:30:45 +00:00
|
|
|
)
|
2021-06-25 20:31:33 +00:00
|
|
|
|
|
|
|
|
2020-01-10 02:19:10 +00:00
|
|
|
class SamsungTVConfigFlow(config_entries.ConfigFlow, domain=DOMAIN):
|
|
|
|
"""Handle a Samsung TV config flow."""
|
|
|
|
|
2021-05-22 15:41:18 +00:00
|
|
|
VERSION = 2
|
2020-01-10 02:19:10 +00:00
|
|
|
|
2021-09-18 04:51:07 +00:00
|
|
|
def __init__(self) -> None:
|
2020-01-10 02:19:10 +00:00
|
|
|
"""Initialize flow."""
|
2021-09-18 04:51:07 +00:00
|
|
|
self._reauth_entry: config_entries.ConfigEntry | None = None
|
|
|
|
self._host: str = ""
|
|
|
|
self._mac: str | None = None
|
|
|
|
self._udn: str | None = None
|
2022-03-08 06:53:59 +00:00
|
|
|
self._upnp_udn: str | None = None
|
2022-03-27 20:30:45 +00:00
|
|
|
self._ssdp_rendering_control_location: str | None = None
|
2022-03-28 08:01:07 +00:00
|
|
|
self._ssdp_main_tv_agent_location: str | None = None
|
2021-09-18 04:51:07 +00:00
|
|
|
self._manufacturer: str | None = None
|
|
|
|
self._model: str | None = None
|
2022-03-27 20:30:45 +00:00
|
|
|
self._connect_result: str | None = None
|
|
|
|
self._method: str | None = None
|
2021-09-18 04:51:07 +00:00
|
|
|
self._name: str | None = None
|
|
|
|
self._title: str = ""
|
|
|
|
self._id: int | None = None
|
2022-03-10 13:48:30 +00:00
|
|
|
self._bridge: SamsungTVBridge | None = None
|
2021-09-18 04:51:07 +00:00
|
|
|
self._device_info: dict[str, Any] | None = None
|
2022-03-27 20:30:45 +00:00
|
|
|
self._authenticator: SamsungTVEncryptedWSAsyncAuthenticator | None = None
|
2021-09-18 04:51:07 +00:00
|
|
|
|
2022-03-27 20:30:45 +00:00
|
|
|
def _base_config_entry(self) -> dict[str, Any]:
|
|
|
|
"""Generate the base config entry without the method."""
|
|
|
|
assert self._bridge is not None
|
|
|
|
return {
|
2020-03-10 10:48:09 +00:00
|
|
|
CONF_HOST: self._host,
|
2021-05-22 15:41:18 +00:00
|
|
|
CONF_MAC: self._mac,
|
|
|
|
CONF_MANUFACTURER: self._manufacturer or DEFAULT_MANUFACTURER,
|
2020-03-10 10:48:09 +00:00
|
|
|
CONF_METHOD: self._bridge.method,
|
|
|
|
CONF_MODEL: self._model,
|
|
|
|
CONF_NAME: self._name,
|
|
|
|
CONF_PORT: self._bridge.port,
|
2022-03-27 20:30:45 +00:00
|
|
|
CONF_SSDP_RENDERING_CONTROL_LOCATION: self._ssdp_rendering_control_location,
|
2022-03-28 08:01:07 +00:00
|
|
|
CONF_SSDP_MAIN_TV_AGENT_LOCATION: self._ssdp_main_tv_agent_location,
|
2020-03-10 10:48:09 +00:00
|
|
|
}
|
2022-03-27 20:30:45 +00:00
|
|
|
|
2022-06-29 09:46:59 +00:00
|
|
|
def _get_entry_from_bridge(self) -> FlowResult:
|
2022-03-27 20:30:45 +00:00
|
|
|
"""Get device entry."""
|
|
|
|
assert self._bridge
|
|
|
|
data = self._base_config_entry()
|
2020-03-10 10:48:09 +00:00
|
|
|
if self._bridge.token:
|
|
|
|
data[CONF_TOKEN] = self._bridge.token
|
2020-08-27 11:56:20 +00:00
|
|
|
return self.async_create_entry(
|
|
|
|
title=self._title,
|
|
|
|
data=data,
|
|
|
|
)
|
2020-01-10 02:19:10 +00:00
|
|
|
|
2021-09-18 04:51:07 +00:00
|
|
|
async def _async_set_device_unique_id(self, raise_on_progress: bool = True) -> None:
|
2021-05-22 15:41:18 +00:00
|
|
|
"""Set device unique_id."""
|
2021-06-24 18:15:16 +00:00
|
|
|
if not await self._async_get_and_check_device_info():
|
2022-06-29 09:46:59 +00:00
|
|
|
raise AbortFlow(RESULT_NOT_SUPPORTED)
|
2021-05-22 15:41:18 +00:00
|
|
|
await self._async_set_unique_id_from_udn(raise_on_progress)
|
2021-06-25 20:31:33 +00:00
|
|
|
self._async_update_and_abort_for_matching_unique_id()
|
2021-05-22 15:41:18 +00:00
|
|
|
|
2021-09-18 04:51:07 +00:00
|
|
|
async def _async_set_unique_id_from_udn(
|
|
|
|
self, raise_on_progress: bool = True
|
|
|
|
) -> None:
|
2021-05-22 15:41:18 +00:00
|
|
|
"""Set the unique id from the udn."""
|
2021-06-05 22:02:36 +00:00
|
|
|
assert self._host is not None
|
2022-03-27 20:30:45 +00:00
|
|
|
# Set the unique id without raising on progress in case
|
|
|
|
# there are two SSDP flows with for each ST
|
|
|
|
await self.async_set_unique_id(self._udn, raise_on_progress=False)
|
2022-03-08 07:30:11 +00:00
|
|
|
if (
|
|
|
|
entry := self._async_update_existing_matching_entry()
|
2022-03-28 08:01:07 +00:00
|
|
|
) and _entry_is_complete(
|
|
|
|
entry,
|
|
|
|
self._ssdp_rendering_control_location,
|
|
|
|
self._ssdp_main_tv_agent_location,
|
|
|
|
):
|
2022-06-29 09:46:59 +00:00
|
|
|
raise AbortFlow("already_configured")
|
2022-03-27 20:30:45 +00:00
|
|
|
# Now that we have updated the config entry, we can raise
|
|
|
|
# if another one is progressing
|
|
|
|
if raise_on_progress:
|
|
|
|
await self.async_set_unique_id(self._udn, raise_on_progress=True)
|
2021-06-25 20:31:33 +00:00
|
|
|
|
2021-09-18 04:51:07 +00:00
|
|
|
def _async_update_and_abort_for_matching_unique_id(self) -> None:
|
2021-06-25 20:31:33 +00:00
|
|
|
"""Abort and update host and mac if we have it."""
|
2021-05-22 15:41:18 +00:00
|
|
|
updates = {CONF_HOST: self._host}
|
|
|
|
if self._mac:
|
|
|
|
updates[CONF_MAC] = self._mac
|
2022-05-17 19:49:19 +00:00
|
|
|
if self._model:
|
|
|
|
updates[CONF_MODEL] = self._model
|
2022-03-27 20:30:45 +00:00
|
|
|
if self._ssdp_rendering_control_location:
|
|
|
|
updates[
|
|
|
|
CONF_SSDP_RENDERING_CONTROL_LOCATION
|
|
|
|
] = self._ssdp_rendering_control_location
|
2022-03-28 08:01:07 +00:00
|
|
|
if self._ssdp_main_tv_agent_location:
|
|
|
|
updates[
|
|
|
|
CONF_SSDP_MAIN_TV_AGENT_LOCATION
|
|
|
|
] = self._ssdp_main_tv_agent_location
|
|
|
|
self._abort_if_unique_id_configured(updates=updates, reload_on_update=False)
|
2021-05-22 15:41:18 +00:00
|
|
|
|
2022-03-27 20:30:45 +00:00
|
|
|
async def _async_create_bridge(self) -> None:
|
|
|
|
"""Create the bridge."""
|
|
|
|
result, method, _info = await self._async_get_device_info_and_method()
|
|
|
|
if result not in SUCCESSFUL_RESULTS:
|
|
|
|
LOGGER.debug("No working config found for %s", self._host)
|
2022-06-29 09:46:59 +00:00
|
|
|
raise AbortFlow(result)
|
2022-03-27 20:30:45 +00:00
|
|
|
assert method is not None
|
|
|
|
self._bridge = SamsungTVBridge.get_bridge(self.hass, method, self._host)
|
|
|
|
return
|
|
|
|
|
|
|
|
async def _async_get_device_info_and_method(
|
|
|
|
self,
|
|
|
|
) -> tuple[str, str | None, dict[str, Any] | None]:
|
|
|
|
"""Get device info and method only once."""
|
|
|
|
if self._connect_result is None:
|
|
|
|
result, _, method, info = await async_get_device_info(self.hass, self._host)
|
|
|
|
self._connect_result = result
|
|
|
|
self._method = method
|
|
|
|
self._device_info = info
|
|
|
|
if not method:
|
|
|
|
LOGGER.debug("Host:%s did not return device info", self._host)
|
|
|
|
return result, None, None
|
|
|
|
return self._connect_result, self._method, self._device_info
|
2021-05-22 15:41:18 +00:00
|
|
|
|
2021-09-18 04:51:07 +00:00
|
|
|
async def _async_get_and_check_device_info(self) -> bool:
|
2021-05-22 15:41:18 +00:00
|
|
|
"""Try to get the device info."""
|
2022-03-27 20:30:45 +00:00
|
|
|
result, _method, info = await self._async_get_device_info_and_method()
|
|
|
|
if result not in SUCCESSFUL_RESULTS:
|
2022-06-29 09:46:59 +00:00
|
|
|
raise AbortFlow(result)
|
2021-05-22 15:41:18 +00:00
|
|
|
if not info:
|
2021-06-24 18:15:16 +00:00
|
|
|
return False
|
2021-05-22 15:41:18 +00:00
|
|
|
dev_info = info.get("device", {})
|
2022-03-27 20:30:45 +00:00
|
|
|
assert dev_info is not None
|
2021-10-22 09:13:05 +00:00
|
|
|
if (device_type := dev_info.get("type")) != "Samsung SmartTV":
|
2022-03-27 20:30:45 +00:00
|
|
|
LOGGER.debug(
|
|
|
|
"Host:%s has type: %s which is not supported", self._host, device_type
|
|
|
|
)
|
2022-06-29 09:46:59 +00:00
|
|
|
raise AbortFlow(RESULT_NOT_SUPPORTED)
|
2021-05-22 15:41:18 +00:00
|
|
|
self._model = dev_info.get("modelName")
|
|
|
|
name = dev_info.get("name")
|
|
|
|
self._name = name.replace("[TV] ", "") if name else device_type
|
|
|
|
self._title = f"{self._name} ({self._model})"
|
|
|
|
self._udn = _strip_uuid(dev_info.get("udn", info["id"]))
|
2021-06-24 18:15:16 +00:00
|
|
|
if mac := mac_from_device_info(info):
|
|
|
|
self._mac = mac
|
2021-11-24 02:16:09 +00:00
|
|
|
elif mac := await self.hass.async_add_executor_job(
|
|
|
|
partial(getmac.get_mac_address, ip=self._host)
|
|
|
|
):
|
2021-07-26 14:43:05 +00:00
|
|
|
self._mac = mac
|
2021-06-24 18:15:16 +00:00
|
|
|
return True
|
2020-01-10 02:19:10 +00:00
|
|
|
|
2022-06-29 09:46:59 +00:00
|
|
|
async def async_step_import(self, user_input: dict[str, Any]) -> FlowResult:
|
2020-01-10 02:19:10 +00:00
|
|
|
"""Handle configuration by yaml file."""
|
2021-05-22 15:41:18 +00:00
|
|
|
# We need to import even if we cannot validate
|
|
|
|
# since the TV may be off at startup
|
|
|
|
await self._async_set_name_host_from_input(user_input)
|
|
|
|
self._async_abort_entries_match({CONF_HOST: self._host})
|
2021-06-24 18:15:16 +00:00
|
|
|
port = user_input.get(CONF_PORT)
|
|
|
|
if port in WEBSOCKET_PORTS:
|
2021-05-22 15:41:18 +00:00
|
|
|
user_input[CONF_METHOD] = METHOD_WEBSOCKET
|
2022-03-22 10:11:41 +00:00
|
|
|
elif port == ENCRYPTED_WEBSOCKET_PORT:
|
|
|
|
user_input[CONF_METHOD] = METHOD_ENCRYPTED_WEBSOCKET
|
2021-06-24 18:15:16 +00:00
|
|
|
elif port == LEGACY_PORT:
|
2021-05-22 15:41:18 +00:00
|
|
|
user_input[CONF_METHOD] = METHOD_LEGACY
|
|
|
|
user_input[CONF_MANUFACTURER] = DEFAULT_MANUFACTURER
|
|
|
|
return self.async_create_entry(
|
|
|
|
title=self._title,
|
|
|
|
data=user_input,
|
|
|
|
)
|
|
|
|
|
2021-09-18 04:51:07 +00:00
|
|
|
async def _async_set_name_host_from_input(self, user_input: dict[str, Any]) -> None:
|
2021-05-22 15:41:18 +00:00
|
|
|
try:
|
|
|
|
self._host = await self.hass.async_add_executor_job(
|
|
|
|
socket.gethostbyname, user_input[CONF_HOST]
|
|
|
|
)
|
|
|
|
except socket.gaierror as err:
|
2022-06-29 09:46:59 +00:00
|
|
|
raise AbortFlow(RESULT_UNKNOWN_HOST) from err
|
2021-09-18 04:51:07 +00:00
|
|
|
self._name = user_input.get(CONF_NAME, self._host) or ""
|
2021-05-22 15:41:18 +00:00
|
|
|
self._title = self._name
|
2020-01-10 02:19:10 +00:00
|
|
|
|
2021-09-18 04:51:07 +00:00
|
|
|
async def async_step_user(
|
|
|
|
self, user_input: dict[str, Any] | None = None
|
2022-06-29 09:46:59 +00:00
|
|
|
) -> FlowResult:
|
2020-01-10 02:19:10 +00:00
|
|
|
"""Handle a flow initialized by the user."""
|
|
|
|
if user_input is not None:
|
2021-05-22 15:41:18 +00:00
|
|
|
await self._async_set_name_host_from_input(user_input)
|
2022-03-27 20:30:45 +00:00
|
|
|
await self._async_create_bridge()
|
2021-09-18 04:51:07 +00:00
|
|
|
assert self._bridge
|
2021-05-22 15:41:18 +00:00
|
|
|
self._async_abort_entries_match({CONF_HOST: self._host})
|
|
|
|
if self._bridge.method != METHOD_LEGACY:
|
|
|
|
# Legacy bridge does not provide device info
|
|
|
|
await self._async_set_device_unique_id(raise_on_progress=False)
|
2022-03-27 20:30:45 +00:00
|
|
|
if self._bridge.method == METHOD_ENCRYPTED_WEBSOCKET:
|
|
|
|
return await self.async_step_encrypted_pairing()
|
|
|
|
return await self.async_step_pairing({})
|
2020-01-10 02:19:10 +00:00
|
|
|
|
2021-05-22 15:41:18 +00:00
|
|
|
return self.async_show_form(step_id="user", data_schema=DATA_SCHEMA)
|
2020-01-10 02:19:10 +00:00
|
|
|
|
2022-03-27 20:30:45 +00:00
|
|
|
async def async_step_pairing(
|
|
|
|
self, user_input: dict[str, Any] | None = None
|
2022-06-29 09:46:59 +00:00
|
|
|
) -> FlowResult:
|
2022-03-27 20:30:45 +00:00
|
|
|
"""Handle a pairing by accepting the message on the TV."""
|
|
|
|
assert self._bridge is not None
|
|
|
|
errors: dict[str, str] = {}
|
|
|
|
if user_input is not None:
|
|
|
|
result = await self._bridge.async_try_connect()
|
|
|
|
if result == RESULT_SUCCESS:
|
|
|
|
return self._get_entry_from_bridge()
|
|
|
|
if result != RESULT_AUTH_MISSING:
|
2022-06-29 09:46:59 +00:00
|
|
|
raise AbortFlow(result)
|
2022-03-27 20:30:45 +00:00
|
|
|
errors = {"base": RESULT_AUTH_MISSING}
|
|
|
|
|
|
|
|
self.context["title_placeholders"] = {"device": self._title}
|
|
|
|
return self.async_show_form(
|
|
|
|
step_id="pairing",
|
|
|
|
errors=errors,
|
|
|
|
description_placeholders={"device": self._title},
|
|
|
|
data_schema=vol.Schema({}),
|
|
|
|
)
|
|
|
|
|
|
|
|
async def async_step_encrypted_pairing(
|
|
|
|
self, user_input: dict[str, Any] | None = None
|
2022-06-29 09:46:59 +00:00
|
|
|
) -> FlowResult:
|
2022-03-27 20:30:45 +00:00
|
|
|
"""Handle a encrypted pairing."""
|
|
|
|
assert self._host is not None
|
|
|
|
await self._async_start_encrypted_pairing(self._host)
|
|
|
|
assert self._authenticator is not None
|
|
|
|
errors: dict[str, str] = {}
|
|
|
|
|
|
|
|
if user_input is not None:
|
|
|
|
if (
|
|
|
|
(pin := user_input.get("pin"))
|
|
|
|
and (token := await self._authenticator.try_pin(pin))
|
|
|
|
and (session_id := await self._authenticator.get_session_id_and_close())
|
|
|
|
):
|
|
|
|
return self.async_create_entry(
|
|
|
|
data={
|
|
|
|
**self._base_config_entry(),
|
|
|
|
CONF_TOKEN: token,
|
|
|
|
CONF_SESSION_ID: session_id,
|
|
|
|
},
|
|
|
|
title=self._title,
|
|
|
|
)
|
|
|
|
errors = {"base": RESULT_INVALID_PIN}
|
|
|
|
|
|
|
|
self.context["title_placeholders"] = {"device": self._title}
|
|
|
|
return self.async_show_form(
|
|
|
|
step_id="encrypted_pairing",
|
|
|
|
errors=errors,
|
|
|
|
description_placeholders={"device": self._title},
|
|
|
|
data_schema=vol.Schema({vol.Required("pin"): str}),
|
|
|
|
)
|
|
|
|
|
2021-05-22 15:41:18 +00:00
|
|
|
@callback
|
2022-03-08 21:27:53 +00:00
|
|
|
def _async_get_existing_matching_entry(
|
|
|
|
self,
|
|
|
|
) -> tuple[config_entries.ConfigEntry | None, bool]:
|
|
|
|
"""Get first existing matching entry (prefer unique id)."""
|
|
|
|
matching_host_entry: config_entries.ConfigEntry | None = None
|
2021-05-22 15:41:18 +00:00
|
|
|
for entry in self._async_current_entries(include_ignore=False):
|
2022-03-08 21:27:53 +00:00
|
|
|
if (self._mac and self._mac == entry.data.get(CONF_MAC)) or (
|
|
|
|
self._upnp_udn and self._upnp_udn == entry.unique_id
|
|
|
|
):
|
|
|
|
LOGGER.debug("Found entry matching unique_id for %s", self._host)
|
|
|
|
return entry, True
|
|
|
|
|
|
|
|
if entry.data[CONF_HOST] == self._host:
|
|
|
|
LOGGER.debug("Found entry matching host for %s", self._host)
|
|
|
|
matching_host_entry = entry
|
|
|
|
|
|
|
|
return matching_host_entry, False
|
2022-03-08 07:30:11 +00:00
|
|
|
|
|
|
|
@callback
|
|
|
|
def _async_update_existing_matching_entry(
|
|
|
|
self,
|
|
|
|
) -> config_entries.ConfigEntry | None:
|
|
|
|
"""Check existing entries and update them.
|
|
|
|
|
|
|
|
Returns the existing entry if it was updated.
|
|
|
|
"""
|
2022-03-08 21:27:53 +00:00
|
|
|
entry, is_unique_match = self._async_get_existing_matching_entry()
|
2022-03-28 08:01:07 +00:00
|
|
|
if not entry:
|
|
|
|
return None
|
|
|
|
entry_kw_args: dict = {}
|
2022-04-12 09:37:05 +00:00
|
|
|
if self.unique_id and (
|
|
|
|
entry.unique_id is None
|
2022-03-28 08:01:07 +00:00
|
|
|
or (is_unique_match and self.unique_id != entry.unique_id)
|
|
|
|
):
|
|
|
|
entry_kw_args["unique_id"] = self.unique_id
|
|
|
|
data: dict[str, Any] = dict(entry.data)
|
|
|
|
update_ssdp_rendering_control_location = (
|
|
|
|
self._ssdp_rendering_control_location
|
|
|
|
and data.get(CONF_SSDP_RENDERING_CONTROL_LOCATION)
|
|
|
|
!= self._ssdp_rendering_control_location
|
|
|
|
)
|
|
|
|
update_ssdp_main_tv_agent_location = (
|
|
|
|
self._ssdp_main_tv_agent_location
|
|
|
|
and data.get(CONF_SSDP_MAIN_TV_AGENT_LOCATION)
|
|
|
|
!= self._ssdp_main_tv_agent_location
|
|
|
|
)
|
|
|
|
update_mac = self._mac and not data.get(CONF_MAC)
|
2022-05-17 19:49:19 +00:00
|
|
|
update_model = self._model and not data.get(CONF_MODEL)
|
2022-03-28 08:01:07 +00:00
|
|
|
if (
|
|
|
|
update_ssdp_rendering_control_location
|
|
|
|
or update_ssdp_main_tv_agent_location
|
|
|
|
or update_mac
|
2022-05-17 19:49:19 +00:00
|
|
|
or update_model
|
2022-03-28 08:01:07 +00:00
|
|
|
):
|
|
|
|
if update_ssdp_rendering_control_location:
|
|
|
|
data[
|
|
|
|
CONF_SSDP_RENDERING_CONTROL_LOCATION
|
|
|
|
] = self._ssdp_rendering_control_location
|
|
|
|
if update_ssdp_main_tv_agent_location:
|
|
|
|
data[
|
|
|
|
CONF_SSDP_MAIN_TV_AGENT_LOCATION
|
|
|
|
] = self._ssdp_main_tv_agent_location
|
|
|
|
if update_mac:
|
|
|
|
data[CONF_MAC] = self._mac
|
2022-05-17 19:49:19 +00:00
|
|
|
if update_model:
|
|
|
|
data[CONF_MODEL] = self._model
|
2022-03-28 08:01:07 +00:00
|
|
|
entry_kw_args["data"] = data
|
|
|
|
if not entry_kw_args:
|
|
|
|
return None
|
|
|
|
LOGGER.debug("Updating existing config entry with %s", entry_kw_args)
|
|
|
|
self.hass.config_entries.async_update_entry(entry, **entry_kw_args)
|
|
|
|
if entry.state != config_entries.ConfigEntryState.LOADED:
|
|
|
|
# If its loaded it already has a reload listener in place
|
|
|
|
# and we do not want to trigger multiple reloads
|
|
|
|
self.hass.async_create_task(
|
|
|
|
self.hass.config_entries.async_reload(entry.entry_id)
|
2022-03-27 20:30:45 +00:00
|
|
|
)
|
2022-03-28 08:01:07 +00:00
|
|
|
return entry
|
2020-01-10 02:19:10 +00:00
|
|
|
|
2022-05-18 06:01:28 +00:00
|
|
|
@callback
|
|
|
|
def _async_start_discovery_with_mac_address(self) -> None:
|
2021-06-05 22:02:36 +00:00
|
|
|
"""Start discovery."""
|
|
|
|
assert self._host is not None
|
2022-03-08 07:30:11 +00:00
|
|
|
if (entry := self._async_update_existing_matching_entry()) and entry.unique_id:
|
2021-06-25 20:31:33 +00:00
|
|
|
# If we have the unique id and the mac we abort
|
|
|
|
# as we do not need anything else
|
2022-06-29 09:46:59 +00:00
|
|
|
raise AbortFlow("already_configured")
|
2021-06-25 20:31:33 +00:00
|
|
|
self._async_abort_if_host_already_in_progress()
|
2020-01-10 02:19:10 +00:00
|
|
|
|
2021-06-25 20:31:33 +00:00
|
|
|
@callback
|
2021-09-18 04:51:07 +00:00
|
|
|
def _async_abort_if_host_already_in_progress(self) -> None:
|
2021-06-05 22:02:36 +00:00
|
|
|
self.context[CONF_HOST] = self._host
|
2021-05-22 15:41:18 +00:00
|
|
|
for progress in self._async_in_progress():
|
2021-06-05 22:02:36 +00:00
|
|
|
if progress.get("context", {}).get(CONF_HOST) == self._host:
|
2022-06-29 09:46:59 +00:00
|
|
|
raise AbortFlow("already_in_progress")
|
2020-01-10 02:19:10 +00:00
|
|
|
|
2021-06-25 20:31:33 +00:00
|
|
|
@callback
|
2021-09-18 04:51:07 +00:00
|
|
|
def _abort_if_manufacturer_is_not_samsung(self) -> None:
|
2021-06-25 20:31:33 +00:00
|
|
|
if not self._manufacturer or not self._manufacturer.lower().startswith(
|
|
|
|
"samsung"
|
|
|
|
):
|
2022-06-29 09:46:59 +00:00
|
|
|
raise AbortFlow(RESULT_NOT_SUPPORTED)
|
2021-06-25 20:31:33 +00:00
|
|
|
|
2022-06-29 09:46:59 +00:00
|
|
|
async def async_step_ssdp(self, discovery_info: ssdp.SsdpServiceInfo) -> FlowResult:
|
2021-05-22 15:41:18 +00:00
|
|
|
"""Handle a flow initialized by ssdp discovery."""
|
2021-06-04 16:02:39 +00:00
|
|
|
LOGGER.debug("Samsung device found via SSDP: %s", discovery_info)
|
2021-12-01 15:42:42 +00:00
|
|
|
model_name: str = discovery_info.upnp.get(ssdp.ATTR_UPNP_MODEL_NAME) or ""
|
2022-03-28 08:01:07 +00:00
|
|
|
if discovery_info.ssdp_st == UPNP_SVC_RENDERING_CONTROL:
|
2022-03-27 20:30:45 +00:00
|
|
|
self._ssdp_rendering_control_location = discovery_info.ssdp_location
|
|
|
|
LOGGER.debug(
|
2022-03-28 08:01:07 +00:00
|
|
|
"Set SSDP RenderingControl location to: %s",
|
|
|
|
self._ssdp_rendering_control_location,
|
|
|
|
)
|
|
|
|
elif discovery_info.ssdp_st == UPNP_SVC_MAIN_TV_AGENT:
|
|
|
|
self._ssdp_main_tv_agent_location = discovery_info.ssdp_location
|
|
|
|
LOGGER.debug(
|
|
|
|
"Set SSDP MainTvAgent location to: %s",
|
|
|
|
self._ssdp_main_tv_agent_location,
|
2022-03-27 20:30:45 +00:00
|
|
|
)
|
2022-03-08 06:53:59 +00:00
|
|
|
self._udn = self._upnp_udn = _strip_uuid(
|
|
|
|
discovery_info.upnp[ssdp.ATTR_UPNP_UDN]
|
|
|
|
)
|
2021-12-01 15:42:42 +00:00
|
|
|
if hostname := urlparse(discovery_info.ssdp_location or "").hostname:
|
2021-09-18 04:51:07 +00:00
|
|
|
self._host = hostname
|
2022-09-21 18:02:54 +00:00
|
|
|
self._manufacturer = discovery_info.upnp.get(ssdp.ATTR_UPNP_MANUFACTURER)
|
2021-06-25 20:31:33 +00:00
|
|
|
self._abort_if_manufacturer_is_not_samsung()
|
2022-03-08 07:54:03 +00:00
|
|
|
|
|
|
|
# Set defaults, in case they cannot be extracted from device_info
|
|
|
|
self._name = self._title = self._model = model_name
|
|
|
|
# Update from device_info (if accessible)
|
|
|
|
await self._async_get_and_check_device_info()
|
2022-03-08 06:53:59 +00:00
|
|
|
|
|
|
|
# The UDN provided by the ssdp discovery doesn't always match the UDN
|
|
|
|
# from the device_info, used by the the other methods so we need to
|
|
|
|
# ensure the device_info is loaded before setting the unique_id
|
|
|
|
await self._async_set_unique_id_from_udn()
|
2021-06-25 20:31:33 +00:00
|
|
|
self._async_update_and_abort_for_matching_unique_id()
|
|
|
|
self._async_abort_if_host_already_in_progress()
|
2022-04-05 23:38:55 +00:00
|
|
|
if self._method == METHOD_LEGACY and discovery_info.ssdp_st in (
|
|
|
|
UPNP_SVC_RENDERING_CONTROL,
|
|
|
|
UPNP_SVC_MAIN_TV_AGENT,
|
|
|
|
):
|
|
|
|
# The UDN we use for the unique id cannot be determined
|
|
|
|
# from device_info for legacy devices
|
|
|
|
return self.async_abort(reason="not_supported")
|
2021-05-22 15:41:18 +00:00
|
|
|
self.context["title_placeholders"] = {"device": self._title}
|
|
|
|
return await self.async_step_confirm()
|
|
|
|
|
2022-06-29 09:46:59 +00:00
|
|
|
async def async_step_dhcp(self, discovery_info: dhcp.DhcpServiceInfo) -> FlowResult:
|
2021-05-22 15:41:18 +00:00
|
|
|
"""Handle a flow initialized by dhcp discovery."""
|
2021-06-04 16:02:39 +00:00
|
|
|
LOGGER.debug("Samsung device found via DHCP: %s", discovery_info)
|
2021-12-01 15:42:42 +00:00
|
|
|
self._mac = discovery_info.macaddress
|
|
|
|
self._host = discovery_info.ip
|
2022-05-18 06:01:28 +00:00
|
|
|
self._async_start_discovery_with_mac_address()
|
2021-05-22 15:41:18 +00:00
|
|
|
await self._async_set_device_unique_id()
|
|
|
|
self.context["title_placeholders"] = {"device": self._title}
|
|
|
|
return await self.async_step_confirm()
|
2020-01-10 02:19:10 +00:00
|
|
|
|
2021-09-18 04:51:07 +00:00
|
|
|
async def async_step_zeroconf(
|
2021-11-15 17:05:45 +00:00
|
|
|
self, discovery_info: zeroconf.ZeroconfServiceInfo
|
2022-06-29 09:46:59 +00:00
|
|
|
) -> FlowResult:
|
2021-05-22 15:41:18 +00:00
|
|
|
"""Handle a flow initialized by zeroconf discovery."""
|
2021-06-04 16:02:39 +00:00
|
|
|
LOGGER.debug("Samsung device found via ZEROCONF: %s", discovery_info)
|
2021-12-01 11:29:41 +00:00
|
|
|
self._mac = format_mac(discovery_info.properties["deviceid"])
|
|
|
|
self._host = discovery_info.host
|
2022-05-18 06:01:28 +00:00
|
|
|
self._async_start_discovery_with_mac_address()
|
2021-05-22 15:41:18 +00:00
|
|
|
await self._async_set_device_unique_id()
|
|
|
|
self.context["title_placeholders"] = {"device": self._title}
|
2020-01-10 02:19:10 +00:00
|
|
|
return await self.async_step_confirm()
|
|
|
|
|
2021-09-18 04:51:07 +00:00
|
|
|
async def async_step_confirm(
|
|
|
|
self, user_input: dict[str, Any] | None = None
|
2022-06-29 09:46:59 +00:00
|
|
|
) -> FlowResult:
|
2020-01-10 02:19:10 +00:00
|
|
|
"""Handle user-confirmation of discovered node."""
|
|
|
|
if user_input is not None:
|
2022-03-27 20:30:45 +00:00
|
|
|
await self._async_create_bridge()
|
2021-09-18 04:51:07 +00:00
|
|
|
assert self._bridge
|
2022-03-27 20:30:45 +00:00
|
|
|
if self._bridge.method == METHOD_ENCRYPTED_WEBSOCKET:
|
|
|
|
return await self.async_step_encrypted_pairing()
|
|
|
|
return await self.async_step_pairing({})
|
2020-01-10 02:19:10 +00:00
|
|
|
|
|
|
|
return self.async_show_form(
|
2021-05-22 15:41:18 +00:00
|
|
|
step_id="confirm", description_placeholders={"device": self._title}
|
2020-01-10 02:19:10 +00:00
|
|
|
)
|
2020-02-03 19:34:02 +00:00
|
|
|
|
2022-06-29 09:46:59 +00:00
|
|
|
async def async_step_reauth(self, entry_data: Mapping[str, Any]) -> FlowResult:
|
2020-02-03 19:34:02 +00:00
|
|
|
"""Handle configuration by re-auth."""
|
2021-05-22 15:41:18 +00:00
|
|
|
self._reauth_entry = self.hass.config_entries.async_get_entry(
|
|
|
|
self.context["entry_id"]
|
|
|
|
)
|
2022-06-29 09:46:59 +00:00
|
|
|
if entry_data.get(CONF_MODEL) and entry_data.get(CONF_NAME):
|
|
|
|
self._title = f"{entry_data[CONF_NAME]} ({entry_data[CONF_MODEL]})"
|
2021-05-22 15:41:18 +00:00
|
|
|
else:
|
2022-06-29 09:46:59 +00:00
|
|
|
self._title = entry_data.get(CONF_NAME) or entry_data[CONF_HOST]
|
2021-05-22 15:41:18 +00:00
|
|
|
return await self.async_step_reauth_confirm()
|
|
|
|
|
2021-09-18 04:51:07 +00:00
|
|
|
async def async_step_reauth_confirm(
|
|
|
|
self, user_input: dict[str, Any] | None = None
|
2022-06-29 09:46:59 +00:00
|
|
|
) -> FlowResult:
|
2021-05-22 15:41:18 +00:00
|
|
|
"""Confirm reauth."""
|
|
|
|
errors = {}
|
2021-09-18 04:51:07 +00:00
|
|
|
assert self._reauth_entry
|
2022-03-22 10:11:41 +00:00
|
|
|
method = self._reauth_entry.data[CONF_METHOD]
|
2021-05-22 15:41:18 +00:00
|
|
|
if user_input is not None:
|
2022-03-27 20:30:45 +00:00
|
|
|
if method == METHOD_ENCRYPTED_WEBSOCKET:
|
|
|
|
return await self.async_step_reauth_confirm_encrypted()
|
2021-05-22 15:41:18 +00:00
|
|
|
bridge = SamsungTVBridge.get_bridge(
|
2022-02-22 18:31:16 +00:00
|
|
|
self.hass,
|
2022-03-22 10:11:41 +00:00
|
|
|
method,
|
2022-02-22 18:31:16 +00:00
|
|
|
self._reauth_entry.data[CONF_HOST],
|
2021-05-22 15:41:18 +00:00
|
|
|
)
|
2022-02-22 18:31:16 +00:00
|
|
|
result = await bridge.async_try_connect()
|
2021-05-22 15:41:18 +00:00
|
|
|
if result == RESULT_SUCCESS:
|
|
|
|
new_data = dict(self._reauth_entry.data)
|
|
|
|
new_data[CONF_TOKEN] = bridge.token
|
|
|
|
self.hass.config_entries.async_update_entry(
|
|
|
|
self._reauth_entry, data=new_data
|
|
|
|
)
|
2021-06-10 17:23:00 +00:00
|
|
|
await self.hass.config_entries.async_reload(self._reauth_entry.entry_id)
|
2021-05-22 15:41:18 +00:00
|
|
|
return self.async_abort(reason="reauth_successful")
|
|
|
|
if result not in (RESULT_AUTH_MISSING, RESULT_CANNOT_CONNECT):
|
|
|
|
return self.async_abort(reason=result)
|
2020-02-03 19:34:02 +00:00
|
|
|
|
2021-05-22 15:41:18 +00:00
|
|
|
# On websocket we will get RESULT_CANNOT_CONNECT when auth is missing
|
|
|
|
errors = {"base": RESULT_AUTH_MISSING}
|
2020-02-03 19:34:02 +00:00
|
|
|
|
2022-03-22 10:11:41 +00:00
|
|
|
self.context["title_placeholders"] = {"device": self._title}
|
|
|
|
return self.async_show_form(
|
2022-03-27 20:30:45 +00:00
|
|
|
step_id="reauth_confirm",
|
2022-03-22 10:11:41 +00:00
|
|
|
errors=errors,
|
|
|
|
description_placeholders={"device": self._title},
|
|
|
|
)
|
|
|
|
|
2022-03-27 20:30:45 +00:00
|
|
|
async def _async_start_encrypted_pairing(self, host: str) -> None:
|
|
|
|
if self._authenticator is None:
|
|
|
|
self._authenticator = SamsungTVEncryptedWSAsyncAuthenticator(
|
|
|
|
host,
|
|
|
|
web_session=async_get_clientsession(self.hass),
|
|
|
|
)
|
|
|
|
await self._authenticator.start_pairing()
|
|
|
|
|
2022-03-22 10:11:41 +00:00
|
|
|
async def async_step_reauth_confirm_encrypted(
|
|
|
|
self, user_input: dict[str, Any] | None = None
|
2022-06-29 09:46:59 +00:00
|
|
|
) -> FlowResult:
|
2022-03-22 10:11:41 +00:00
|
|
|
"""Confirm reauth (encrypted method)."""
|
|
|
|
errors = {}
|
|
|
|
assert self._reauth_entry
|
2022-03-27 20:30:45 +00:00
|
|
|
await self._async_start_encrypted_pairing(self._reauth_entry.data[CONF_HOST])
|
|
|
|
assert self._authenticator is not None
|
2022-03-22 10:11:41 +00:00
|
|
|
|
2022-03-27 20:30:45 +00:00
|
|
|
if user_input is not None:
|
|
|
|
if (
|
|
|
|
(pin := user_input.get("pin"))
|
|
|
|
and (token := await self._authenticator.try_pin(pin))
|
|
|
|
and (session_id := await self._authenticator.get_session_id_and_close())
|
|
|
|
):
|
2022-03-22 10:11:41 +00:00
|
|
|
self.hass.config_entries.async_update_entry(
|
2022-03-27 20:30:45 +00:00
|
|
|
self._reauth_entry,
|
|
|
|
data={
|
|
|
|
**self._reauth_entry.data,
|
|
|
|
CONF_TOKEN: token,
|
|
|
|
CONF_SESSION_ID: session_id,
|
|
|
|
},
|
2022-03-22 10:11:41 +00:00
|
|
|
)
|
|
|
|
await self.hass.config_entries.async_reload(self._reauth_entry.entry_id)
|
|
|
|
return self.async_abort(reason="reauth_successful")
|
|
|
|
|
|
|
|
errors = {"base": RESULT_INVALID_PIN}
|
|
|
|
|
2021-05-22 15:41:18 +00:00
|
|
|
self.context["title_placeholders"] = {"device": self._title}
|
|
|
|
return self.async_show_form(
|
2022-03-22 10:11:41 +00:00
|
|
|
step_id="reauth_confirm_encrypted",
|
2021-05-22 15:41:18 +00:00
|
|
|
errors=errors,
|
|
|
|
description_placeholders={"device": self._title},
|
2022-03-22 10:11:41 +00:00
|
|
|
data_schema=vol.Schema({vol.Required("pin"): str}),
|
2021-05-22 15:41:18 +00:00
|
|
|
)
|