2019-06-01 16:17:51 +02:00
|
|
|
"""FHEM integration"""
|
|
|
|
|
|
|
|
import logging
|
|
|
|
import voluptuous as vol
|
2019-06-15 11:50:20 +02:00
|
|
|
from collections import defaultdict
|
2019-06-01 16:17:51 +02:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2019-06-19 19:22:06 +02:00
|
|
|
from homeassistant.const import CONF_HOST, CONF_PORT
|
|
|
|
from ..reconnecting_client import ReconnectingClient
|
2019-06-01 16:17:51 +02:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
CONF_CUL_DEVICE_NAME = 'cul_device_name'
|
2019-06-15 11:50:20 +02:00
|
|
|
CONF_FHEM_SENSOR_TYPE = 'fhem_sensor_type'
|
|
|
|
CONF_FHEM_IDS = 'fhem_ids'
|
2019-06-01 16:17:51 +02:00
|
|
|
DOMAIN = 'fhem'
|
|
|
|
DATA_FHEM = "data_fhem"
|
|
|
|
CONFIG_SCHEMA = vol.Schema({
|
|
|
|
DOMAIN: vol.Schema({
|
|
|
|
vol.Required(CONF_HOST): cv.string,
|
|
|
|
vol.Optional(CONF_PORT, default=7072): cv.port,
|
|
|
|
vol.Required(CONF_CUL_DEVICE_NAME): cv.string,
|
|
|
|
})
|
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
|
|
|
|
|
|
|
|
|
|
|
async def async_setup(hass, config):
|
2019-06-19 19:22:06 +02:00
|
|
|
connection = FhemConnection(hass, config[DOMAIN])
|
2019-06-01 16:17:51 +02:00
|
|
|
hass.data[DATA_FHEM] = connection
|
|
|
|
await connection.start()
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
2019-06-19 19:22:06 +02:00
|
|
|
class FhemConnection(ReconnectingClient):
|
2019-06-01 16:17:51 +02:00
|
|
|
|
|
|
|
def __init__(self, hass, config):
|
2019-06-19 19:22:06 +02:00
|
|
|
super().__init__(hass, config[CONF_HOST], config[CONF_PORT], "FHEM",
|
|
|
|
receive_line_callback=self._process_line,
|
|
|
|
connection_status_changed_callback=self._update_all_devices)
|
|
|
|
self._cul_device_name = config[CONF_CUL_DEVICE_NAME]
|
2019-06-15 11:50:20 +02:00
|
|
|
self._devices = defaultdict(list)
|
|
|
|
|
|
|
|
def register_device(self, id, d):
|
|
|
|
self._devices[id].append(d)
|
2019-06-19 16:12:22 +02:00
|
|
|
if self._writer:
|
2019-07-07 17:30:45 +02:00
|
|
|
d.refresh()
|
2019-06-15 11:50:20 +02:00
|
|
|
|
2019-06-19 22:10:03 +02:00
|
|
|
async def _update_all_devices(self, state):
|
|
|
|
if state == 'connected':
|
|
|
|
self.write_line("displayattr .*")
|
|
|
|
self.write_line("inform on")
|
2019-06-15 11:50:20 +02:00
|
|
|
for device_list in self._devices.values():
|
|
|
|
for device in device_list:
|
2019-07-07 17:30:45 +02:00
|
|
|
device.refresh()
|
2020-01-17 21:32:06 +01:00
|
|
|
# await device.async_update_ha_state()
|
2019-06-01 16:17:51 +02:00
|
|
|
|
|
|
|
async def _process_line(self, line):
|
|
|
|
if line.startswith(self._cul_device_name + " "): # Status update message
|
|
|
|
_, device_name, command = line.split(" ", 2)
|
2019-06-15 11:50:20 +02:00
|
|
|
for device in self._devices[device_name]:
|
2020-04-19 15:59:51 +02:00
|
|
|
#_LOGGER.debug("FHEM line received (device): " + device_name + ": " + line)
|
2019-06-15 11:50:20 +02:00
|
|
|
await device.line_received(command.strip())
|
2019-06-01 16:17:51 +02:00
|
|
|
else: # potential response to displayattr
|
|
|
|
split_line = line.split(" ", 1)
|
|
|
|
if len(split_line) == 2:
|
|
|
|
device_name, command = split_line
|
2019-06-15 11:50:20 +02:00
|
|
|
for device in self._devices[device_name]:
|
|
|
|
await device.line_received(command.strip())
|
2019-06-01 16:17:51 +02:00
|
|
|
|
|
|
|
def write_line(self, line):
|
|
|
|
if self._writer:
|
|
|
|
line += '\n'
|
|
|
|
self._writer.write(line.encode())
|
2020-04-19 15:59:51 +02:00
|
|
|
#_LOGGER.debug(f"FHEM write line {line}")
|
2019-12-02 20:40:30 +01:00
|
|
|
else:
|
|
|
|
_LOGGER.debug(f"FHEM Failed to write line {line}")
|
|
|
|
|
2019-06-01 16:17:51 +02:00
|
|
|
def fhem_set(self, id, *arguments):
|
|
|
|
"""
|
|
|
|
Send command to FHEM using this device
|
|
|
|
:param arguments: string or list of strings containing command parameters
|
|
|
|
"""
|
|
|
|
arguments = " ".join([str(a) for a in arguments])
|
2019-06-19 19:22:06 +02:00
|
|
|
self.write_line("set {} {}\n".format(id, arguments))
|
2019-06-15 11:50:20 +02:00
|
|
|
|
|
|
|
|
|
|
|
def device_error_reporting(hass, received_line, component_type, component_name):
|
|
|
|
if received_line.startswith('overheat'):
|
|
|
|
overheat = received_line.split(':')[1]
|
|
|
|
overheat = overheat.strip().lower()
|
|
|
|
assert overheat == 'on' or overheat == 'off'
|
|
|
|
if overheat == 'on':
|
|
|
|
text = "FHEM: {} overheated: <br><b>{}</b>".format(component_type, component_name)
|
|
|
|
hass.components.persistent_notification.async_create(text, title="{} overheat".format(component_type))
|
|
|
|
elif received_line.startswith('overload'):
|
|
|
|
overload = received_line.split(':')[1]
|
|
|
|
overload = overload.strip().lower()
|
|
|
|
assert overload == 'on' or overload == 'off'
|
|
|
|
if overload == 'on':
|
|
|
|
text = "FHEM: {} overloaded: <br><b>{}</b>".format(component_type, component_name)
|
|
|
|
hass.components.persistent_notification.async_create(text, title="{} overloaded".format(component_type))
|