2019-02-13 20:21:14 +00:00
|
|
|
"""Support for raspihats board switches."""
|
2017-06-05 06:56:21 +00:00
|
|
|
import logging
|
2018-01-21 06:35:38 +00:00
|
|
|
|
2017-06-05 06:56:21 +00:00
|
|
|
import voluptuous as vol
|
2018-01-21 06:35:38 +00:00
|
|
|
|
2017-06-05 06:56:21 +00:00
|
|
|
from homeassistant.components.switch import PLATFORM_SCHEMA
|
2019-02-19 13:09:06 +00:00
|
|
|
from homeassistant.const import CONF_ADDRESS, CONF_NAME, DEVICE_DEFAULT_NAME
|
2018-01-21 06:35:38 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2017-06-05 06:56:21 +00:00
|
|
|
from homeassistant.helpers.entity import ToggleEntity
|
|
|
|
|
2019-03-21 05:56:46 +00:00
|
|
|
from . import (
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_BOARD,
|
|
|
|
CONF_CHANNELS,
|
|
|
|
CONF_I2C_HATS,
|
|
|
|
CONF_INDEX,
|
|
|
|
CONF_INITIAL_STATE,
|
|
|
|
CONF_INVERT_LOGIC,
|
|
|
|
I2C_HAT_NAMES,
|
|
|
|
I2C_HATS_MANAGER,
|
|
|
|
I2CHatsException,
|
|
|
|
)
|
2019-03-21 05:56:46 +00:00
|
|
|
|
2017-06-05 06:56:21 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
_CHANNELS_SCHEMA = vol.Schema(
|
|
|
|
[
|
|
|
|
{
|
|
|
|
vol.Required(CONF_INDEX): cv.positive_int,
|
|
|
|
vol.Required(CONF_NAME): cv.string,
|
|
|
|
vol.Optional(CONF_INVERT_LOGIC, default=False): cv.boolean,
|
|
|
|
vol.Optional(CONF_INITIAL_STATE): cv.boolean,
|
|
|
|
}
|
|
|
|
]
|
|
|
|
)
|
|
|
|
|
|
|
|
_I2C_HATS_SCHEMA = vol.Schema(
|
|
|
|
[
|
|
|
|
{
|
|
|
|
vol.Required(CONF_BOARD): vol.In(I2C_HAT_NAMES),
|
|
|
|
vol.Required(CONF_ADDRESS): vol.Coerce(int),
|
|
|
|
vol.Required(CONF_CHANNELS): _CHANNELS_SCHEMA,
|
|
|
|
}
|
|
|
|
]
|
|
|
|
)
|
|
|
|
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
|
|
|
{vol.Optional(CONF_I2C_HATS): _I2C_HATS_SCHEMA}
|
|
|
|
)
|
2017-06-05 06:56:21 +00:00
|
|
|
|
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Set up the raspihats switch devices."""
|
2017-06-05 06:56:21 +00:00
|
|
|
I2CHatSwitch.I2C_HATS_MANAGER = hass.data[I2C_HATS_MANAGER]
|
|
|
|
switches = []
|
|
|
|
i2c_hat_configs = config.get(CONF_I2C_HATS)
|
|
|
|
for i2c_hat_config in i2c_hat_configs:
|
|
|
|
board = i2c_hat_config[CONF_BOARD]
|
|
|
|
address = i2c_hat_config[CONF_ADDRESS]
|
|
|
|
try:
|
|
|
|
I2CHatSwitch.I2C_HATS_MANAGER.register_board(board, address)
|
|
|
|
for channel_config in i2c_hat_config[CONF_CHANNELS]:
|
|
|
|
switches.append(
|
|
|
|
I2CHatSwitch(
|
2019-07-31 19:25:30 +00:00
|
|
|
board,
|
|
|
|
address,
|
|
|
|
channel_config[CONF_INDEX],
|
2017-06-05 06:56:21 +00:00
|
|
|
channel_config[CONF_NAME],
|
|
|
|
channel_config[CONF_INVERT_LOGIC],
|
2019-07-31 19:25:30 +00:00
|
|
|
channel_config.get(CONF_INITIAL_STATE),
|
2017-06-05 06:56:21 +00:00
|
|
|
)
|
|
|
|
)
|
|
|
|
except I2CHatsException as ex:
|
|
|
|
_LOGGER.error(
|
2019-07-31 19:25:30 +00:00
|
|
|
"Failed to register %s I2CHat@%s %s", board, hex(address), str(ex)
|
|
|
|
)
|
2018-08-24 14:37:30 +00:00
|
|
|
add_entities(switches)
|
2017-06-05 06:56:21 +00:00
|
|
|
|
|
|
|
|
|
|
|
class I2CHatSwitch(ToggleEntity):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Representation a switch that uses a I2C-HAT digital output."""
|
2017-06-05 06:56:21 +00:00
|
|
|
|
|
|
|
I2C_HATS_MANAGER = None
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
def __init__(self, board, address, channel, name, invert_logic, initial_state):
|
2017-06-05 06:56:21 +00:00
|
|
|
"""Initialize switch."""
|
|
|
|
self._board = board
|
|
|
|
self._address = address
|
|
|
|
self._channel = channel
|
|
|
|
self._name = name or DEVICE_DEFAULT_NAME
|
|
|
|
self._invert_logic = invert_logic
|
|
|
|
if initial_state is not None:
|
|
|
|
if self._invert_logic:
|
|
|
|
state = not initial_state
|
|
|
|
else:
|
|
|
|
state = initial_state
|
2019-07-31 19:25:30 +00:00
|
|
|
self.I2C_HATS_MANAGER.write_dq(self._address, self._channel, state)
|
2017-06-05 06:56:21 +00:00
|
|
|
|
|
|
|
def online_callback():
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Call fired when board is online."""
|
2017-06-05 06:56:21 +00:00
|
|
|
self.schedule_update_ha_state()
|
|
|
|
|
|
|
|
self.I2C_HATS_MANAGER.register_online_callback(
|
2019-07-31 19:25:30 +00:00
|
|
|
self._address, self._channel, online_callback
|
|
|
|
)
|
2017-06-05 06:56:21 +00:00
|
|
|
|
|
|
|
def _log_message(self, message):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Create log message."""
|
2020-04-05 14:01:41 +00:00
|
|
|
string = f"{self._name} "
|
2020-04-04 21:09:34 +00:00
|
|
|
string += f"{self._board}I2CHat@{hex(self._address)} "
|
|
|
|
string += f"channel:{str(self._channel)}{message}"
|
2017-06-05 06:56:21 +00:00
|
|
|
return string
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the switch."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def should_poll(self):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Return the polling state."""
|
2017-06-05 06:56:21 +00:00
|
|
|
return False
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self):
|
|
|
|
"""Return true if device is on."""
|
|
|
|
try:
|
2018-01-21 06:35:38 +00:00
|
|
|
state = self.I2C_HATS_MANAGER.read_dq(self._address, self._channel)
|
2017-06-05 06:56:21 +00:00
|
|
|
return state != self._invert_logic
|
|
|
|
except I2CHatsException as ex:
|
2020-01-03 13:47:06 +00:00
|
|
|
_LOGGER.error(self._log_message(f"Is ON check failed, {ex!s}"))
|
2017-06-05 06:56:21 +00:00
|
|
|
return False
|
|
|
|
|
2018-02-11 17:20:28 +00:00
|
|
|
def turn_on(self, **kwargs):
|
2017-06-05 06:56:21 +00:00
|
|
|
"""Turn the device on."""
|
|
|
|
try:
|
2018-12-06 10:54:44 +00:00
|
|
|
state = self._invert_logic is False
|
2018-01-21 06:35:38 +00:00
|
|
|
self.I2C_HATS_MANAGER.write_dq(self._address, self._channel, state)
|
2017-06-05 06:56:21 +00:00
|
|
|
self.schedule_update_ha_state()
|
|
|
|
except I2CHatsException as ex:
|
2020-01-03 13:47:06 +00:00
|
|
|
_LOGGER.error(self._log_message(f"Turn ON failed, {ex!s}"))
|
2017-06-05 06:56:21 +00:00
|
|
|
|
2018-02-11 17:20:28 +00:00
|
|
|
def turn_off(self, **kwargs):
|
2017-06-05 06:56:21 +00:00
|
|
|
"""Turn the device off."""
|
|
|
|
try:
|
2018-12-06 10:54:44 +00:00
|
|
|
state = self._invert_logic is not False
|
2018-01-21 06:35:38 +00:00
|
|
|
self.I2C_HATS_MANAGER.write_dq(self._address, self._channel, state)
|
2017-06-05 06:56:21 +00:00
|
|
|
self.schedule_update_ha_state()
|
|
|
|
except I2CHatsException as ex:
|
2020-01-03 13:47:06 +00:00
|
|
|
_LOGGER.error(self._log_message(f"Turn OFF failed, {ex!s}"))
|