* Connect to PLM and process simple protocol callbacks * Baseline commit * Connect to PLM and process simple protocol callbacks * Baseline commit * Connection working again * Async add devices is working via callback now * Beginning to interface with PLM library for control and state * Deal with brightness in 255 levels with library * Change sub names to match API changes * Remove PLM-level update callback * Support dimmable based on underlying PLM device attributes * Expand to non-light platforms * Stubs for turn on and off * Current version of Python library * Amend to use switch device attributes * Use asyncio endpoints for control * Add logging line * Bump module version to 0.7.1 * Auto-load platforms, display device info/attributes * Unify method name for getting a device attribute * Require Current version of insteonplm module * Import the component function in each platform in the balloob-recommend manner * For consistency, handle switch state as onlevel just like lights * Use level 0xff for on state, even with binary switches Observing the behavior of a 2477S switch, it looks like even the non-dimmable devices use 0x00 and 0xff for off/on respectively. I was using 0x01 for on previously, but that yields unnecessary state change callbacks when message traffic ends up flipping the onlevel from 0xff to 0x01 or 0x01 to 0xff. * Use sensorstate attribute for sensor onoff * Move new device callback to devices attribute * Add support for platform override on a device * Bump version of insteonplm module * Default overrides is an empty list * Avoid calling private methods when doing common attributes * Remove unused CONF_DEBUG for now * flake8 and pylint code cleanup * Move get_component to local function where it is needed * Update to include insteonplm module. * New files for insteon_plm component * Legitimate class doctring instead of stub * Docstring changes. * Style changes as requested by @SEJeff * Changes requested by @pvizeli * Add @callback decorator to callback functions * Opportunistic platform loading triggered by qualifying device detection Instead of loading all the constituent platforms that comprise the insteon_plm component, instead we defer and wait until we receive a callback for a device that requires the platform.
97 lines
2.8 KiB
Python
97 lines
2.8 KiB
Python
"""
|
|
Support for INSTEON dimmers via PowerLinc Modem.
|
|
|
|
For more details about this component, please refer to the documentation at
|
|
https://home-assistant.io/components/insteon_plm/
|
|
"""
|
|
import logging
|
|
import asyncio
|
|
|
|
from homeassistant.core import callback
|
|
from homeassistant.components.switch import (SwitchDevice)
|
|
from homeassistant.loader import get_component
|
|
|
|
DEPENDENCIES = ['insteon_plm']
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
@asyncio.coroutine
|
|
def async_setup_platform(hass, config, async_add_devices, discovery_info=None):
|
|
"""Set up the INSTEON PLM device class for the hass platform."""
|
|
plm = hass.data['insteon_plm']
|
|
|
|
device_list = []
|
|
for device in discovery_info:
|
|
name = device.get('address')
|
|
address = device.get('address_hex')
|
|
|
|
_LOGGER.info('Registered %s with switch platform.', name)
|
|
|
|
device_list.append(
|
|
InsteonPLMSwitchDevice(hass, plm, address, name)
|
|
)
|
|
|
|
hass.async_add_job(async_add_devices(device_list))
|
|
|
|
|
|
class InsteonPLMSwitchDevice(SwitchDevice):
|
|
"""A Class for an Insteon device."""
|
|
|
|
def __init__(self, hass, plm, address, name):
|
|
"""Initialize the switch."""
|
|
self._hass = hass
|
|
self._plm = plm.protocol
|
|
self._address = address
|
|
self._name = name
|
|
|
|
self._plm.add_update_callback(
|
|
self.async_switch_update, {'address': self._address})
|
|
|
|
@property
|
|
def should_poll(self):
|
|
"""No polling needed."""
|
|
return False
|
|
|
|
@property
|
|
def address(self):
|
|
"""Return the the address of the node."""
|
|
return self._address
|
|
|
|
@property
|
|
def name(self):
|
|
"""Return the the name of the node."""
|
|
return self._name
|
|
|
|
@property
|
|
def is_on(self):
|
|
"""Return the boolean response if the node is on."""
|
|
onlevel = self._plm.get_device_attr(self._address, 'onlevel')
|
|
_LOGGER.debug('on level for %s is %s', self._address, onlevel)
|
|
return bool(onlevel)
|
|
|
|
@property
|
|
def device_state_attributes(self):
|
|
"""Provide attributes for display on device card."""
|
|
insteon_plm = get_component('insteon_plm')
|
|
return insteon_plm.common_attributes(self)
|
|
|
|
def get_attr(self, key):
|
|
"""Return specified attribute for this device."""
|
|
return self._plm.get_device_attr(self.address, key)
|
|
|
|
@callback
|
|
def async_switch_update(self, message):
|
|
"""Receive notification from transport that new data exists."""
|
|
_LOGGER.info('Received update calback from PLM for %s', self._address)
|
|
self._hass.async_add_job(self.async_update_ha_state())
|
|
|
|
@asyncio.coroutine
|
|
def async_turn_on(self, **kwargs):
|
|
"""Turn device on."""
|
|
self._plm.turn_on(self._address)
|
|
|
|
@asyncio.coroutine
|
|
def async_turn_off(self, **kwargs):
|
|
"""Turn device off."""
|
|
self._plm.turn_off(self._address)
|