mirror of
https://github.com/home-assistant/core.git
synced 2025-07-19 11:17:21 +00:00
Merge pull request #781 from balloob/mqtt-light-template
Add template support to MQTT lights
This commit is contained in:
commit
14246fe467
16
.coveragerc
16
.coveragerc
@ -44,24 +44,23 @@ omit =
|
|||||||
homeassistant/components/device_tracker/geofancy.py
|
homeassistant/components/device_tracker/geofancy.py
|
||||||
homeassistant/components/device_tracker/icloud.py
|
homeassistant/components/device_tracker/icloud.py
|
||||||
homeassistant/components/device_tracker/luci.py
|
homeassistant/components/device_tracker/luci.py
|
||||||
homeassistant/components/device_tracker/ubus.py
|
|
||||||
homeassistant/components/device_tracker/netgear.py
|
homeassistant/components/device_tracker/netgear.py
|
||||||
homeassistant/components/device_tracker/nmap_tracker.py
|
homeassistant/components/device_tracker/nmap_tracker.py
|
||||||
homeassistant/components/device_tracker/owntracks.py
|
homeassistant/components/device_tracker/owntracks.py
|
||||||
|
homeassistant/components/device_tracker/snmp.py
|
||||||
homeassistant/components/device_tracker/thomson.py
|
homeassistant/components/device_tracker/thomson.py
|
||||||
homeassistant/components/device_tracker/tomato.py
|
homeassistant/components/device_tracker/tomato.py
|
||||||
homeassistant/components/device_tracker/tplink.py
|
homeassistant/components/device_tracker/tplink.py
|
||||||
homeassistant/components/device_tracker/snmp.py
|
homeassistant/components/device_tracker/ubus.py
|
||||||
homeassistant/components/discovery.py
|
homeassistant/components/discovery.py
|
||||||
homeassistant/components/downloader.py
|
homeassistant/components/downloader.py
|
||||||
homeassistant/components/ifttt.py
|
homeassistant/components/ifttt.py
|
||||||
homeassistant/components/influxdb.py
|
homeassistant/components/influxdb.py
|
||||||
homeassistant/components/keyboard.py
|
homeassistant/components/keyboard.py
|
||||||
homeassistant/components/light/hue.py
|
|
||||||
homeassistant/components/light/mqtt.py
|
|
||||||
homeassistant/components/light/limitlessled.py
|
|
||||||
homeassistant/components/light/blinksticklight.py
|
homeassistant/components/light/blinksticklight.py
|
||||||
|
homeassistant/components/light/hue.py
|
||||||
homeassistant/components/light/hyperion.py
|
homeassistant/components/light/hyperion.py
|
||||||
|
homeassistant/components/light/limitlessled.py
|
||||||
homeassistant/components/media_player/cast.py
|
homeassistant/components/media_player/cast.py
|
||||||
homeassistant/components/media_player/denon.py
|
homeassistant/components/media_player/denon.py
|
||||||
homeassistant/components/media_player/firetv.py
|
homeassistant/components/media_player/firetv.py
|
||||||
@ -69,9 +68,8 @@ omit =
|
|||||||
homeassistant/components/media_player/kodi.py
|
homeassistant/components/media_player/kodi.py
|
||||||
homeassistant/components/media_player/mpd.py
|
homeassistant/components/media_player/mpd.py
|
||||||
homeassistant/components/media_player/plex.py
|
homeassistant/components/media_player/plex.py
|
||||||
homeassistant/components/media_player/squeezebox.py
|
|
||||||
homeassistant/components/media_player/sonos.py
|
homeassistant/components/media_player/sonos.py
|
||||||
homeassistant/components/notify/file.py
|
homeassistant/components/media_player/squeezebox.py
|
||||||
homeassistant/components/notify/instapush.py
|
homeassistant/components/notify/instapush.py
|
||||||
homeassistant/components/notify/nma.py
|
homeassistant/components/notify/nma.py
|
||||||
homeassistant/components/notify/pushbullet.py
|
homeassistant/components/notify/pushbullet.py
|
||||||
@ -84,7 +82,6 @@ omit =
|
|||||||
homeassistant/components/notify/xmpp.py
|
homeassistant/components/notify/xmpp.py
|
||||||
homeassistant/components/sensor/arest.py
|
homeassistant/components/sensor/arest.py
|
||||||
homeassistant/components/sensor/bitcoin.py
|
homeassistant/components/sensor/bitcoin.py
|
||||||
homeassistant/components/sensor/command_sensor.py
|
|
||||||
homeassistant/components/sensor/cpuspeed.py
|
homeassistant/components/sensor/cpuspeed.py
|
||||||
homeassistant/components/sensor/dht.py
|
homeassistant/components/sensor/dht.py
|
||||||
homeassistant/components/sensor/dweet.py
|
homeassistant/components/sensor/dweet.py
|
||||||
@ -105,7 +102,6 @@ omit =
|
|||||||
homeassistant/components/sensor/twitch.py
|
homeassistant/components/sensor/twitch.py
|
||||||
homeassistant/components/sensor/worldclock.py
|
homeassistant/components/sensor/worldclock.py
|
||||||
homeassistant/components/switch/arest.py
|
homeassistant/components/switch/arest.py
|
||||||
homeassistant/components/switch/command_switch.py
|
|
||||||
homeassistant/components/switch/edimax.py
|
homeassistant/components/switch/edimax.py
|
||||||
homeassistant/components/switch/hikvisioncam.py
|
homeassistant/components/switch/hikvisioncam.py
|
||||||
homeassistant/components/switch/mystrom.py
|
homeassistant/components/switch/mystrom.py
|
||||||
@ -114,11 +110,11 @@ omit =
|
|||||||
homeassistant/components/switch/rpi_gpio.py
|
homeassistant/components/switch/rpi_gpio.py
|
||||||
homeassistant/components/switch/transmission.py
|
homeassistant/components/switch/transmission.py
|
||||||
homeassistant/components/switch/wemo.py
|
homeassistant/components/switch/wemo.py
|
||||||
|
homeassistant/components/thermostat/heatmiser.py
|
||||||
homeassistant/components/thermostat/homematic.py
|
homeassistant/components/thermostat/homematic.py
|
||||||
homeassistant/components/thermostat/honeywell.py
|
homeassistant/components/thermostat/honeywell.py
|
||||||
homeassistant/components/thermostat/nest.py
|
homeassistant/components/thermostat/nest.py
|
||||||
homeassistant/components/thermostat/radiotherm.py
|
homeassistant/components/thermostat/radiotherm.py
|
||||||
homeassistant/components/thermostat/heatmiser.py
|
|
||||||
|
|
||||||
|
|
||||||
[report]
|
[report]
|
||||||
|
@ -6,24 +6,24 @@ Allows to configure a MQTT light.
|
|||||||
For more details about this platform, please refer to the documentation at
|
For more details about this platform, please refer to the documentation at
|
||||||
https://home-assistant.io/components/light.mqtt/
|
https://home-assistant.io/components/light.mqtt/
|
||||||
"""
|
"""
|
||||||
|
from functools import partial
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
import homeassistant.components.mqtt as mqtt
|
import homeassistant.components.mqtt as mqtt
|
||||||
from homeassistant.components.light import (Light,
|
from homeassistant.components.light import (Light,
|
||||||
ATTR_BRIGHTNESS, ATTR_RGB_COLOR)
|
ATTR_BRIGHTNESS, ATTR_RGB_COLOR)
|
||||||
|
from homeassistant.util.template import render_with_possible_json_value
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
DEFAULT_NAME = "MQTT Light"
|
DEFAULT_NAME = 'MQTT Light'
|
||||||
DEFAULT_QOS = 0
|
DEFAULT_QOS = 0
|
||||||
DEFAULT_PAYLOAD_ON = "on"
|
DEFAULT_PAYLOAD_ON = 'ON'
|
||||||
DEFAULT_PAYLOAD_OFF = "off"
|
DEFAULT_PAYLOAD_OFF = 'OFF'
|
||||||
DEFAULT_OPTIMISTIC = False
|
DEFAULT_OPTIMISTIC = False
|
||||||
|
|
||||||
DEPENDENCIES = ['mqtt']
|
DEPENDENCIES = ['mqtt']
|
||||||
|
|
||||||
# pylint: disable=unused-argument
|
|
||||||
|
|
||||||
|
|
||||||
def setup_platform(hass, config, add_devices_callback, discovery_info=None):
|
def setup_platform(hass, config, add_devices_callback, discovery_info=None):
|
||||||
""" Add MQTT Light. """
|
""" Add MQTT Light. """
|
||||||
@ -35,18 +35,16 @@ def setup_platform(hass, config, add_devices_callback, discovery_info=None):
|
|||||||
add_devices_callback([MqttLight(
|
add_devices_callback([MqttLight(
|
||||||
hass,
|
hass,
|
||||||
config.get('name', DEFAULT_NAME),
|
config.get('name', DEFAULT_NAME),
|
||||||
{
|
{key: config.get(key) for key in
|
||||||
"state_topic": config.get('state_topic'),
|
(typ + topic
|
||||||
"command_topic": config.get('command_topic'),
|
for typ in ('', 'brightness_', 'rgb_')
|
||||||
"brightness_state_topic": config.get('brightness_state_topic'),
|
for topic in ('state_topic', 'command_topic'))},
|
||||||
"brightness_command_topic": config.get('brightness_command_topic'),
|
{key: config.get(key + '_value_template')
|
||||||
"rgb_state_topic": config.get('rgb_state_topic'),
|
for key in ('state', 'brightness', 'rgb')},
|
||||||
"rgb_command_topic": config.get('rgb_command_topic')
|
|
||||||
},
|
|
||||||
config.get('qos', DEFAULT_QOS),
|
config.get('qos', DEFAULT_QOS),
|
||||||
{
|
{
|
||||||
"on": config.get('payload_on', DEFAULT_PAYLOAD_ON),
|
'on': config.get('payload_on', DEFAULT_PAYLOAD_ON),
|
||||||
"off": config.get('payload_off', DEFAULT_PAYLOAD_OFF)
|
'off': config.get('payload_off', DEFAULT_PAYLOAD_OFF)
|
||||||
},
|
},
|
||||||
config.get('optimistic', DEFAULT_OPTIMISTIC))])
|
config.get('optimistic', DEFAULT_OPTIMISTIC))])
|
||||||
|
|
||||||
@ -55,7 +53,7 @@ class MqttLight(Light):
|
|||||||
""" Provides a MQTT light. """
|
""" Provides a MQTT light. """
|
||||||
|
|
||||||
# pylint: disable=too-many-arguments,too-many-instance-attributes
|
# pylint: disable=too-many-arguments,too-many-instance-attributes
|
||||||
def __init__(self, hass, name, topic, qos, payload, optimistic):
|
def __init__(self, hass, name, topic, templates, qos, payload, optimistic):
|
||||||
|
|
||||||
self._hass = hass
|
self._hass = hass
|
||||||
self._name = name
|
self._name = name
|
||||||
@ -68,8 +66,13 @@ class MqttLight(Light):
|
|||||||
topic["brightness_state_topic"] is None)
|
topic["brightness_state_topic"] is None)
|
||||||
self._state = False
|
self._state = False
|
||||||
|
|
||||||
|
templates = {key: ((lambda value: value) if tpl is None else
|
||||||
|
partial(render_with_possible_json_value, hass, tpl))
|
||||||
|
for key, tpl in templates.items()}
|
||||||
|
|
||||||
def state_received(topic, payload, qos):
|
def state_received(topic, payload, qos):
|
||||||
""" A new MQTT message has been received. """
|
""" A new MQTT message has been received. """
|
||||||
|
payload = templates['state'](payload)
|
||||||
if payload == self._payload["on"]:
|
if payload == self._payload["on"]:
|
||||||
self._state = True
|
self._state = True
|
||||||
elif payload == self._payload["off"]:
|
elif payload == self._payload["off"]:
|
||||||
@ -83,7 +86,7 @@ class MqttLight(Light):
|
|||||||
|
|
||||||
def brightness_received(topic, payload, qos):
|
def brightness_received(topic, payload, qos):
|
||||||
""" A new MQTT message for the brightness has been received. """
|
""" A new MQTT message for the brightness has been received. """
|
||||||
self._brightness = int(payload)
|
self._brightness = int(templates['brightness'](payload))
|
||||||
self.update_ha_state()
|
self.update_ha_state()
|
||||||
|
|
||||||
if self._topic["brightness_state_topic"] is not None:
|
if self._topic["brightness_state_topic"] is not None:
|
||||||
@ -95,7 +98,8 @@ class MqttLight(Light):
|
|||||||
|
|
||||||
def rgb_received(topic, payload, qos):
|
def rgb_received(topic, payload, qos):
|
||||||
""" A new MQTT message has been received. """
|
""" A new MQTT message has been received. """
|
||||||
self._rgb = [int(val) for val in payload.split(',')]
|
self._rgb = [int(val) for val in
|
||||||
|
templates['rgb'](payload).split(',')]
|
||||||
self.update_ha_state()
|
self.update_ha_state()
|
||||||
|
|
||||||
if self._topic["rgb_state_topic"] is not None:
|
if self._topic["rgb_state_topic"] is not None:
|
||||||
|
@ -62,6 +62,15 @@ class TestLightMQTT(unittest.TestCase):
|
|||||||
""" Stop down stuff we started. """
|
""" Stop down stuff we started. """
|
||||||
self.hass.stop()
|
self.hass.stop()
|
||||||
|
|
||||||
|
def test_fail_setup_if_no_command_topic(self):
|
||||||
|
self.assertTrue(light.setup(self.hass, {
|
||||||
|
'light': {
|
||||||
|
'platform': 'mqtt',
|
||||||
|
'name': 'test',
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
self.assertIsNone(self.hass.states.get('light.test'))
|
||||||
|
|
||||||
def test_no_color_or_brightness_if_no_topics(self):
|
def test_no_color_or_brightness_if_no_topics(self):
|
||||||
self.assertTrue(light.setup(self.hass, {
|
self.assertTrue(light.setup(self.hass, {
|
||||||
'light': {
|
'light': {
|
||||||
@ -77,7 +86,7 @@ class TestLightMQTT(unittest.TestCase):
|
|||||||
self.assertIsNone(state.attributes.get('rgb_color'))
|
self.assertIsNone(state.attributes.get('rgb_color'))
|
||||||
self.assertIsNone(state.attributes.get('brightness'))
|
self.assertIsNone(state.attributes.get('brightness'))
|
||||||
|
|
||||||
fire_mqtt_message(self.hass, 'test_light_rgb/status', 'on')
|
fire_mqtt_message(self.hass, 'test_light_rgb/status', 'ON')
|
||||||
self.hass.pool.block_till_done()
|
self.hass.pool.block_till_done()
|
||||||
|
|
||||||
state = self.hass.states.get('light.test')
|
state = self.hass.states.get('light.test')
|
||||||
@ -143,6 +152,39 @@ class TestLightMQTT(unittest.TestCase):
|
|||||||
self.assertEqual([125, 125, 125],
|
self.assertEqual([125, 125, 125],
|
||||||
light_state.attributes.get('rgb_color'))
|
light_state.attributes.get('rgb_color'))
|
||||||
|
|
||||||
|
def test_controlling_state_via_topic_with_templates(self):
|
||||||
|
self.assertTrue(light.setup(self.hass, {
|
||||||
|
'light': {
|
||||||
|
'platform': 'mqtt',
|
||||||
|
'name': 'test',
|
||||||
|
'state_topic': 'test_light_rgb/status',
|
||||||
|
'command_topic': 'test_light_rgb/set',
|
||||||
|
'brightness_state_topic': 'test_light_rgb/brightness/status',
|
||||||
|
'rgb_state_topic': 'test_light_rgb/rgb/status',
|
||||||
|
'state_value_template': '{{ value_json.hello }}',
|
||||||
|
'brightness_value_template': '{{ value_json.hello }}',
|
||||||
|
'rgb_value_template': '{{ value_json.hello | join(",") }}',
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
|
||||||
|
state = self.hass.states.get('light.test')
|
||||||
|
self.assertEqual(STATE_OFF, state.state)
|
||||||
|
self.assertIsNone(state.attributes.get('brightness'))
|
||||||
|
self.assertIsNone(state.attributes.get('rgb_color'))
|
||||||
|
|
||||||
|
fire_mqtt_message(self.hass, 'test_light_rgb/status',
|
||||||
|
'{"hello": "ON"}')
|
||||||
|
fire_mqtt_message(self.hass, 'test_light_rgb/brightness/status',
|
||||||
|
'{"hello": "50"}')
|
||||||
|
fire_mqtt_message(self.hass, 'test_light_rgb/rgb/status',
|
||||||
|
'{"hello": [1, 2, 3]}')
|
||||||
|
self.hass.pool.block_till_done()
|
||||||
|
|
||||||
|
state = self.hass.states.get('light.test')
|
||||||
|
self.assertEqual(STATE_ON, state.state)
|
||||||
|
self.assertEqual(50, state.attributes.get('brightness'))
|
||||||
|
self.assertEqual([1, 2, 3], state.attributes.get('rgb_color'))
|
||||||
|
|
||||||
def test_sending_mqtt_commands_and_optimistic(self):
|
def test_sending_mqtt_commands_and_optimistic(self):
|
||||||
self.assertTrue(light.setup(self.hass, {
|
self.assertTrue(light.setup(self.hass, {
|
||||||
'light': {
|
'light': {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user