2019-02-13 20:21:14 +00:00
|
|
|
"""Support for SCSGate switches."""
|
2022-01-03 15:44:20 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2016-01-09 13:51:49 +00:00
|
|
|
import logging
|
2022-09-06 11:35:14 +00:00
|
|
|
from typing import Any
|
2016-01-09 13:51:49 +00:00
|
|
|
|
2019-12-01 05:24:54 +00:00
|
|
|
from scsgate.messages import ScenarioTriggeredMessage, StateMessage
|
|
|
|
from scsgate.tasks import ToggleStatusTask
|
2016-09-13 05:23:53 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2020-04-26 16:50:37 +00:00
|
|
|
from homeassistant.components.switch import PLATFORM_SCHEMA, SwitchEntity
|
2019-12-01 05:24:54 +00:00
|
|
|
from homeassistant.const import ATTR_ENTITY_ID, ATTR_STATE, CONF_DEVICES, CONF_NAME
|
2022-01-03 15:44:20 +00:00
|
|
|
from homeassistant.core import HomeAssistant
|
2016-09-13 05:23:53 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2022-01-03 15:44:20 +00:00
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
|
|
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
2016-09-13 05:23:53 +00:00
|
|
|
|
2020-04-05 21:47:11 +00:00
|
|
|
from . import CONF_SCS_ID, DOMAIN, SCSGATE_SCHEMA
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_SCENARIO_ID = "scenario_id"
|
2016-01-09 13:51:49 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_TRADITIONAL = "traditional"
|
|
|
|
CONF_SCENARIO = "scenario"
|
2016-09-13 05:23:53 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
2020-04-05 21:47:11 +00:00
|
|
|
{vol.Required(CONF_DEVICES): cv.schema_with_slug_keys(SCSGATE_SCHEMA)}
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2016-09-13 05:23:53 +00:00
|
|
|
|
|
|
|
|
2022-01-03 15:44:20 +00:00
|
|
|
def setup_platform(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
config: ConfigType,
|
|
|
|
add_entities: AddEntitiesCallback,
|
|
|
|
discovery_info: DiscoveryInfoType | None = None,
|
|
|
|
) -> None:
|
2017-05-02 16:18:47 +00:00
|
|
|
"""Set up the SCSGate switches."""
|
2016-01-09 13:51:49 +00:00
|
|
|
logger = logging.getLogger(__name__)
|
2020-04-05 21:47:11 +00:00
|
|
|
scsgate = hass.data[DOMAIN]
|
2016-01-09 13:51:49 +00:00
|
|
|
|
|
|
|
_setup_traditional_switches(
|
2020-04-05 21:47:11 +00:00
|
|
|
logger=logger,
|
|
|
|
config=config,
|
|
|
|
scsgate=scsgate,
|
|
|
|
add_entities_callback=add_entities,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2016-01-09 13:51:49 +00:00
|
|
|
|
2020-04-05 21:47:11 +00:00
|
|
|
_setup_scenario_switches(logger=logger, config=config, scsgate=scsgate, hass=hass)
|
2016-01-09 13:51:49 +00:00
|
|
|
|
|
|
|
|
2020-04-05 21:47:11 +00:00
|
|
|
def _setup_traditional_switches(logger, config, scsgate, add_entities_callback):
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Add traditional SCSGate switches."""
|
2016-09-13 05:23:53 +00:00
|
|
|
traditional = config.get(CONF_TRADITIONAL)
|
2016-01-09 13:51:49 +00:00
|
|
|
switches = []
|
|
|
|
|
|
|
|
if traditional:
|
2020-10-28 10:59:07 +00:00
|
|
|
for entity_info in traditional.values():
|
2020-04-05 21:47:11 +00:00
|
|
|
if entity_info[CONF_SCS_ID] in scsgate.devices:
|
2016-01-09 13:51:49 +00:00
|
|
|
continue
|
|
|
|
|
2016-09-13 05:23:53 +00:00
|
|
|
name = entity_info[CONF_NAME]
|
2020-04-05 21:47:11 +00:00
|
|
|
scs_id = entity_info[CONF_SCS_ID]
|
2016-01-09 13:51:49 +00:00
|
|
|
|
2016-09-13 05:23:53 +00:00
|
|
|
logger.info("Adding %s scsgate.traditional_switch", name)
|
2016-01-09 13:51:49 +00:00
|
|
|
|
2020-04-05 21:47:11 +00:00
|
|
|
switch = SCSGateSwitch(
|
|
|
|
name=name, scs_id=scs_id, logger=logger, scsgate=scsgate
|
|
|
|
)
|
2016-01-09 13:51:49 +00:00
|
|
|
switches.append(switch)
|
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
add_entities_callback(switches)
|
2020-04-05 21:47:11 +00:00
|
|
|
scsgate.add_devices_to_register(switches)
|
2016-01-09 13:51:49 +00:00
|
|
|
|
|
|
|
|
2020-04-05 21:47:11 +00:00
|
|
|
def _setup_scenario_switches(logger, config, scsgate, hass):
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Add only SCSGate scenario switches."""
|
2021-10-22 12:07:19 +00:00
|
|
|
if scenario := config.get(CONF_SCENARIO):
|
2020-10-28 10:59:07 +00:00
|
|
|
for entity_info in scenario.values():
|
2020-04-05 21:47:11 +00:00
|
|
|
if entity_info[CONF_SCS_ID] in scsgate.devices:
|
2016-01-09 13:51:49 +00:00
|
|
|
continue
|
|
|
|
|
2016-09-13 05:23:53 +00:00
|
|
|
name = entity_info[CONF_NAME]
|
2020-04-05 21:47:11 +00:00
|
|
|
scs_id = entity_info[CONF_SCS_ID]
|
2016-01-09 13:51:49 +00:00
|
|
|
|
2016-09-13 05:23:53 +00:00
|
|
|
logger.info("Adding %s scsgate.scenario_switch", name)
|
2016-01-09 13:51:49 +00:00
|
|
|
|
|
|
|
switch = SCSGateScenarioSwitch(
|
2019-07-31 19:25:30 +00:00
|
|
|
name=name, scs_id=scs_id, logger=logger, hass=hass
|
|
|
|
)
|
2020-04-05 21:47:11 +00:00
|
|
|
scsgate.add_device(switch)
|
2016-01-09 13:51:49 +00:00
|
|
|
|
|
|
|
|
2020-04-26 16:50:37 +00:00
|
|
|
class SCSGateSwitch(SwitchEntity):
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Representation of a SCSGate switch."""
|
|
|
|
|
2022-08-27 02:03:50 +00:00
|
|
|
_attr_should_poll = False
|
|
|
|
|
2020-04-05 21:47:11 +00:00
|
|
|
def __init__(self, scs_id, name, logger, scsgate):
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Initialize the switch."""
|
2016-01-09 13:51:49 +00:00
|
|
|
self._name = name
|
|
|
|
self._scs_id = scs_id
|
|
|
|
self._toggled = False
|
|
|
|
self._logger = logger
|
2020-04-05 21:47:11 +00:00
|
|
|
self._scsgate = scsgate
|
2016-01-09 13:51:49 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def scs_id(self):
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Return the SCS ID."""
|
2016-01-09 13:51:49 +00:00
|
|
|
return self._scs_id
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Return the name of the device if any."""
|
2016-01-09 13:51:49 +00:00
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self):
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Return true if switch is on."""
|
2016-01-09 13:51:49 +00:00
|
|
|
return self._toggled
|
|
|
|
|
2022-09-06 11:35:14 +00:00
|
|
|
def turn_on(self, **kwargs: Any) -> None:
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Turn the device on."""
|
2016-01-09 13:51:49 +00:00
|
|
|
|
2020-04-05 21:47:11 +00:00
|
|
|
self._scsgate.append_task(ToggleStatusTask(target=self._scs_id, toggled=True))
|
2016-01-09 13:51:49 +00:00
|
|
|
|
|
|
|
self._toggled = True
|
2016-11-16 16:26:29 +00:00
|
|
|
self.schedule_update_ha_state()
|
2016-01-09 13:51:49 +00:00
|
|
|
|
2022-09-06 11:35:14 +00:00
|
|
|
def turn_off(self, **kwargs: Any) -> None:
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Turn the device off."""
|
2016-01-09 13:51:49 +00:00
|
|
|
|
2020-04-05 21:47:11 +00:00
|
|
|
self._scsgate.append_task(ToggleStatusTask(target=self._scs_id, toggled=False))
|
2016-01-09 13:51:49 +00:00
|
|
|
|
|
|
|
self._toggled = False
|
2016-11-16 16:26:29 +00:00
|
|
|
self.schedule_update_ha_state()
|
2016-01-09 13:51:49 +00:00
|
|
|
|
|
|
|
def process_event(self, message):
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Handle a SCSGate message related with this switch."""
|
2016-01-09 13:51:49 +00:00
|
|
|
if self._toggled == message.toggled:
|
|
|
|
self._logger.info(
|
|
|
|
"Switch %s, ignoring message %s because state already active",
|
2019-07-31 19:25:30 +00:00
|
|
|
self._scs_id,
|
|
|
|
message,
|
|
|
|
)
|
2016-01-09 13:51:49 +00:00
|
|
|
# Nothing changed, ignoring
|
|
|
|
return
|
|
|
|
|
|
|
|
self._toggled = message.toggled
|
2017-03-04 23:10:36 +00:00
|
|
|
self.schedule_update_ha_state()
|
2016-01-09 13:51:49 +00:00
|
|
|
|
|
|
|
command = "off"
|
|
|
|
if self._toggled:
|
|
|
|
command = "on"
|
|
|
|
|
|
|
|
self.hass.bus.fire(
|
2019-07-31 19:25:30 +00:00
|
|
|
"button_pressed", {ATTR_ENTITY_ID: self._scs_id, ATTR_STATE: command}
|
2016-01-09 13:51:49 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2018-07-20 08:45:20 +00:00
|
|
|
class SCSGateScenarioSwitch:
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Provides a SCSGate scenario switch.
|
2016-01-09 13:51:49 +00:00
|
|
|
|
2018-01-27 19:58:27 +00:00
|
|
|
This switch is always in an 'off" state, when toggled it's used to trigger
|
2016-03-08 12:35:39 +00:00
|
|
|
events.
|
2016-01-09 13:51:49 +00:00
|
|
|
"""
|
2016-03-08 12:35:39 +00:00
|
|
|
|
2016-01-09 13:51:49 +00:00
|
|
|
def __init__(self, scs_id, name, logger, hass):
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Initialize the scenario."""
|
2016-01-09 13:51:49 +00:00
|
|
|
self._name = name
|
|
|
|
self._scs_id = scs_id
|
|
|
|
self._logger = logger
|
|
|
|
self._hass = hass
|
|
|
|
|
|
|
|
@property
|
|
|
|
def scs_id(self):
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Return the SCS ID."""
|
2016-01-09 13:51:49 +00:00
|
|
|
return self._scs_id
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Return the name of the device if any."""
|
2016-01-09 13:51:49 +00:00
|
|
|
return self._name
|
|
|
|
|
|
|
|
def process_event(self, message):
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Handle a SCSGate message related with this switch."""
|
2016-01-09 13:51:49 +00:00
|
|
|
|
|
|
|
if isinstance(message, StateMessage):
|
|
|
|
scenario_id = message.bytes[4]
|
|
|
|
elif isinstance(message, ScenarioTriggeredMessage):
|
|
|
|
scenario_id = message.scenario
|
|
|
|
else:
|
2021-11-18 12:13:45 +00:00
|
|
|
self._logger.warning(
|
|
|
|
"Scenario switch: received unknown message %s", message
|
|
|
|
)
|
2016-01-09 13:51:49 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
self._hass.bus.fire(
|
2019-07-31 19:25:30 +00:00
|
|
|
"scenario_switch_triggered",
|
|
|
|
{ATTR_ENTITY_ID: int(self._scs_id), ATTR_SCENARIO_ID: int(scenario_id, 16)},
|
2016-01-09 13:51:49 +00:00
|
|
|
)
|