208 lines
5.9 KiB
Python
208 lines
5.9 KiB
Python
"""The Samsung TV integration."""
|
|
from __future__ import annotations
|
|
|
|
from functools import partial
|
|
import socket
|
|
from typing import Any
|
|
|
|
import getmac
|
|
import voluptuous as vol
|
|
|
|
from homeassistant import config_entries
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.const import (
|
|
CONF_HOST,
|
|
CONF_MAC,
|
|
CONF_METHOD,
|
|
CONF_NAME,
|
|
CONF_PORT,
|
|
CONF_TOKEN,
|
|
EVENT_HOMEASSISTANT_STOP,
|
|
Platform,
|
|
)
|
|
from homeassistant.core import Event, HomeAssistant, callback
|
|
from homeassistant.exceptions import ConfigEntryNotReady
|
|
import homeassistant.helpers.config_validation as cv
|
|
from homeassistant.helpers.typing import ConfigType
|
|
|
|
from .bridge import (
|
|
SamsungTVBridge,
|
|
SamsungTVLegacyBridge,
|
|
SamsungTVWSBridge,
|
|
async_get_device_info,
|
|
mac_from_device_info,
|
|
)
|
|
from .const import (
|
|
CONF_ON_ACTION,
|
|
DEFAULT_NAME,
|
|
DOMAIN,
|
|
LEGACY_PORT,
|
|
LOGGER,
|
|
METHOD_LEGACY,
|
|
METHOD_WEBSOCKET,
|
|
)
|
|
|
|
|
|
def ensure_unique_hosts(value: dict[Any, Any]) -> dict[Any, Any]:
|
|
"""Validate that all configs have a unique host."""
|
|
vol.Schema(vol.Unique("duplicate host entries found"))(
|
|
[entry[CONF_HOST] for entry in value]
|
|
)
|
|
return value
|
|
|
|
|
|
PLATFORMS = [Platform.MEDIA_PLAYER]
|
|
|
|
CONFIG_SCHEMA = vol.Schema(
|
|
{
|
|
DOMAIN: vol.All(
|
|
cv.ensure_list,
|
|
[
|
|
cv.deprecated(CONF_PORT),
|
|
vol.Schema(
|
|
{
|
|
vol.Required(CONF_HOST): cv.string,
|
|
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
|
|
vol.Optional(CONF_PORT): cv.port,
|
|
vol.Optional(CONF_ON_ACTION): cv.SCRIPT_SCHEMA,
|
|
}
|
|
),
|
|
],
|
|
ensure_unique_hosts,
|
|
)
|
|
},
|
|
extra=vol.ALLOW_EXTRA,
|
|
)
|
|
|
|
|
|
async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool:
|
|
"""Set up the Samsung TV integration."""
|
|
hass.data[DOMAIN] = {}
|
|
if DOMAIN not in config:
|
|
return True
|
|
|
|
for entry_config in config[DOMAIN]:
|
|
ip_address = await hass.async_add_executor_job(
|
|
socket.gethostbyname, entry_config[CONF_HOST]
|
|
)
|
|
hass.data[DOMAIN][ip_address] = {
|
|
CONF_ON_ACTION: entry_config.get(CONF_ON_ACTION)
|
|
}
|
|
hass.async_create_task(
|
|
hass.config_entries.flow.async_init(
|
|
DOMAIN,
|
|
context={"source": config_entries.SOURCE_IMPORT},
|
|
data=entry_config,
|
|
)
|
|
)
|
|
return True
|
|
|
|
|
|
@callback
|
|
def _async_get_device_bridge(
|
|
data: dict[str, Any]
|
|
) -> SamsungTVLegacyBridge | SamsungTVWSBridge:
|
|
"""Get device bridge."""
|
|
return SamsungTVBridge.get_bridge(
|
|
data[CONF_METHOD],
|
|
data[CONF_HOST],
|
|
data[CONF_PORT],
|
|
data.get(CONF_TOKEN),
|
|
)
|
|
|
|
|
|
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|
"""Set up the Samsung TV platform."""
|
|
|
|
# Initialize bridge
|
|
bridge = await _async_create_bridge_with_updated_data(hass, entry)
|
|
|
|
def stop_bridge(event: Event) -> None:
|
|
"""Stop SamsungTV bridge connection."""
|
|
bridge.stop()
|
|
|
|
entry.async_on_unload(
|
|
hass.bus.async_listen_once(EVENT_HOMEASSISTANT_STOP, stop_bridge)
|
|
)
|
|
|
|
hass.data[DOMAIN][entry.entry_id] = bridge
|
|
hass.config_entries.async_setup_platforms(entry, PLATFORMS)
|
|
return True
|
|
|
|
|
|
async def _async_create_bridge_with_updated_data(
|
|
hass: HomeAssistant, entry: ConfigEntry
|
|
) -> SamsungTVLegacyBridge | SamsungTVWSBridge:
|
|
"""Create a bridge object and update any missing data in the config entry."""
|
|
updated_data = {}
|
|
host = entry.data[CONF_HOST]
|
|
port = entry.data.get(CONF_PORT)
|
|
method = entry.data.get(CONF_METHOD)
|
|
info = None
|
|
|
|
if not port or not method:
|
|
if method == METHOD_LEGACY:
|
|
port = LEGACY_PORT
|
|
else:
|
|
# When we imported from yaml we didn't setup the method
|
|
# because we didn't know it
|
|
port, method, info = await async_get_device_info(hass, None, host)
|
|
if not port:
|
|
raise ConfigEntryNotReady(
|
|
"Failed to determine connection method, make sure the device is on."
|
|
)
|
|
|
|
updated_data[CONF_PORT] = port
|
|
updated_data[CONF_METHOD] = method
|
|
|
|
bridge = _async_get_device_bridge({**entry.data, **updated_data})
|
|
|
|
mac = entry.data.get(CONF_MAC)
|
|
if not mac and bridge.method == METHOD_WEBSOCKET:
|
|
if info:
|
|
mac = mac_from_device_info(info)
|
|
else:
|
|
mac = await hass.async_add_executor_job(bridge.mac_from_device)
|
|
|
|
if not mac:
|
|
mac = await hass.async_add_executor_job(
|
|
partial(getmac.get_mac_address, ip=host)
|
|
)
|
|
if mac:
|
|
updated_data[CONF_MAC] = mac
|
|
|
|
if updated_data:
|
|
data = {**entry.data, **updated_data}
|
|
hass.config_entries.async_update_entry(entry, data=data)
|
|
|
|
return bridge
|
|
|
|
|
|
async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|
"""Unload a config entry."""
|
|
unload_ok = await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
|
|
if unload_ok:
|
|
hass.data[DOMAIN][entry.entry_id].stop()
|
|
return unload_ok
|
|
|
|
|
|
async def async_migrate_entry(hass: HomeAssistant, config_entry: ConfigEntry) -> bool:
|
|
"""Migrate old entry."""
|
|
version = config_entry.version
|
|
|
|
LOGGER.debug("Migrating from version %s", version)
|
|
|
|
# 1 -> 2: Unique ID format changed, so delete and re-import:
|
|
if version == 1:
|
|
dev_reg = await hass.helpers.device_registry.async_get_registry()
|
|
dev_reg.async_clear_config_entry(config_entry)
|
|
|
|
en_reg = await hass.helpers.entity_registry.async_get_registry()
|
|
en_reg.async_clear_config_entry(config_entry)
|
|
|
|
version = config_entry.version = 2
|
|
hass.config_entries.async_update_entry(config_entry)
|
|
LOGGER.debug("Migration to version %s successful", version)
|
|
|
|
return True
|