2015-04-15 14:47:42 +00:00
|
|
|
"""
|
2016-03-08 16:55:57 +00:00
|
|
|
Support for Modbus.
|
2015-08-06 15:53:49 +00:00
|
|
|
|
2015-10-21 19:11:18 +00:00
|
|
|
For more details about this component, please refer to the documentation at
|
2015-11-09 12:12:18 +00:00
|
|
|
https://home-assistant.io/components/modbus/
|
2015-04-15 14:47:42 +00:00
|
|
|
"""
|
|
|
|
import logging
|
2016-09-07 01:21:38 +00:00
|
|
|
import threading
|
2015-04-15 14:47:42 +00:00
|
|
|
|
2016-09-13 20:47:44 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2016-02-19 05:27:50 +00:00
|
|
|
from homeassistant.const import (
|
2016-09-13 20:47:44 +00:00
|
|
|
EVENT_HOMEASSISTANT_START, EVENT_HOMEASSISTANT_STOP,
|
|
|
|
CONF_HOST, CONF_METHOD, CONF_PORT)
|
|
|
|
import homeassistant.helpers.config_validation as cv
|
2015-04-15 14:47:42 +00:00
|
|
|
|
|
|
|
DOMAIN = "modbus"
|
|
|
|
|
2015-09-09 02:49:27 +00:00
|
|
|
REQUIREMENTS = ['https://github.com/bashwork/pymodbus/archive/'
|
|
|
|
'd7fc4f1cc975631e0a9011390e8017f64b612661.zip#pymodbus==1.2.0']
|
2015-04-15 14:47:42 +00:00
|
|
|
|
|
|
|
# Type of network
|
2016-09-13 20:47:44 +00:00
|
|
|
CONF_BAUDRATE = "baudrate"
|
|
|
|
CONF_BYTESIZE = "bytesize"
|
|
|
|
CONF_STOPBITS = "stopbits"
|
|
|
|
CONF_TYPE = "type"
|
|
|
|
CONF_PARITY = "parity"
|
|
|
|
|
|
|
|
SERIAL_SCHEMA = {
|
|
|
|
vol.Required(CONF_BAUDRATE): cv.positive_int,
|
|
|
|
vol.Required(CONF_BYTESIZE): vol.Any(5, 6, 7, 8),
|
|
|
|
vol.Required(CONF_METHOD): vol.Any('rtu', 'ascii'),
|
|
|
|
vol.Required(CONF_PORT): cv.string,
|
|
|
|
vol.Required(CONF_PARITY): vol.Any('E', 'O', 'N'),
|
|
|
|
vol.Required(CONF_STOPBITS): vol.Any(1, 2),
|
|
|
|
vol.Required(CONF_TYPE): 'serial',
|
|
|
|
}
|
|
|
|
|
|
|
|
ETHERNET_SCHEMA = {
|
|
|
|
vol.Required(CONF_HOST): cv.string,
|
|
|
|
vol.Required(CONF_PORT): cv.positive_int,
|
|
|
|
vol.Required(CONF_TYPE): vol.Any('tcp', 'udp'),
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
CONFIG_SCHEMA = vol.Schema({
|
|
|
|
DOMAIN: vol.Any(SERIAL_SCHEMA, ETHERNET_SCHEMA)
|
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
2015-04-15 14:47:42 +00:00
|
|
|
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2016-08-25 08:20:08 +00:00
|
|
|
SERVICE_WRITE_REGISTER = "write_register"
|
|
|
|
|
|
|
|
ATTR_ADDRESS = "address"
|
|
|
|
ATTR_UNIT = "unit"
|
|
|
|
ATTR_VALUE = "value"
|
|
|
|
|
2016-09-13 20:47:44 +00:00
|
|
|
SERVICE_WRITE_REGISTER_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(ATTR_UNIT): cv.positive_int,
|
|
|
|
vol.Required(ATTR_ADDRESS): cv.positive_int,
|
2017-02-02 09:23:13 +00:00
|
|
|
vol.Required(ATTR_VALUE): vol.All(cv.ensure_list, [cv.positive_int])
|
2016-09-13 20:47:44 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
|
2016-09-07 01:21:38 +00:00
|
|
|
HUB = None
|
2015-04-15 14:47:42 +00:00
|
|
|
|
2015-04-21 14:40:13 +00:00
|
|
|
|
2015-04-15 14:47:42 +00:00
|
|
|
def setup(hass, config):
|
2016-03-08 16:55:57 +00:00
|
|
|
"""Setup Modbus component."""
|
2015-04-21 14:40:13 +00:00
|
|
|
# Modbus connection type
|
|
|
|
# pylint: disable=global-statement, import-error
|
2016-09-13 20:47:44 +00:00
|
|
|
client_type = config[DOMAIN][CONF_TYPE]
|
2015-04-15 14:47:42 +00:00
|
|
|
|
|
|
|
# Connect to Modbus network
|
2015-04-21 14:40:13 +00:00
|
|
|
# pylint: disable=global-statement, import-error
|
2015-04-15 14:47:42 +00:00
|
|
|
|
2016-09-13 20:47:44 +00:00
|
|
|
if client_type == "serial":
|
2015-04-15 14:47:42 +00:00
|
|
|
from pymodbus.client.sync import ModbusSerialClient as ModbusClient
|
2016-09-13 20:47:44 +00:00
|
|
|
client = ModbusClient(method=config[DOMAIN][CONF_METHOD],
|
|
|
|
port=config[DOMAIN][CONF_PORT],
|
|
|
|
baudrate=config[DOMAIN][CONF_BAUDRATE],
|
|
|
|
stopbits=config[DOMAIN][CONF_STOPBITS],
|
|
|
|
bytesize=config[DOMAIN][CONF_BYTESIZE],
|
|
|
|
parity=config[DOMAIN][CONF_PARITY])
|
|
|
|
elif client_type == "tcp":
|
2015-04-15 14:47:42 +00:00
|
|
|
from pymodbus.client.sync import ModbusTcpClient as ModbusClient
|
2016-09-13 20:47:44 +00:00
|
|
|
client = ModbusClient(host=config[DOMAIN][CONF_HOST],
|
|
|
|
port=config[DOMAIN][CONF_PORT])
|
|
|
|
elif client_type == "udp":
|
2015-04-15 14:47:42 +00:00
|
|
|
from pymodbus.client.sync import ModbusUdpClient as ModbusClient
|
2016-09-13 20:47:44 +00:00
|
|
|
client = ModbusClient(host=config[DOMAIN][CONF_HOST],
|
|
|
|
port=config[DOMAIN][CONF_PORT])
|
2015-04-15 14:47:42 +00:00
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
2016-09-07 01:21:38 +00:00
|
|
|
global HUB
|
|
|
|
HUB = ModbusHub(client)
|
|
|
|
|
2015-04-15 14:47:42 +00:00
|
|
|
def stop_modbus(event):
|
2016-03-08 16:55:57 +00:00
|
|
|
"""Stop Modbus service."""
|
2016-09-07 01:21:38 +00:00
|
|
|
HUB.close()
|
2015-04-15 14:47:42 +00:00
|
|
|
|
|
|
|
def start_modbus(event):
|
2016-03-08 16:55:57 +00:00
|
|
|
"""Start Modbus service."""
|
2016-09-07 01:21:38 +00:00
|
|
|
HUB.connect()
|
2015-04-15 14:47:42 +00:00
|
|
|
hass.bus.listen_once(EVENT_HOMEASSISTANT_STOP, stop_modbus)
|
|
|
|
|
2016-08-25 08:20:08 +00:00
|
|
|
# Register services for modbus
|
2016-09-13 20:47:44 +00:00
|
|
|
hass.services.register(DOMAIN, SERVICE_WRITE_REGISTER, write_register,
|
|
|
|
schema=SERVICE_WRITE_REGISTER_SCHEMA)
|
2016-08-25 08:20:08 +00:00
|
|
|
|
|
|
|
def write_register(service):
|
2016-09-10 15:17:28 +00:00
|
|
|
"""Write modbus registers."""
|
2016-08-25 08:20:08 +00:00
|
|
|
unit = int(float(service.data.get(ATTR_UNIT)))
|
|
|
|
address = int(float(service.data.get(ATTR_ADDRESS)))
|
2016-09-10 15:17:28 +00:00
|
|
|
value = service.data.get(ATTR_VALUE)
|
|
|
|
if isinstance(value, list):
|
|
|
|
HUB.write_registers(
|
|
|
|
unit,
|
|
|
|
address,
|
|
|
|
[int(float(i)) for i in value])
|
|
|
|
else:
|
|
|
|
HUB.write_register(
|
|
|
|
unit,
|
|
|
|
address,
|
|
|
|
int(float(value)))
|
2016-08-25 08:20:08 +00:00
|
|
|
|
2015-04-15 14:47:42 +00:00
|
|
|
hass.bus.listen_once(EVENT_HOMEASSISTANT_START, start_modbus)
|
|
|
|
|
|
|
|
return True
|
2016-09-07 01:21:38 +00:00
|
|
|
|
|
|
|
|
|
|
|
class ModbusHub(object):
|
|
|
|
"""Thread safe wrapper class for pymodbus."""
|
|
|
|
|
|
|
|
def __init__(self, modbus_client):
|
|
|
|
"""Initialize the modbus hub."""
|
|
|
|
self._client = modbus_client
|
|
|
|
self._lock = threading.Lock()
|
|
|
|
|
|
|
|
def close(self):
|
|
|
|
"""Disconnect client."""
|
|
|
|
with self._lock:
|
|
|
|
self._client.close()
|
|
|
|
|
|
|
|
def connect(self):
|
|
|
|
"""Connect client."""
|
|
|
|
with self._lock:
|
|
|
|
self._client.connect()
|
|
|
|
|
|
|
|
def read_coils(self, unit, address, count):
|
|
|
|
"""Read coils."""
|
|
|
|
with self._lock:
|
2016-09-13 20:47:44 +00:00
|
|
|
kwargs = {'unit': unit} if unit else {}
|
2016-09-07 01:21:38 +00:00
|
|
|
return self._client.read_coils(
|
|
|
|
address,
|
|
|
|
count,
|
2016-09-13 20:47:44 +00:00
|
|
|
**kwargs)
|
2016-09-07 01:21:38 +00:00
|
|
|
|
|
|
|
def read_holding_registers(self, unit, address, count):
|
|
|
|
"""Read holding registers."""
|
|
|
|
with self._lock:
|
2016-09-13 20:47:44 +00:00
|
|
|
kwargs = {'unit': unit} if unit else {}
|
2016-09-07 01:21:38 +00:00
|
|
|
return self._client.read_holding_registers(
|
|
|
|
address,
|
|
|
|
count,
|
2016-09-13 20:47:44 +00:00
|
|
|
**kwargs)
|
2016-09-07 01:21:38 +00:00
|
|
|
|
|
|
|
def write_coil(self, unit, address, value):
|
|
|
|
"""Write coil."""
|
|
|
|
with self._lock:
|
2016-09-13 20:47:44 +00:00
|
|
|
kwargs = {'unit': unit} if unit else {}
|
2016-09-07 01:21:38 +00:00
|
|
|
self._client.write_coil(
|
|
|
|
address,
|
|
|
|
value,
|
2016-09-13 20:47:44 +00:00
|
|
|
**kwargs)
|
2016-09-07 01:21:38 +00:00
|
|
|
|
|
|
|
def write_register(self, unit, address, value):
|
|
|
|
"""Write register."""
|
|
|
|
with self._lock:
|
2016-09-13 20:47:44 +00:00
|
|
|
kwargs = {'unit': unit} if unit else {}
|
2016-09-07 01:21:38 +00:00
|
|
|
self._client.write_register(
|
|
|
|
address,
|
|
|
|
value,
|
2016-09-13 20:47:44 +00:00
|
|
|
**kwargs)
|
2016-09-10 15:17:28 +00:00
|
|
|
|
|
|
|
def write_registers(self, unit, address, values):
|
|
|
|
"""Write registers."""
|
|
|
|
with self._lock:
|
2016-09-13 20:47:44 +00:00
|
|
|
kwargs = {'unit': unit} if unit else {}
|
2016-09-10 15:17:28 +00:00
|
|
|
self._client.write_registers(
|
|
|
|
address,
|
|
|
|
values,
|
2016-09-13 20:47:44 +00:00
|
|
|
**kwargs)
|