2019-04-03 15:40:03 +00:00
|
|
|
"""Support for collecting data from the ARWN project."""
|
2016-10-14 07:06:04 +00:00
|
|
|
import json
|
|
|
|
import logging
|
2016-12-16 06:12:33 +00:00
|
|
|
|
2018-07-18 09:54:27 +00:00
|
|
|
from homeassistant.components import mqtt
|
2021-03-22 11:37:16 +00:00
|
|
|
from homeassistant.components.sensor import SensorEntity
|
2021-07-12 20:45:29 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
DEGREE,
|
|
|
|
DEVICE_CLASS_TEMPERATURE,
|
2021-07-21 18:14:47 +00:00
|
|
|
PRECIPITATION_INCHES,
|
2021-07-12 20:45:29 +00:00
|
|
|
TEMP_CELSIUS,
|
|
|
|
TEMP_FAHRENHEIT,
|
|
|
|
)
|
2017-03-03 11:09:10 +00:00
|
|
|
from homeassistant.core import callback
|
2016-10-14 07:06:04 +00:00
|
|
|
from homeassistant.util import slugify
|
|
|
|
|
2016-12-16 06:12:33 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DOMAIN = "arwn"
|
2016-10-14 07:06:04 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DATA_ARWN = "arwn"
|
|
|
|
TOPIC = "arwn/#"
|
2016-10-14 07:06:04 +00:00
|
|
|
|
|
|
|
|
|
|
|
def discover_sensors(topic, payload):
|
2017-03-03 11:09:10 +00:00
|
|
|
"""Given a topic, dynamically create the right sensor type.
|
|
|
|
|
|
|
|
Async friendly.
|
|
|
|
"""
|
2019-07-31 19:25:30 +00:00
|
|
|
parts = topic.split("/")
|
|
|
|
unit = payload.get("units", "")
|
2016-10-14 07:06:04 +00:00
|
|
|
domain = parts[1]
|
2019-07-31 19:25:30 +00:00
|
|
|
if domain == "temperature":
|
2016-10-14 07:06:04 +00:00
|
|
|
name = parts[2]
|
2019-07-31 19:25:30 +00:00
|
|
|
if unit == "F":
|
2016-10-14 07:06:04 +00:00
|
|
|
unit = TEMP_FAHRENHEIT
|
|
|
|
else:
|
|
|
|
unit = TEMP_CELSIUS
|
2021-07-12 20:45:29 +00:00
|
|
|
return ArwnSensor(
|
|
|
|
topic, name, "temp", unit, device_class=DEVICE_CLASS_TEMPERATURE
|
|
|
|
)
|
2017-07-11 21:09:05 +00:00
|
|
|
if domain == "moisture":
|
2020-04-04 23:32:58 +00:00
|
|
|
name = f"{parts[2]} Moisture"
|
2020-11-05 13:55:53 +00:00
|
|
|
return ArwnSensor(topic, name, "moisture", unit, "mdi:water-percent")
|
2017-07-11 21:09:05 +00:00
|
|
|
if domain == "rain":
|
2017-10-09 13:41:18 +00:00
|
|
|
if len(parts) >= 3 and parts[2] == "today":
|
2019-07-31 19:25:30 +00:00
|
|
|
return ArwnSensor(
|
2021-07-21 18:14:47 +00:00
|
|
|
topic,
|
|
|
|
"Rain Since Midnight",
|
|
|
|
"since_midnight",
|
|
|
|
PRECIPITATION_INCHES,
|
|
|
|
"mdi:water",
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2020-05-17 21:17:08 +00:00
|
|
|
return (
|
2020-11-05 13:55:53 +00:00
|
|
|
ArwnSensor(topic + "/total", "Total Rainfall", "total", unit, "mdi:water"),
|
|
|
|
ArwnSensor(topic + "/rate", "Rainfall Rate", "rate", unit, "mdi:water"),
|
2020-05-17 21:17:08 +00:00
|
|
|
)
|
2019-07-31 19:25:30 +00:00
|
|
|
if domain == "barometer":
|
2020-11-05 13:55:53 +00:00
|
|
|
return ArwnSensor(topic, "Barometer", "pressure", unit, "mdi:thermometer-lines")
|
2019-07-31 19:25:30 +00:00
|
|
|
if domain == "wind":
|
|
|
|
return (
|
2020-11-05 13:55:53 +00:00
|
|
|
ArwnSensor(
|
|
|
|
topic + "/speed", "Wind Speed", "speed", unit, "mdi:speedometer"
|
|
|
|
),
|
|
|
|
ArwnSensor(topic + "/gust", "Wind Gust", "gust", unit, "mdi:speedometer"),
|
|
|
|
ArwnSensor(
|
|
|
|
topic + "/dir", "Wind Direction", "direction", DEGREE, "mdi:compass"
|
|
|
|
),
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2016-10-14 07:06:04 +00:00
|
|
|
|
|
|
|
|
|
|
|
def _slug(name):
|
2020-02-23 21:38:05 +00:00
|
|
|
return f"sensor.arwn_{slugify(name)}"
|
2016-10-14 07:06:04 +00:00
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
async def async_setup_platform(hass, config, async_add_entities, discovery_info=None):
|
2016-10-14 07:06:04 +00:00
|
|
|
"""Set up the ARWN platform."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2017-03-03 11:09:10 +00:00
|
|
|
@callback
|
2019-03-14 17:58:32 +00:00
|
|
|
def async_sensor_event_received(msg):
|
2016-10-14 07:06:04 +00:00
|
|
|
"""Process events as sensors.
|
|
|
|
|
|
|
|
When a new event on our topic (arwn/#) is received we map it
|
|
|
|
into a known kind of sensor based on topic name. If we've
|
|
|
|
never seen this before, we keep this sensor around in a global
|
|
|
|
cache. If we have seen it before, we update the values of the
|
|
|
|
existing sensor. Either way, we push an ha state update at the
|
|
|
|
end for the new event we've seen.
|
|
|
|
|
|
|
|
This lets us dynamically incorporate sensors without any
|
|
|
|
configuration on our side.
|
|
|
|
"""
|
2019-03-14 17:58:32 +00:00
|
|
|
event = json.loads(msg.payload)
|
|
|
|
sensors = discover_sensors(msg.topic, event)
|
2016-10-14 07:06:04 +00:00
|
|
|
if not sensors:
|
|
|
|
return
|
|
|
|
|
2021-10-22 12:07:19 +00:00
|
|
|
if (store := hass.data.get(DATA_ARWN)) is None:
|
2017-03-03 11:09:10 +00:00
|
|
|
store = hass.data[DATA_ARWN] = {}
|
|
|
|
|
2017-02-09 03:56:44 +00:00
|
|
|
if isinstance(sensors, ArwnSensor):
|
2019-07-31 19:25:30 +00:00
|
|
|
sensors = (sensors,)
|
2017-02-09 03:56:44 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
if "timestamp" in event:
|
|
|
|
del event["timestamp"]
|
2016-10-14 07:06:04 +00:00
|
|
|
|
|
|
|
for sensor in sensors:
|
2017-03-03 11:09:10 +00:00
|
|
|
if sensor.name not in store:
|
2016-10-14 07:06:04 +00:00
|
|
|
sensor.hass = hass
|
|
|
|
sensor.set_event(event)
|
2017-03-03 11:09:10 +00:00
|
|
|
store[sensor.name] = sensor
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.debug(
|
2020-11-05 13:55:53 +00:00
|
|
|
"Registering sensor %(name)s => %(event)s",
|
2020-10-06 13:02:23 +00:00
|
|
|
{"name": sensor.name, "event": event},
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-08-24 14:37:30 +00:00
|
|
|
async_add_entities((sensor,), True)
|
2016-10-14 07:06:04 +00:00
|
|
|
else:
|
2020-11-05 13:55:53 +00:00
|
|
|
_LOGGER.debug(
|
|
|
|
"Recording sensor %(name)s => %(event)s",
|
|
|
|
{"name": sensor.name, "event": event},
|
|
|
|
)
|
2017-03-03 11:09:10 +00:00
|
|
|
store[sensor.name].set_event(event)
|
2016-10-14 07:06:04 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
await mqtt.async_subscribe(hass, TOPIC, async_sensor_event_received, 0)
|
2016-10-14 07:06:04 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
2021-03-22 11:37:16 +00:00
|
|
|
class ArwnSensor(SensorEntity):
|
2016-12-16 06:12:33 +00:00
|
|
|
"""Representation of an ARWN sensor."""
|
2016-10-14 07:06:04 +00:00
|
|
|
|
2021-07-18 21:19:25 +00:00
|
|
|
_attr_should_poll = False
|
|
|
|
|
2021-07-12 20:45:29 +00:00
|
|
|
def __init__(self, topic, name, state_key, units, icon=None, device_class=None):
|
2016-10-14 07:06:04 +00:00
|
|
|
"""Initialize the sensor."""
|
|
|
|
self.entity_id = _slug(name)
|
2021-07-18 21:19:25 +00:00
|
|
|
self._attr_name = name
|
2020-11-05 13:55:53 +00:00
|
|
|
# This mqtt topic for the sensor which is its uid
|
2021-07-18 21:19:25 +00:00
|
|
|
self._attr_unique_id = topic
|
2016-10-14 07:06:04 +00:00
|
|
|
self._state_key = state_key
|
2021-08-11 08:45:05 +00:00
|
|
|
self._attr_native_unit_of_measurement = units
|
2021-07-18 21:19:25 +00:00
|
|
|
self._attr_icon = icon
|
|
|
|
self._attr_device_class = device_class
|
2016-10-14 07:06:04 +00:00
|
|
|
|
|
|
|
def set_event(self, event):
|
|
|
|
"""Update the sensor with the most recent event."""
|
2021-07-18 21:19:25 +00:00
|
|
|
ev = {}
|
|
|
|
ev.update(event)
|
|
|
|
self._attr_extra_state_attributes = ev
|
2021-08-11 08:45:05 +00:00
|
|
|
self._attr_native_value = ev.get(self._state_key, None)
|
2020-04-01 21:19:51 +00:00
|
|
|
self.async_write_ha_state()
|