2020-10-24 01:57:16 +00:00
|
|
|
"""Hub for communication with 1-Wire server or mount_dir."""
|
2021-05-11 15:28:17 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2021-10-22 09:45:40 +00:00
|
|
|
import logging
|
2020-10-24 01:57:16 +00:00
|
|
|
import os
|
2021-10-22 09:45:40 +00:00
|
|
|
from typing import TYPE_CHECKING
|
2020-10-24 01:57:16 +00:00
|
|
|
|
2020-10-27 04:36:51 +00:00
|
|
|
from pi1wire import Pi1Wire
|
2020-10-24 01:57:16 +00:00
|
|
|
from pyownet import protocol
|
|
|
|
|
2020-10-26 15:33:13 +00:00
|
|
|
from homeassistant.config_entries import ConfigEntry
|
2021-10-22 09:45:40 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
ATTR_IDENTIFIERS,
|
|
|
|
ATTR_MANUFACTURER,
|
|
|
|
ATTR_MODEL,
|
|
|
|
ATTR_NAME,
|
2021-10-22 15:04:25 +00:00
|
|
|
ATTR_VIA_DEVICE,
|
2021-10-22 09:45:40 +00:00
|
|
|
CONF_HOST,
|
|
|
|
CONF_PORT,
|
|
|
|
CONF_TYPE,
|
|
|
|
)
|
2021-04-22 18:23:19 +00:00
|
|
|
from homeassistant.core import HomeAssistant
|
2020-10-26 15:33:13 +00:00
|
|
|
from homeassistant.exceptions import HomeAssistantError
|
2021-10-22 09:45:40 +00:00
|
|
|
from homeassistant.helpers import device_registry as dr
|
|
|
|
from homeassistant.helpers.entity import DeviceInfo
|
2020-10-24 01:57:16 +00:00
|
|
|
|
2021-10-27 10:02:07 +00:00
|
|
|
from .const import (
|
|
|
|
CONF_MOUNT_DIR,
|
|
|
|
CONF_TYPE_OWSERVER,
|
|
|
|
CONF_TYPE_SYSBUS,
|
2021-11-09 17:30:05 +00:00
|
|
|
DEVICE_SUPPORT_OWSERVER,
|
|
|
|
DEVICE_SUPPORT_SYSBUS,
|
2021-10-27 10:02:07 +00:00
|
|
|
DOMAIN,
|
|
|
|
MANUFACTURER_EDS,
|
|
|
|
MANUFACTURER_HOBBYBOARDS,
|
|
|
|
MANUFACTURER_MAXIM,
|
|
|
|
)
|
2021-10-22 09:45:40 +00:00
|
|
|
from .model import (
|
|
|
|
OWDeviceDescription,
|
|
|
|
OWDirectDeviceDescription,
|
|
|
|
OWServerDeviceDescription,
|
|
|
|
)
|
2020-10-24 01:57:16 +00:00
|
|
|
|
2020-12-07 01:09:32 +00:00
|
|
|
DEVICE_COUPLERS = {
|
|
|
|
# Family : [branches]
|
|
|
|
"1F": ["aux", "main"]
|
|
|
|
}
|
|
|
|
|
2021-10-27 10:02:07 +00:00
|
|
|
DEVICE_MANUFACTURER = {
|
|
|
|
"7E": MANUFACTURER_EDS,
|
|
|
|
"EF": MANUFACTURER_HOBBYBOARDS,
|
|
|
|
}
|
|
|
|
|
2021-10-22 09:45:40 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2020-10-24 01:57:16 +00:00
|
|
|
|
2021-11-09 17:30:05 +00:00
|
|
|
def _is_known_owserver_device(device_family: str, device_type: str) -> bool:
|
|
|
|
"""Check if device family/type is known to the library."""
|
|
|
|
if device_family in ("7E", "EF"): # EDS or HobbyBoard
|
|
|
|
return device_type in DEVICE_SUPPORT_OWSERVER[device_family]
|
|
|
|
return device_family in DEVICE_SUPPORT_OWSERVER
|
|
|
|
|
|
|
|
|
2020-10-24 01:57:16 +00:00
|
|
|
class OneWireHub:
|
|
|
|
"""Hub to communicate with SysBus or OWServer."""
|
|
|
|
|
2021-05-20 13:58:17 +00:00
|
|
|
def __init__(self, hass: HomeAssistant) -> None:
|
2020-10-24 01:57:16 +00:00
|
|
|
"""Initialize."""
|
|
|
|
self.hass = hass
|
2021-05-11 15:28:17 +00:00
|
|
|
self.type: str | None = None
|
|
|
|
self.pi1proxy: Pi1Wire | None = None
|
|
|
|
self.owproxy: protocol._Proxy | None = None
|
2021-10-22 09:45:40 +00:00
|
|
|
self.devices: list[OWDeviceDescription] | None = None
|
2020-10-24 01:57:16 +00:00
|
|
|
|
2020-10-26 15:33:13 +00:00
|
|
|
async def connect(self, host: str, port: int) -> None:
|
|
|
|
"""Connect to the owserver host."""
|
2020-10-24 01:57:16 +00:00
|
|
|
try:
|
2020-10-26 15:33:13 +00:00
|
|
|
self.owproxy = await self.hass.async_add_executor_job(
|
|
|
|
protocol.proxy, host, port
|
2020-10-24 01:57:16 +00:00
|
|
|
)
|
2020-10-26 15:33:13 +00:00
|
|
|
except protocol.ConnError as exc:
|
|
|
|
raise CannotConnect from exc
|
2020-10-24 01:57:16 +00:00
|
|
|
|
2020-10-26 15:33:13 +00:00
|
|
|
async def check_mount_dir(self, mount_dir: str) -> None:
|
2020-10-24 01:57:16 +00:00
|
|
|
"""Test that the mount_dir is a valid path."""
|
|
|
|
if not await self.hass.async_add_executor_job(os.path.isdir, mount_dir):
|
2020-10-26 15:33:13 +00:00
|
|
|
raise InvalidPath
|
2020-10-27 04:36:51 +00:00
|
|
|
self.pi1proxy = Pi1Wire(mount_dir)
|
2020-10-26 15:33:13 +00:00
|
|
|
|
|
|
|
async def initialize(self, config_entry: ConfigEntry) -> None:
|
|
|
|
"""Initialize a config entry."""
|
2020-10-27 04:36:51 +00:00
|
|
|
self.type = config_entry.data[CONF_TYPE]
|
|
|
|
if self.type == CONF_TYPE_SYSBUS:
|
2021-11-09 17:30:05 +00:00
|
|
|
mount_dir = config_entry.data[CONF_MOUNT_DIR]
|
|
|
|
_LOGGER.debug("Initializing using SysBus %s", mount_dir)
|
2022-01-06 11:09:43 +00:00
|
|
|
_LOGGER.warning(
|
|
|
|
"Using the 1-Wire integration via SysBus is deprecated and will be removed "
|
|
|
|
"in Home Assistant Core 2022.6; this integration is being adjusted to comply "
|
|
|
|
"with Architectural Decision Record 0019, more information can be found here: "
|
|
|
|
"https://github.com/home-assistant/architecture/blob/master/adr/0019-GPIO.md "
|
|
|
|
"Access via OWServer is still supported"
|
|
|
|
)
|
|
|
|
|
2021-11-09 17:30:05 +00:00
|
|
|
await self.check_mount_dir(mount_dir)
|
2020-10-27 04:36:51 +00:00
|
|
|
elif self.type == CONF_TYPE_OWSERVER:
|
2020-10-26 15:33:13 +00:00
|
|
|
host = config_entry.data[CONF_HOST]
|
|
|
|
port = config_entry.data[CONF_PORT]
|
2021-11-09 17:30:05 +00:00
|
|
|
_LOGGER.debug("Initializing using OWServer %s:%s", host, port)
|
2020-10-26 15:33:13 +00:00
|
|
|
await self.connect(host, port)
|
2020-10-27 04:36:51 +00:00
|
|
|
await self.discover_devices()
|
2021-10-22 09:45:40 +00:00
|
|
|
if TYPE_CHECKING:
|
|
|
|
assert self.devices
|
|
|
|
# Register discovered devices on Hub
|
|
|
|
device_registry = dr.async_get(self.hass)
|
|
|
|
for device in self.devices:
|
|
|
|
device_info: DeviceInfo = device.device_info
|
|
|
|
device_registry.async_get_or_create(
|
|
|
|
config_entry_id=config_entry.entry_id,
|
|
|
|
identifiers=device_info[ATTR_IDENTIFIERS],
|
|
|
|
manufacturer=device_info[ATTR_MANUFACTURER],
|
|
|
|
model=device_info[ATTR_MODEL],
|
|
|
|
name=device_info[ATTR_NAME],
|
2021-10-22 15:04:25 +00:00
|
|
|
via_device=device_info.get(ATTR_VIA_DEVICE),
|
2021-10-22 09:45:40 +00:00
|
|
|
)
|
2020-10-27 04:36:51 +00:00
|
|
|
|
2021-05-11 15:28:17 +00:00
|
|
|
async def discover_devices(self) -> None:
|
2020-10-27 04:36:51 +00:00
|
|
|
"""Discover all devices."""
|
|
|
|
if self.devices is None:
|
|
|
|
if self.type == CONF_TYPE_SYSBUS:
|
|
|
|
self.devices = await self.hass.async_add_executor_job(
|
2021-10-22 09:45:40 +00:00
|
|
|
self._discover_devices_sysbus
|
2020-10-27 04:36:51 +00:00
|
|
|
)
|
|
|
|
if self.type == CONF_TYPE_OWSERVER:
|
|
|
|
self.devices = await self.hass.async_add_executor_job(
|
|
|
|
self._discover_devices_owserver
|
|
|
|
)
|
|
|
|
|
2021-10-22 09:45:40 +00:00
|
|
|
def _discover_devices_sysbus(self) -> list[OWDeviceDescription]:
|
|
|
|
"""Discover all sysbus devices."""
|
|
|
|
devices: list[OWDeviceDescription] = []
|
|
|
|
assert self.pi1proxy
|
2021-11-09 17:30:05 +00:00
|
|
|
all_sensors = self.pi1proxy.find_all_sensors()
|
|
|
|
if not all_sensors:
|
|
|
|
_LOGGER.error(
|
|
|
|
"No onewire sensor found. Check if dtoverlay=w1-gpio "
|
|
|
|
"is in your /boot/config.txt. "
|
|
|
|
"Check the mount_dir parameter if it's defined"
|
|
|
|
)
|
|
|
|
for interface in all_sensors:
|
2021-10-27 10:02:07 +00:00
|
|
|
device_family = interface.mac_address[:2]
|
|
|
|
device_id = f"{device_family}-{interface.mac_address[2:]}"
|
2021-11-09 17:30:05 +00:00
|
|
|
if device_family not in DEVICE_SUPPORT_SYSBUS:
|
|
|
|
_LOGGER.warning(
|
|
|
|
"Ignoring unknown device family (%s) found for device %s",
|
|
|
|
device_family,
|
|
|
|
device_id,
|
|
|
|
)
|
|
|
|
continue
|
2021-10-22 09:45:40 +00:00
|
|
|
device_info: DeviceInfo = {
|
|
|
|
ATTR_IDENTIFIERS: {(DOMAIN, device_id)},
|
2021-10-27 10:02:07 +00:00
|
|
|
ATTR_MANUFACTURER: DEVICE_MANUFACTURER.get(
|
|
|
|
device_family, MANUFACTURER_MAXIM
|
|
|
|
),
|
|
|
|
ATTR_MODEL: device_family,
|
2021-10-22 09:45:40 +00:00
|
|
|
ATTR_NAME: device_id,
|
|
|
|
}
|
|
|
|
device = OWDirectDeviceDescription(
|
|
|
|
device_info=device_info,
|
|
|
|
interface=interface,
|
|
|
|
)
|
|
|
|
devices.append(device)
|
|
|
|
return devices
|
|
|
|
|
2021-05-11 15:28:17 +00:00
|
|
|
def _discover_devices_owserver(
|
2021-10-22 09:45:40 +00:00
|
|
|
self, path: str = "/", parent_id: str | None = None
|
|
|
|
) -> list[OWDeviceDescription]:
|
2020-10-27 04:36:51 +00:00
|
|
|
"""Discover all owserver devices."""
|
2021-10-22 09:45:40 +00:00
|
|
|
devices: list[OWDeviceDescription] = []
|
2021-05-11 15:28:17 +00:00
|
|
|
assert self.owproxy
|
2020-12-07 01:09:32 +00:00
|
|
|
for device_path in self.owproxy.dir(path):
|
2021-10-22 09:45:40 +00:00
|
|
|
device_id = os.path.split(os.path.split(device_path)[0])[1]
|
2020-12-07 01:09:32 +00:00
|
|
|
device_family = self.owproxy.read(f"{device_path}family").decode()
|
2021-10-22 09:45:40 +00:00
|
|
|
_LOGGER.debug("read `%sfamily`: %s", device_path, device_family)
|
2021-10-27 17:19:55 +00:00
|
|
|
device_type = self._get_device_type_owserver(device_path)
|
2021-11-09 17:30:05 +00:00
|
|
|
if not _is_known_owserver_device(device_family, device_type):
|
|
|
|
_LOGGER.warning(
|
|
|
|
"Ignoring unknown device family/type (%s/%s) found for device %s",
|
|
|
|
device_family,
|
|
|
|
device_type,
|
|
|
|
device_id,
|
|
|
|
)
|
|
|
|
continue
|
2021-10-22 09:45:40 +00:00
|
|
|
device_info: DeviceInfo = {
|
|
|
|
ATTR_IDENTIFIERS: {(DOMAIN, device_id)},
|
2021-10-27 10:02:07 +00:00
|
|
|
ATTR_MANUFACTURER: DEVICE_MANUFACTURER.get(
|
|
|
|
device_family, MANUFACTURER_MAXIM
|
|
|
|
),
|
2021-10-22 09:45:40 +00:00
|
|
|
ATTR_MODEL: device_type,
|
|
|
|
ATTR_NAME: device_id,
|
|
|
|
}
|
|
|
|
if parent_id:
|
2021-10-22 15:04:25 +00:00
|
|
|
device_info[ATTR_VIA_DEVICE] = (DOMAIN, parent_id)
|
2021-10-22 09:45:40 +00:00
|
|
|
device = OWServerDeviceDescription(
|
|
|
|
device_info=device_info,
|
|
|
|
id=device_id,
|
|
|
|
family=device_family,
|
|
|
|
path=device_path,
|
|
|
|
type=device_type,
|
|
|
|
)
|
|
|
|
devices.append(device)
|
2021-10-18 12:01:23 +00:00
|
|
|
if device_branches := DEVICE_COUPLERS.get(device_family):
|
2020-12-07 01:09:32 +00:00
|
|
|
for branch in device_branches:
|
2021-10-22 09:45:40 +00:00
|
|
|
devices += self._discover_devices_owserver(
|
|
|
|
f"{device_path}{branch}", device_id
|
|
|
|
)
|
|
|
|
|
2020-10-27 04:36:51 +00:00
|
|
|
return devices
|
2020-10-26 15:33:13 +00:00
|
|
|
|
2021-10-27 17:19:55 +00:00
|
|
|
def _get_device_type_owserver(self, device_path: str) -> str:
|
|
|
|
"""Get device model."""
|
|
|
|
if TYPE_CHECKING:
|
|
|
|
assert self.owproxy
|
|
|
|
device_type = self.owproxy.read(f"{device_path}type").decode()
|
|
|
|
_LOGGER.debug("read `%stype`: %s", device_path, device_type)
|
|
|
|
if device_type == "EDS":
|
|
|
|
device_type = self.owproxy.read(f"{device_path}device_type").decode()
|
|
|
|
_LOGGER.debug("read `%sdevice_type`: %s", device_path, device_type)
|
|
|
|
if TYPE_CHECKING:
|
|
|
|
assert isinstance(device_type, str)
|
|
|
|
return device_type
|
|
|
|
|
2020-10-26 15:33:13 +00:00
|
|
|
|
|
|
|
class CannotConnect(HomeAssistantError):
|
|
|
|
"""Error to indicate we cannot connect."""
|
|
|
|
|
|
|
|
|
|
|
|
class InvalidPath(HomeAssistantError):
|
|
|
|
"""Error to indicate the path is invalid."""
|