2019-04-03 15:40:03 +00:00
|
|
|
"""Support for LaCrosse sensor components."""
|
2017-11-03 07:59:11 +00:00
|
|
|
from datetime import timedelta
|
2018-01-21 06:35:38 +00:00
|
|
|
import logging
|
2017-11-03 07:59:11 +00:00
|
|
|
|
2019-12-04 09:49:55 +00:00
|
|
|
import pylacrosse
|
|
|
|
from serial import SerialException
|
2017-11-03 07:59:11 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2018-01-21 06:35:38 +00:00
|
|
|
from homeassistant.components.sensor import ENTITY_ID_FORMAT, PLATFORM_SCHEMA
|
2017-11-03 07:59:11 +00:00
|
|
|
from homeassistant.const import (
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_DEVICE,
|
|
|
|
CONF_ID,
|
|
|
|
CONF_NAME,
|
|
|
|
CONF_SENSORS,
|
|
|
|
CONF_TYPE,
|
|
|
|
EVENT_HOMEASSISTANT_STOP,
|
|
|
|
TEMP_CELSIUS,
|
2020-02-28 19:46:48 +00:00
|
|
|
UNIT_PERCENTAGE,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-01-21 06:35:38 +00:00
|
|
|
from homeassistant.core import callback
|
|
|
|
import homeassistant.helpers.config_validation as cv
|
2017-11-03 07:59:11 +00:00
|
|
|
from homeassistant.helpers.entity import Entity, async_generate_entity_id
|
|
|
|
from homeassistant.helpers.event import async_track_point_in_utc_time
|
|
|
|
from homeassistant.util import dt as dt_util
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_BAUD = "baud"
|
|
|
|
CONF_DATARATE = "datarate"
|
|
|
|
CONF_EXPIRE_AFTER = "expire_after"
|
|
|
|
CONF_FREQUENCY = "frequency"
|
|
|
|
CONF_JEELINK_LED = "led"
|
|
|
|
CONF_TOGGLE_INTERVAL = "toggle_interval"
|
|
|
|
CONF_TOGGLE_MASK = "toggle_mask"
|
2017-11-03 07:59:11 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DEFAULT_DEVICE = "/dev/ttyUSB0"
|
|
|
|
DEFAULT_BAUD = "57600"
|
2017-11-03 07:59:11 +00:00
|
|
|
DEFAULT_EXPIRE_AFTER = 300
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
TYPES = ["battery", "humidity", "temperature"]
|
|
|
|
|
|
|
|
SENSOR_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_ID): cv.positive_int,
|
|
|
|
vol.Required(CONF_TYPE): vol.In(TYPES),
|
|
|
|
vol.Optional(CONF_EXPIRE_AFTER): cv.positive_int,
|
|
|
|
vol.Optional(CONF_NAME): cv.string,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_SENSORS): cv.schema_with_slug_keys(SENSOR_SCHEMA),
|
|
|
|
vol.Optional(CONF_BAUD, default=DEFAULT_BAUD): cv.string,
|
|
|
|
vol.Optional(CONF_DATARATE): cv.positive_int,
|
|
|
|
vol.Optional(CONF_DEVICE, default=DEFAULT_DEVICE): cv.string,
|
|
|
|
vol.Optional(CONF_FREQUENCY): cv.positive_int,
|
|
|
|
vol.Optional(CONF_JEELINK_LED): cv.boolean,
|
|
|
|
vol.Optional(CONF_TOGGLE_INTERVAL): cv.positive_int,
|
|
|
|
vol.Optional(CONF_TOGGLE_MASK): cv.positive_int,
|
|
|
|
}
|
|
|
|
)
|
2017-11-03 07:59:11 +00:00
|
|
|
|
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
2017-11-03 07:59:11 +00:00
|
|
|
"""Set up the LaCrosse sensors."""
|
|
|
|
|
|
|
|
usb_device = config.get(CONF_DEVICE)
|
|
|
|
baud = int(config.get(CONF_BAUD))
|
|
|
|
expire_after = config.get(CONF_EXPIRE_AFTER)
|
|
|
|
|
|
|
|
_LOGGER.debug("%s %s", usb_device, baud)
|
|
|
|
|
|
|
|
try:
|
|
|
|
lacrosse = pylacrosse.LaCrosse(usb_device, baud)
|
|
|
|
lacrosse.open()
|
|
|
|
except SerialException as exc:
|
|
|
|
_LOGGER.warning("Unable to open serial port: %s", exc)
|
|
|
|
return False
|
|
|
|
|
|
|
|
hass.bus.listen_once(EVENT_HOMEASSISTANT_STOP, lacrosse.close)
|
|
|
|
|
2018-01-15 21:49:17 +00:00
|
|
|
if CONF_JEELINK_LED in config:
|
|
|
|
lacrosse.led_mode_state(config.get(CONF_JEELINK_LED))
|
|
|
|
if CONF_FREQUENCY in config:
|
|
|
|
lacrosse.set_frequency(config.get(CONF_FREQUENCY))
|
|
|
|
if CONF_DATARATE in config:
|
|
|
|
lacrosse.set_datarate(config.get(CONF_DATARATE))
|
|
|
|
if CONF_TOGGLE_INTERVAL in config:
|
|
|
|
lacrosse.set_toggle_interval(config.get(CONF_TOGGLE_INTERVAL))
|
|
|
|
if CONF_TOGGLE_MASK in config:
|
|
|
|
lacrosse.set_toggle_mask(config.get(CONF_TOGGLE_MASK))
|
|
|
|
|
|
|
|
lacrosse.start_scan()
|
|
|
|
|
2017-11-03 07:59:11 +00:00
|
|
|
sensors = []
|
|
|
|
for device, device_config in config[CONF_SENSORS].items():
|
|
|
|
_LOGGER.debug("%s %s", device, device_config)
|
|
|
|
|
|
|
|
typ = device_config.get(CONF_TYPE)
|
|
|
|
sensor_class = TYPE_CLASSES[typ]
|
|
|
|
name = device_config.get(CONF_NAME, device)
|
|
|
|
|
|
|
|
sensors.append(
|
2019-07-31 19:25:30 +00:00
|
|
|
sensor_class(hass, lacrosse, device, name, expire_after, device_config)
|
2017-11-03 07:59:11 +00:00
|
|
|
)
|
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
add_entities(sensors)
|
2017-11-03 07:59:11 +00:00
|
|
|
|
|
|
|
|
|
|
|
class LaCrosseSensor(Entity):
|
|
|
|
"""Implementation of a Lacrosse sensor."""
|
|
|
|
|
|
|
|
_temperature = None
|
|
|
|
_humidity = None
|
|
|
|
_low_battery = None
|
|
|
|
_new_battery = None
|
|
|
|
|
|
|
|
def __init__(self, hass, lacrosse, device_id, name, expire_after, config):
|
|
|
|
"""Initialize the sensor."""
|
|
|
|
self.hass = hass
|
|
|
|
self.entity_id = async_generate_entity_id(
|
2019-07-31 19:25:30 +00:00
|
|
|
ENTITY_ID_FORMAT, device_id, hass=hass
|
|
|
|
)
|
2017-11-03 07:59:11 +00:00
|
|
|
self._config = config
|
|
|
|
self._name = name
|
|
|
|
self._value = None
|
|
|
|
self._expire_after = expire_after
|
|
|
|
self._expiration_trigger = None
|
|
|
|
|
|
|
|
lacrosse.register_callback(
|
2019-07-31 19:25:30 +00:00
|
|
|
int(self._config["id"]), self._callback_lacrosse, None
|
|
|
|
)
|
2017-11-03 07:59:11 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the sensor."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def device_state_attributes(self):
|
|
|
|
"""Return the state attributes."""
|
|
|
|
attributes = {
|
2019-07-31 19:25:30 +00:00
|
|
|
"low_battery": self._low_battery,
|
|
|
|
"new_battery": self._new_battery,
|
2017-11-03 07:59:11 +00:00
|
|
|
}
|
|
|
|
return attributes
|
|
|
|
|
|
|
|
def _callback_lacrosse(self, lacrosse_sensor, user_data):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Handle a function that is called from pylacrosse with new values."""
|
2017-11-03 07:59:11 +00:00
|
|
|
if self._expire_after is not None and self._expire_after > 0:
|
|
|
|
# Reset old trigger
|
|
|
|
if self._expiration_trigger:
|
|
|
|
self._expiration_trigger()
|
|
|
|
self._expiration_trigger = None
|
|
|
|
|
|
|
|
# Set new trigger
|
2019-07-31 19:25:30 +00:00
|
|
|
expiration_at = dt_util.utcnow() + timedelta(seconds=self._expire_after)
|
2017-11-03 07:59:11 +00:00
|
|
|
|
|
|
|
self._expiration_trigger = async_track_point_in_utc_time(
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass, self.value_is_expired, expiration_at
|
|
|
|
)
|
2017-11-03 07:59:11 +00:00
|
|
|
|
2018-03-09 23:53:28 +00:00
|
|
|
self._temperature = lacrosse_sensor.temperature
|
2017-11-03 07:59:11 +00:00
|
|
|
self._humidity = lacrosse_sensor.humidity
|
|
|
|
self._low_battery = lacrosse_sensor.low_battery
|
|
|
|
self._new_battery = lacrosse_sensor.new_battery
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def value_is_expired(self, *_):
|
|
|
|
"""Triggered when value is expired."""
|
|
|
|
self._expiration_trigger = None
|
|
|
|
self._value = None
|
2020-04-01 21:19:51 +00:00
|
|
|
self.async_write_ha_state()
|
2017-11-03 07:59:11 +00:00
|
|
|
|
|
|
|
|
|
|
|
class LaCrosseTemperature(LaCrosseSensor):
|
|
|
|
"""Implementation of a Lacrosse temperature sensor."""
|
|
|
|
|
|
|
|
@property
|
|
|
|
def unit_of_measurement(self):
|
|
|
|
"""Return the unit of measurement."""
|
|
|
|
return TEMP_CELSIUS
|
|
|
|
|
|
|
|
@property
|
|
|
|
def state(self):
|
|
|
|
"""Return the state of the sensor."""
|
|
|
|
return self._temperature
|
|
|
|
|
|
|
|
|
|
|
|
class LaCrosseHumidity(LaCrosseSensor):
|
|
|
|
"""Implementation of a Lacrosse humidity sensor."""
|
|
|
|
|
|
|
|
@property
|
|
|
|
def unit_of_measurement(self):
|
|
|
|
"""Return the unit of measurement."""
|
2020-02-28 19:46:48 +00:00
|
|
|
return UNIT_PERCENTAGE
|
2017-11-03 07:59:11 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def state(self):
|
|
|
|
"""Return the state of the sensor."""
|
|
|
|
return self._humidity
|
|
|
|
|
|
|
|
@property
|
|
|
|
def icon(self):
|
|
|
|
"""Icon to use in the frontend."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return "mdi:water-percent"
|
2017-11-03 07:59:11 +00:00
|
|
|
|
|
|
|
|
|
|
|
class LaCrosseBattery(LaCrosseSensor):
|
|
|
|
"""Implementation of a Lacrosse battery sensor."""
|
|
|
|
|
|
|
|
@property
|
|
|
|
def state(self):
|
|
|
|
"""Return the state of the sensor."""
|
|
|
|
if self._low_battery is None:
|
|
|
|
state = None
|
|
|
|
elif self._low_battery is True:
|
2019-07-31 19:25:30 +00:00
|
|
|
state = "low"
|
2017-11-03 07:59:11 +00:00
|
|
|
else:
|
2019-07-31 19:25:30 +00:00
|
|
|
state = "ok"
|
2017-11-03 07:59:11 +00:00
|
|
|
return state
|
|
|
|
|
|
|
|
@property
|
|
|
|
def icon(self):
|
|
|
|
"""Icon to use in the frontend."""
|
|
|
|
if self._low_battery is None:
|
2019-07-31 19:25:30 +00:00
|
|
|
icon = "mdi:battery-unknown"
|
2017-11-03 07:59:11 +00:00
|
|
|
elif self._low_battery is True:
|
2019-07-31 19:25:30 +00:00
|
|
|
icon = "mdi:battery-alert"
|
2017-11-03 07:59:11 +00:00
|
|
|
else:
|
2019-07-31 19:25:30 +00:00
|
|
|
icon = "mdi:battery"
|
2017-11-03 07:59:11 +00:00
|
|
|
return icon
|
|
|
|
|
|
|
|
|
|
|
|
TYPE_CLASSES = {
|
2019-07-31 19:25:30 +00:00
|
|
|
"temperature": LaCrosseTemperature,
|
|
|
|
"humidity": LaCrosseHumidity,
|
|
|
|
"battery": LaCrosseBattery,
|
2017-11-03 07:59:11 +00:00
|
|
|
}
|