From cfb318287dcf8b92b08f9d2a831d5e11701420ed Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Daniel=20H=C3=B8yer=20Iversen?= Date: Wed, 6 Apr 2016 19:34:51 +0200 Subject: [PATCH] Refactor rfxtrx component --- homeassistant/components/light/rfxtrx.py | 148 ++----------------- homeassistant/components/rfxtrx.py | 164 +++++++++++++++++++++- homeassistant/components/switch/rfxtrx.py | 132 ++--------------- 3 files changed, 179 insertions(+), 265 deletions(-) diff --git a/homeassistant/components/light/rfxtrx.py b/homeassistant/components/light/rfxtrx.py index a97f98e750a..9d867831074 100644 --- a/homeassistant/components/light/rfxtrx.py +++ b/homeassistant/components/light/rfxtrx.py @@ -8,13 +8,8 @@ import logging import homeassistant.components.rfxtrx as rfxtrx from homeassistant.components.light import ATTR_BRIGHTNESS, Light -from homeassistant.components.rfxtrx import ( - ATTR_FIREEVENT, ATTR_NAME, ATTR_PACKETID, ATTR_STATE, EVENT_BUTTON_PRESSED) -from homeassistant.const import ATTR_ENTITY_ID -from homeassistant.util import slugify DEPENDENCIES = ['rfxtrx'] -SIGNAL_REPETITIONS = 1 _LOGGER = logging.getLogger(__name__) @@ -23,25 +18,7 @@ def setup_platform(hass, config, add_devices_callback, discovery_info=None): """Setup the RFXtrx platform.""" import RFXtrx as rfxtrxmod - lights = [] - signal_repetitions = config.get('signal_repetitions', SIGNAL_REPETITIONS) - - for device_id, entity_info in config.get('devices', {}).items(): - if device_id in rfxtrx.RFX_DEVICES: - continue - _LOGGER.info("Add %s rfxtrx.light", entity_info[ATTR_NAME]) - - # Check if i must fire event - fire_event = entity_info.get(ATTR_FIREEVENT, False) - datas = {ATTR_STATE: False, ATTR_FIREEVENT: fire_event} - - rfxobject = rfxtrx.get_rfx_object(entity_info[ATTR_PACKETID]) - new_light = RfxtrxLight( - entity_info[ATTR_NAME], rfxobject, datas, - signal_repetitions) - rfxtrx.RFX_DEVICES[device_id] = new_light - lights.append(new_light) - + lights = rfxtrx.get_devices_from_config(config, RfxtrxLight) add_devices_callback(lights) def light_update(event): @@ -50,141 +27,32 @@ def setup_platform(hass, config, add_devices_callback, discovery_info=None): not event.device.known_to_be_dimmable: return - # Add entity if not exist and the automatic_add is True - device_id = slugify(event.device.id_string.lower()) - if device_id not in rfxtrx.RFX_DEVICES: - automatic_add = config.get('automatic_add', False) - if not automatic_add: - return + new_device = rfxtrx.get_new_device(event, config, RfxtrxLight) + if new_device: + add_devices_callback([new_device]) - _LOGGER.info( - "Automatic add %s rfxtrx.light (Class: %s Sub: %s)", - device_id, - event.device.__class__.__name__, - event.device.subtype - ) - pkt_id = "".join("{0:02x}".format(x) for x in event.data) - entity_name = "%s : %s" % (device_id, pkt_id) - datas = {ATTR_STATE: False, ATTR_FIREEVENT: False} - signal_repetitions = config.get('signal_repetitions', - SIGNAL_REPETITIONS) - new_light = RfxtrxLight(entity_name, event, datas, - signal_repetitions) - rfxtrx.RFX_DEVICES[device_id] = new_light - add_devices_callback([new_light]) - - # Check if entity exists or previously added automatically - if device_id in rfxtrx.RFX_DEVICES: - _LOGGER.debug( - "EntityID: %s light_update. Command: %s", - device_id, - event.values['Command'] - ) - - if event.values['Command'] == 'On'\ - or event.values['Command'] == 'Off': - - # Update the rfxtrx device state - is_on = event.values['Command'] == 'On' - # pylint: disable=protected-access - rfxtrx.RFX_DEVICES[device_id]._state = is_on - rfxtrx.RFX_DEVICES[device_id].update_ha_state() - - elif event.values['Command'] == 'Set level': - # pylint: disable=protected-access - rfxtrx.RFX_DEVICES[device_id]._brightness = \ - (event.values['Dim level'] * 255 // 100) - - # Update the rfxtrx device state - is_on = rfxtrx.RFX_DEVICES[device_id]._brightness > 0 - rfxtrx.RFX_DEVICES[device_id]._state = is_on - rfxtrx.RFX_DEVICES[device_id].update_ha_state() - else: - return - - # Fire event - if rfxtrx.RFX_DEVICES[device_id].should_fire_event: - rfxtrx.RFX_DEVICES[device_id].hass.bus.fire( - EVENT_BUTTON_PRESSED, { - ATTR_ENTITY_ID: - rfxtrx.RFX_DEVICES[device_id].entity_id, - ATTR_STATE: event.values['Command'].lower() - } - ) + rfxtrx.apply_received_command(event) # Subscribe to main rfxtrx events if light_update not in rfxtrx.RECEIVED_EVT_SUBSCRIBERS: rfxtrx.RECEIVED_EVT_SUBSCRIBERS.append(light_update) -class RfxtrxLight(Light): +class RfxtrxLight(rfxtrx.RfxtrxDevice, Light): """Represenation of a RFXtrx light.""" - def __init__(self, name, event, datas, signal_repetitions): - """Initialize the light.""" - self._name = name - self._event = event - self._state = datas[ATTR_STATE] - self._should_fire_event = datas[ATTR_FIREEVENT] - self.signal_repetitions = signal_repetitions - self._brightness = 0 - - @property - def should_poll(self): - """No polling needed for a light.""" - return False - - @property - def name(self): - """Return the name of the light if any.""" - return self._name - - @property - def should_fire_event(self): - """Return true if the device must fire event.""" - return self._should_fire_event - - @property - def is_on(self): - """Return true if light is on.""" - return self._state - @property def brightness(self): """Return the brightness of this light between 0..255.""" return self._brightness - @property - def assumed_state(self): - """Return True if unable to access real state of entity.""" - return True - def turn_on(self, **kwargs): """Turn the light on.""" brightness = kwargs.get(ATTR_BRIGHTNESS) - if not self._event: - return if brightness is None: self._brightness = 255 - for _ in range(self.signal_repetitions): - self._event.device.send_on(rfxtrx.RFXOBJECT.transport) + self._send_command("turn_on") else: self._brightness = brightness _brightness = (brightness * 100 // 255) - for _ in range(self.signal_repetitions): - self._event.device.send_dim(rfxtrx.RFXOBJECT.transport, - _brightness) - self._state = True - self.update_ha_state() - - def turn_off(self, **kwargs): - """Turn the light off.""" - if not self._event: - return - - for _ in range(self.signal_repetitions): - self._event.device.send_off(rfxtrx.RFXOBJECT.transport) - - self._brightness = 0 - self._state = False - self.update_ha_state() + self._send_command("dim", _brightness) diff --git a/homeassistant/components/rfxtrx.py b/homeassistant/components/rfxtrx.py index f8adaa43223..642fbc1d3a8 100644 --- a/homeassistant/components/rfxtrx.py +++ b/homeassistant/components/rfxtrx.py @@ -8,7 +8,8 @@ import logging from homeassistant.util import slugify from homeassistant.const import EVENT_HOMEASSISTANT_STOP - +from homeassistant.helpers.entity import Entity +from homeassistant.const import ATTR_ENTITY_ID REQUIREMENTS = ['pyRFXtrx==0.6.5'] DOMAIN = "rfxtrx" @@ -21,6 +22,7 @@ ATTR_PACKETID = 'packetid' ATTR_FIREEVENT = 'fire_event' ATTR_DATA_TYPE = 'data_type' ATTR_DUMMY = "dummy" +SIGNAL_REPETITIONS = 1 EVENT_BUTTON_PRESSED = 'button_pressed' @@ -97,3 +99,163 @@ def get_rfx_object(packetid): return obj return None + + +def get_devices_from_config(config, device): + """Read rfxtrx configuration.""" + signal_repetitions = config.get('signal_repetitions', SIGNAL_REPETITIONS) + + devices = [] + for device_id, entity_info in config.get('devices', {}).items(): + if device_id in RFX_DEVICES: + continue + _LOGGER.info("Add %s rfxtrx", entity_info[ATTR_NAME]) + + # Check if i must fire event + fire_event = entity_info.get(ATTR_FIREEVENT, False) + datas = {ATTR_STATE: False, ATTR_FIREEVENT: fire_event} + + rfxobject = get_rfx_object(entity_info[ATTR_PACKETID]) + new_device = device(entity_info[ATTR_NAME], rfxobject, datas, + signal_repetitions) + RFX_DEVICES[device_id] = new_device + devices.append(new_device) + return devices + + +def get_new_device(event, config, device): + """Add entity if not exist and the automatic_add is True.""" + device_id = slugify(event.device.id_string.lower()) + if device_id not in RFX_DEVICES: + automatic_add = config.get('automatic_add', False) + if not automatic_add: + return + + _LOGGER.info( + "Automatic add %s rfxtrx device (Class: %s Sub: %s)", + device_id, + event.device.__class__.__name__, + event.device.subtype + ) + pkt_id = "".join("{0:02x}".format(x) for x in event.data) + entity_name = "%s : %s" % (device_id, pkt_id) + datas = {ATTR_STATE: False, ATTR_FIREEVENT: False} + signal_repetitions = config.get('signal_repetitions', + SIGNAL_REPETITIONS) + new_device = device(entity_name, event, datas, + signal_repetitions) + RFX_DEVICES[device_id] = new_device + return new_device + + +def apply_received_command(event): + """Apply command from rfxtrx.""" + device_id = slugify(event.device.id_string.lower()) + # Check if entity exists or previously added automatically + if device_id in RFX_DEVICES: + _LOGGER.debug( + "EntityID: %s light_update. Command: %s", + device_id, + event.values['Command'] + ) + + if event.values['Command'] == 'On'\ + or event.values['Command'] == 'Off': + + # Update the rfxtrx device state + is_on = event.values['Command'] == 'On' + # pylint: disable=protected-access + RFX_DEVICES[device_id]._state = is_on + RFX_DEVICES[device_id].update_ha_state() + + elif hasattr(RFX_DEVICES[device_id], 'brightness')\ + and event.values['Command'] == 'Set level': + # pylint: disable=protected-access + RFX_DEVICES[device_id]._brightness = \ + (event.values['Dim level'] * 255 // 100) + + # Update the rfxtrx device state + is_on = RFX_DEVICES[device_id]._brightness > 0 + RFX_DEVICES[device_id]._state = is_on + RFX_DEVICES[device_id].update_ha_state() + else: + return + + # Fire event + if RFX_DEVICES[device_id].should_fire_event: + RFX_DEVICES[device_id].hass.bus.fire( + EVENT_BUTTON_PRESSED, { + ATTR_ENTITY_ID: + RFX_DEVICES[device_id].entity_id, + ATTR_STATE: event.values['Command'].lower() + } + ) + + +class RfxtrxDevice(Entity): + """Represents a Rfxtrx device. + + Contains the common logic for all Rfxtrx devices. + + """ + + def __init__(self, name, event, datas, signal_repetitions): + """Initialize the device.""" + self._name = name + self._event = event + self._state = datas[ATTR_STATE] + self._should_fire_event = datas[ATTR_FIREEVENT] + self.signal_repetitions = signal_repetitions + self._brightness = 0 + + @property + def should_poll(self): + """No polling needed for a RFXtrx switch.""" + return False + + @property + def name(self): + """Return the name of the device if any.""" + return self._name + + @property + def should_fire_event(self): + """Return is the device must fire event.""" + return self._should_fire_event + + @property + def is_on(self): + """Return true if light is on.""" + return self._state + + @property + def assumed_state(self): + """Return true if unable to access real state of entity.""" + return True + + def turn_off(self, **kwargs): + """Turn the light off.""" + self._send_command("turn_off") + + def _send_command(self, command, brightness=0): + if not self._event: + return + + if command == "turn_on": + for _ in range(self.signal_repetitions): + self._event.device.send_on(RFXOBJECT.transport) + self._state = True + + elif command == "dim": + for _ in range(self.signal_repetitions): + self._event.device.send_dim(RFXOBJECT.transport, + brightness) + self._state = True + + elif command == 'turn_off': + for _ in range(self.signal_repetitions): + self._event.device.send_off(RFXOBJECT.transport) + self._state = False + self._brightness = 0 + + self.update_ha_state() diff --git a/homeassistant/components/switch/rfxtrx.py b/homeassistant/components/switch/rfxtrx.py index 4d70c01b6cb..df9f42be9b5 100644 --- a/homeassistant/components/switch/rfxtrx.py +++ b/homeassistant/components/switch/rfxtrx.py @@ -7,14 +7,9 @@ https://home-assistant.io/components/switch.rfxtrx/ import logging import homeassistant.components.rfxtrx as rfxtrx -from homeassistant.components.rfxtrx import ( - ATTR_FIREEVENT, ATTR_NAME, ATTR_PACKETID, ATTR_STATE, EVENT_BUTTON_PRESSED) from homeassistant.components.switch import SwitchDevice -from homeassistant.const import ATTR_ENTITY_ID -from homeassistant.util import slugify DEPENDENCIES = ['rfxtrx'] -SIGNAL_REPETITIONS = 1 _LOGGER = logging.getLogger(__name__) @@ -24,25 +19,8 @@ def setup_platform(hass, config, add_devices_callback, discovery_info=None): import RFXtrx as rfxtrxmod # Add switch from config file - switchs = [] - signal_repetitions = config.get('signal_repetitions', SIGNAL_REPETITIONS) - for device_id, entity_info in config.get('devices', {}).items(): - if device_id in rfxtrx.RFX_DEVICES: - continue - _LOGGER.info("Add %s rfxtrx.switch", entity_info[ATTR_NAME]) - - # Check if i must fire event - fire_event = entity_info.get(ATTR_FIREEVENT, False) - datas = {ATTR_STATE: False, ATTR_FIREEVENT: fire_event} - - rfxobject = rfxtrx.get_rfx_object(entity_info[ATTR_PACKETID]) - newswitch = RfxtrxSwitch( - entity_info[ATTR_NAME], rfxobject, datas, - signal_repetitions) - rfxtrx.RFX_DEVICES[device_id] = newswitch - switchs.append(newswitch) - - add_devices_callback(switchs) + switches = rfxtrx.get_devices_from_config(config, RfxtrxSwitch) + add_devices_callback(switches) def switch_update(event): """Callback for sensor updates from the RFXtrx gateway.""" @@ -50,114 +28,20 @@ def setup_platform(hass, config, add_devices_callback, discovery_info=None): event.device.known_to_be_dimmable: return - # Add entity if not exist and the automatic_add is True - device_id = slugify(event.device.id_string.lower()) - if device_id not in rfxtrx.RFX_DEVICES: - automatic_add = config.get('automatic_add', False) - if not automatic_add: - return + new_device = rfxtrx.get_new_device(event, config, RfxtrxSwitch) + if new_device: + add_devices_callback([new_device]) - _LOGGER.info( - "Automatic add %s rfxtrx.switch (Class: %s Sub: %s)", - device_id, - event.device.__class__.__name__, - event.device.subtype - ) - pkt_id = "".join("{0:02x}".format(x) for x in event.data) - entity_name = "%s : %s" % (device_id, pkt_id) - datas = {ATTR_STATE: False, ATTR_FIREEVENT: False} - signal_repetitions = config.get('signal_repetitions', - SIGNAL_REPETITIONS) - new_switch = RfxtrxSwitch(entity_name, event, datas, - signal_repetitions) - rfxtrx.RFX_DEVICES[device_id] = new_switch - add_devices_callback([new_switch]) - - # Check if entity exists or previously added automatically - if device_id in rfxtrx.RFX_DEVICES: - _LOGGER.debug( - "EntityID: %s switch_update. Command: %s", - device_id, - event.values['Command'] - ) - if event.values['Command'] == 'On'\ - or event.values['Command'] == 'Off': - - # Update the rfxtrx device state - is_on = event.values['Command'] == 'On' - # pylint: disable=protected-access - rfxtrx.RFX_DEVICES[device_id]._state = is_on - rfxtrx.RFX_DEVICES[device_id].update_ha_state() - - # Fire event - if rfxtrx.RFX_DEVICES[device_id].should_fire_event: - rfxtrx.RFX_DEVICES[device_id].hass.bus.fire( - EVENT_BUTTON_PRESSED, { - ATTR_ENTITY_ID: - rfxtrx.RFX_DEVICES[device_id].entity_id, - ATTR_STATE: event.values['Command'].lower() - } - ) + rfxtrx.apply_received_command(event) # Subscribe to main rfxtrx events if switch_update not in rfxtrx.RECEIVED_EVT_SUBSCRIBERS: rfxtrx.RECEIVED_EVT_SUBSCRIBERS.append(switch_update) -class RfxtrxSwitch(SwitchDevice): +class RfxtrxSwitch(rfxtrx.RfxtrxDevice, SwitchDevice): """Representation of a RFXtrx switch.""" - def __init__(self, name, event, datas, signal_repetitions): - """Initialize the switch.""" - self._name = name - self._event = event - self._state = datas[ATTR_STATE] - self._should_fire_event = datas[ATTR_FIREEVENT] - self.signal_repetitions = signal_repetitions - - @property - def should_poll(self): - """No polling needed for a RFXtrx switch.""" - return False - - @property - def name(self): - """Return the name of the device if any.""" - return self._name - - @property - def should_fire_event(self): - """Return is the device must fire event.""" - return self._should_fire_event - - @property - def is_on(self): - """Return true if light is on.""" - return self._state - - @property - def assumed_state(self): - """Return true if unable to access real state of entity.""" - return True - def turn_on(self, **kwargs): """Turn the device on.""" - if not self._event: - return - - for _ in range(self.signal_repetitions): - self._event.device.send_on(rfxtrx.RFXOBJECT.transport) - - self._state = True - self.update_ha_state() - - def turn_off(self, **kwargs): - """Turn the device off.""" - if not self._event: - return - - for _ in range(self.signal_repetitions): - self._event.device.send_off(rfxtrx.RFXOBJECT.transport) - - self._state = False - self.update_ha_state() + self._send_command("turn_on")