2016-02-10 16:32:18 +00:00
|
|
|
"""
|
|
|
|
Provides a sensor to track various status aspects of a UPS.
|
2016-02-12 06:54:25 +00:00
|
|
|
|
|
|
|
For more details about this platform, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/sensor.apcupsd/
|
2016-02-10 16:32:18 +00:00
|
|
|
"""
|
|
|
|
import logging
|
|
|
|
|
2016-08-30 19:34:33 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.components.sensor import PLATFORM_SCHEMA
|
|
|
|
import homeassistant.helpers.config_validation as cv
|
2016-02-19 05:27:50 +00:00
|
|
|
from homeassistant.components import apcupsd
|
2016-08-30 19:34:33 +00:00
|
|
|
from homeassistant.const import (TEMP_CELSIUS, CONF_RESOURCES)
|
2016-02-10 16:32:18 +00:00
|
|
|
from homeassistant.helpers.entity import Entity
|
|
|
|
|
2016-08-30 19:34:33 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2016-02-10 16:32:18 +00:00
|
|
|
DEPENDENCIES = [apcupsd.DOMAIN]
|
2016-07-07 01:25:57 +00:00
|
|
|
|
|
|
|
SENSOR_PREFIX = 'UPS '
|
|
|
|
SENSOR_TYPES = {
|
|
|
|
'alarmdel': ['Alarm Delay', '', 'mdi:alarm'],
|
|
|
|
'ambtemp': ['Ambient Temperature', '', 'mdi:thermometer'],
|
|
|
|
'apc': ['Status Data', '', 'mdi:information-outline'],
|
|
|
|
'apcmodel': ['Model', '', 'mdi:information-outline'],
|
|
|
|
'badbatts': ['Bad Batteries', '', 'mdi:information-outline'],
|
|
|
|
'battdate': ['Battery Replaced', '', 'mdi:calendar-clock'],
|
|
|
|
'battstat': ['Battery Status', '', 'mdi:information-outline'],
|
|
|
|
'battv': ['Battery Voltage', 'V', 'mdi:flash'],
|
|
|
|
'bcharge': ['Battery', '%', 'mdi:battery'],
|
|
|
|
'cable': ['Cable Type', '', 'mdi:ethernet-cable'],
|
|
|
|
'cumonbatt': ['Total Time on Battery', '', 'mdi:timer'],
|
|
|
|
'date': ['Status Date', '', 'mdi:calendar-clock'],
|
|
|
|
'dipsw': ['Dip Switch Settings', '', 'mdi:information-outline'],
|
|
|
|
'dlowbatt': ['Low Battery Signal', '', 'mdi:clock-alert'],
|
|
|
|
'driver': ['Driver', '', 'mdi:information-outline'],
|
|
|
|
'dshutd': ['Shutdown Delay', '', 'mdi:timer'],
|
|
|
|
'dwake': ['Wake Delay', '', 'mdi:timer'],
|
|
|
|
'endapc': ['Date and Time', '', 'mdi:calendar-clock'],
|
|
|
|
'extbatts': ['External Batteries', '', 'mdi:information-outline'],
|
|
|
|
'firmware': ['Firmware Version', '', 'mdi:information-outline'],
|
|
|
|
'hitrans': ['Transfer High', 'V', 'mdi:flash'],
|
|
|
|
'hostname': ['Hostname', '', 'mdi:information-outline'],
|
|
|
|
'humidity': ['Ambient Humidity', '%', 'mdi:water-percent'],
|
|
|
|
'itemp': ['Internal Temperature', TEMP_CELSIUS, 'mdi:thermometer'],
|
|
|
|
'lastxfer': ['Last Transfer', '', 'mdi:transfer'],
|
|
|
|
'linefail': ['Input Voltage Status', '', 'mdi:information-outline'],
|
|
|
|
'linefreq': ['Line Frequency', 'Hz', 'mdi:information-outline'],
|
|
|
|
'linev': ['Input Voltage', 'V', 'mdi:flash'],
|
|
|
|
'loadpct': ['Load', '%', 'mdi:gauge'],
|
|
|
|
'lotrans': ['Transfer Low', 'V', 'mdi:flash'],
|
|
|
|
'mandate': ['Manufacture Date', '', 'mdi:calendar'],
|
|
|
|
'masterupd': ['Master Update', '', 'mdi:information-outline'],
|
|
|
|
'maxlinev': ['Input Voltage High', 'V', 'mdi:flash'],
|
|
|
|
'maxtime': ['Battery Timeout', '', 'mdi:timer-off'],
|
|
|
|
'mbattchg': ['Battery Shutdown', '%', 'mdi:battery-alert'],
|
|
|
|
'minlinev': ['Input Voltage Low', 'V', 'mdi:flash'],
|
|
|
|
'mintimel': ['Shutdown Time', '', 'mdi:timer'],
|
|
|
|
'model': ['Model', '', 'mdi:information-outline'],
|
|
|
|
'nombattv': ['Battery Nominal Voltage', 'V', 'mdi:flash'],
|
|
|
|
'nominv': ['Nominal Input Voltage', 'V', 'mdi:flash'],
|
|
|
|
'nomoutv': ['Nominal Output Voltage', 'V', 'mdi:flash'],
|
|
|
|
'nompower': ['Nominal Output Power', 'W', 'mdi:flash'],
|
|
|
|
'numxfers': ['Transfer Count', '', 'mdi:counter'],
|
|
|
|
'outputv': ['Output Voltage', 'V', 'mdi:flash'],
|
|
|
|
'reg1': ['Register 1 Fault', '', 'mdi:information-outline'],
|
|
|
|
'reg2': ['Register 2 Fault', '', 'mdi:information-outline'],
|
|
|
|
'reg3': ['Register 3 Fault', '', 'mdi:information-outline'],
|
|
|
|
'retpct': ['Restore Requirement', '%', 'mdi:battery-alert'],
|
|
|
|
'selftest': ['Last Self Test', '', 'mdi:calendar-clock'],
|
|
|
|
'sense': ['Sensitivity', '', 'mdi:information-outline'],
|
|
|
|
'serialno': ['Serial Number', '', 'mdi:information-outline'],
|
|
|
|
'starttime': ['Startup Time', '', 'mdi:calendar-clock'],
|
|
|
|
'statflag': ['Status Flag', '', 'mdi:information-outline'],
|
|
|
|
'status': ['Status', '', 'mdi:information-outline'],
|
|
|
|
'stesti': ['Self Test Interval', '', 'mdi:information-outline'],
|
|
|
|
'timeleft': ['Time Left', '', 'mdi:clock-alert'],
|
|
|
|
'tonbatt': ['Time on Battery', '', 'mdi:timer'],
|
|
|
|
'upsmode': ['Mode', '', 'mdi:information-outline'],
|
|
|
|
'upsname': ['Name', '', 'mdi:information-outline'],
|
|
|
|
'version': ['Daemon Info', '', 'mdi:information-outline'],
|
|
|
|
'xoffbat': ['Transfer from Battery', '', 'mdi:transfer'],
|
|
|
|
'xoffbatt': ['Transfer from Battery', '', 'mdi:transfer'],
|
|
|
|
'xonbatt': ['Transfer to Battery', '', 'mdi:transfer'],
|
|
|
|
}
|
|
|
|
|
2016-02-10 16:32:18 +00:00
|
|
|
SPECIFIC_UNITS = {
|
2016-07-07 01:25:57 +00:00
|
|
|
'ITEMP': TEMP_CELSIUS
|
|
|
|
}
|
|
|
|
INFERRED_UNITS = {
|
|
|
|
' Minutes': 'min',
|
|
|
|
' Seconds': 'sec',
|
|
|
|
' Percent': '%',
|
|
|
|
' Volts': 'V',
|
|
|
|
' Watts': 'W',
|
|
|
|
' Hz': 'Hz',
|
|
|
|
' C': TEMP_CELSIUS,
|
2017-07-02 09:24:07 +00:00
|
|
|
' Percent Load Capacity': '%',
|
2016-02-10 16:32:18 +00:00
|
|
|
}
|
|
|
|
|
2016-08-30 19:34:33 +00:00
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
|
|
|
vol.Required(CONF_RESOURCES, default=[]):
|
|
|
|
vol.All(cv.ensure_list, [vol.In(SENSOR_TYPES)]),
|
|
|
|
})
|
2016-02-10 16:32:18 +00:00
|
|
|
|
|
|
|
|
|
|
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
2017-05-02 16:18:47 +00:00
|
|
|
"""Set up the APCUPSd sensors."""
|
2016-07-07 01:25:57 +00:00
|
|
|
entities = []
|
|
|
|
|
2016-08-30 19:34:33 +00:00
|
|
|
for resource in config[CONF_RESOURCES]:
|
2016-07-07 01:25:57 +00:00
|
|
|
sensor_type = resource.lower()
|
|
|
|
|
|
|
|
if sensor_type not in SENSOR_TYPES:
|
|
|
|
SENSOR_TYPES[sensor_type] = [
|
|
|
|
sensor_type.title(), '', 'mdi:information-outline']
|
|
|
|
|
|
|
|
if sensor_type.upper() not in apcupsd.DATA.status:
|
|
|
|
_LOGGER.warning(
|
2017-05-02 16:18:47 +00:00
|
|
|
"Sensor type: %s does not appear in the APCUPSd status output",
|
|
|
|
sensor_type)
|
2016-07-07 01:25:57 +00:00
|
|
|
|
|
|
|
entities.append(APCUPSdSensor(apcupsd.DATA, sensor_type))
|
|
|
|
|
|
|
|
add_entities(entities)
|
2016-02-10 16:32:18 +00:00
|
|
|
|
|
|
|
|
|
|
|
def infer_unit(value):
|
2016-03-08 15:46:34 +00:00
|
|
|
"""If the value ends with any of the units from ALL_UNITS.
|
|
|
|
|
|
|
|
Split the unit off the end of the value and return the value, unit tuple
|
|
|
|
pair. Else return the original value and None as the unit.
|
2016-02-10 16:32:18 +00:00
|
|
|
"""
|
|
|
|
from apcaccess.status import ALL_UNITS
|
|
|
|
for unit in ALL_UNITS:
|
|
|
|
if value.endswith(unit):
|
2016-07-07 01:25:57 +00:00
|
|
|
return value[:-len(unit)], INFERRED_UNITS.get(unit, unit.strip())
|
2016-02-10 16:32:18 +00:00
|
|
|
return value, None
|
|
|
|
|
|
|
|
|
2016-07-07 01:25:57 +00:00
|
|
|
class APCUPSdSensor(Entity):
|
2016-03-08 15:46:34 +00:00
|
|
|
"""Representation of a sensor entity for APCUPSd status values."""
|
|
|
|
|
2016-07-07 01:25:57 +00:00
|
|
|
def __init__(self, data, sensor_type):
|
2016-03-08 15:46:34 +00:00
|
|
|
"""Initialize the sensor."""
|
2016-02-11 07:33:53 +00:00
|
|
|
self._data = data
|
2016-07-07 01:25:57 +00:00
|
|
|
self.type = sensor_type
|
|
|
|
self._name = SENSOR_PREFIX + SENSOR_TYPES[sensor_type][0]
|
|
|
|
self._unit = SENSOR_TYPES[sensor_type][1]
|
2016-02-10 16:32:18 +00:00
|
|
|
self._inferred_unit = None
|
2016-02-11 07:33:53 +00:00
|
|
|
self.update()
|
2016-02-10 16:32:18 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
2016-03-08 15:46:34 +00:00
|
|
|
"""Return the name of the UPS sensor."""
|
2016-07-07 01:25:57 +00:00
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def icon(self):
|
|
|
|
"""Icon to use in the frontend, if any."""
|
|
|
|
return SENSOR_TYPES[self.type][2]
|
2016-02-10 16:32:18 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def state(self):
|
2016-03-08 15:46:34 +00:00
|
|
|
"""Return true if the UPS is online, else False."""
|
2016-02-10 16:32:18 +00:00
|
|
|
return self._state
|
|
|
|
|
|
|
|
@property
|
|
|
|
def unit_of_measurement(self):
|
2016-03-08 15:46:34 +00:00
|
|
|
"""Return the unit of measurement of this entity, if any."""
|
2016-07-07 01:25:57 +00:00
|
|
|
if not self._unit:
|
2016-02-10 16:32:18 +00:00
|
|
|
return self._inferred_unit
|
|
|
|
return self._unit
|
|
|
|
|
|
|
|
def update(self):
|
2016-02-23 05:21:49 +00:00
|
|
|
"""Get the latest status and use it to update our sensor state."""
|
2016-07-07 01:25:57 +00:00
|
|
|
if self.type.upper() not in self._data.status:
|
|
|
|
self._state = None
|
|
|
|
self._inferred_unit = None
|
|
|
|
else:
|
|
|
|
self._state, self._inferred_unit = infer_unit(
|
|
|
|
self._data.status[self.type.upper()])
|