2015-11-21 18:01:47 +00:00
|
|
|
"""
|
2016-02-26 22:52:54 +00:00
|
|
|
A component which allows you to send data to an Influx database.
|
2015-11-21 18:01:47 +00:00
|
|
|
|
|
|
|
For more details about this component, please refer to the documentation at
|
2015-12-07 06:33:07 +00:00
|
|
|
https://home-assistant.io/components/influxdb/
|
2015-11-21 18:01:47 +00:00
|
|
|
"""
|
2017-11-24 00:58:18 +00:00
|
|
|
from datetime import timedelta
|
|
|
|
from functools import wraps, partial
|
2015-11-21 18:01:47 +00:00
|
|
|
import logging
|
Make percentage string values as floats/ints in InfluxDB (#7879)
* Make percentage string values as floats in InfluxDB
Currently Z-wave and other compontents report an attributes battery
level as an integer, for example
```yaml
{
"is_awake": false,
"battery_level": 61,
}
```
However, some other components like Vera add the battery level as a
string
```yaml
{
"Vera Device Id": 25,
"device_armed": "False",
"battery_level": "63%",
"device_tripped": "False",
}
```
By removing any % signs in the field, this will send the value to
InfluxDB as an int, which can then be used to plot the data in graphs
correctly, like other percentage fields.
* Add tests and remove all trailing non digits
Adds tests and now removes all trailing non-numeric characters for
better use
* Update variable name for InfluxDB digit checks
Updates the variable used for the regex to remove trailing non digits
* Fix linting errors for InfluxDB component
Fixes a small linting error on the InfluxDB component
2017-06-13 22:42:55 +00:00
|
|
|
import re
|
|
|
|
|
2017-11-09 19:17:01 +00:00
|
|
|
import requests.exceptions
|
2016-09-18 22:32:18 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.const import (
|
2017-01-14 17:52:47 +00:00
|
|
|
EVENT_STATE_CHANGED, STATE_UNAVAILABLE, STATE_UNKNOWN, CONF_HOST,
|
2017-04-26 19:14:52 +00:00
|
|
|
CONF_PORT, CONF_SSL, CONF_VERIFY_SSL, CONF_USERNAME, CONF_PASSWORD,
|
|
|
|
CONF_EXCLUDE, CONF_INCLUDE, CONF_DOMAINS, CONF_ENTITIES)
|
2016-02-11 17:13:57 +00:00
|
|
|
from homeassistant.helpers import state as state_helper
|
2017-08-03 14:26:01 +00:00
|
|
|
from homeassistant.helpers.entity_values import EntityValues
|
2017-11-24 00:58:18 +00:00
|
|
|
from homeassistant.util import utcnow
|
2016-09-18 22:32:18 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
|
2017-10-02 15:17:08 +00:00
|
|
|
REQUIREMENTS = ['influxdb==4.1.1']
|
2015-11-21 18:01:47 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2016-09-18 22:32:18 +00:00
|
|
|
CONF_DB_NAME = 'database'
|
|
|
|
CONF_TAGS = 'tags'
|
2016-12-02 06:13:55 +00:00
|
|
|
CONF_DEFAULT_MEASUREMENT = 'default_measurement'
|
2017-01-14 17:52:47 +00:00
|
|
|
CONF_OVERRIDE_MEASUREMENT = 'override_measurement'
|
2017-08-03 14:26:01 +00:00
|
|
|
CONF_TAGS_ATTRIBUTES = 'tags_attributes'
|
|
|
|
CONF_COMPONENT_CONFIG = 'component_config'
|
|
|
|
CONF_COMPONENT_CONFIG_GLOB = 'component_config_glob'
|
|
|
|
CONF_COMPONENT_CONFIG_DOMAIN = 'component_config_domain'
|
2017-11-24 00:58:18 +00:00
|
|
|
CONF_RETRY_COUNT = 'max_retries'
|
|
|
|
CONF_RETRY_QUEUE = 'retry_queue_limit'
|
2015-11-21 18:01:47 +00:00
|
|
|
|
2016-09-18 22:32:18 +00:00
|
|
|
DEFAULT_DATABASE = 'home_assistant'
|
2016-12-06 07:39:22 +00:00
|
|
|
DEFAULT_VERIFY_SSL = True
|
2016-09-18 22:32:18 +00:00
|
|
|
DOMAIN = 'influxdb'
|
2016-10-02 05:29:06 +00:00
|
|
|
TIMEOUT = 5
|
2016-09-18 22:32:18 +00:00
|
|
|
|
2017-08-03 14:26:01 +00:00
|
|
|
COMPONENT_CONFIG_SCHEMA_ENTRY = vol.Schema({
|
|
|
|
vol.Optional(CONF_OVERRIDE_MEASUREMENT): cv.string,
|
|
|
|
})
|
|
|
|
|
2016-09-18 22:32:18 +00:00
|
|
|
CONFIG_SCHEMA = vol.Schema({
|
|
|
|
DOMAIN: vol.Schema({
|
2016-12-06 07:39:22 +00:00
|
|
|
vol.Optional(CONF_HOST): cv.string,
|
2016-10-15 04:10:04 +00:00
|
|
|
vol.Inclusive(CONF_USERNAME, 'authentication'): cv.string,
|
|
|
|
vol.Inclusive(CONF_PASSWORD, 'authentication'): cv.string,
|
2017-04-26 19:14:52 +00:00
|
|
|
vol.Optional(CONF_EXCLUDE, default={}): vol.Schema({
|
|
|
|
vol.Optional(CONF_ENTITIES, default=[]): cv.entity_ids,
|
|
|
|
vol.Optional(CONF_DOMAINS, default=[]):
|
|
|
|
vol.All(cv.ensure_list, [cv.string])
|
|
|
|
}),
|
|
|
|
vol.Optional(CONF_INCLUDE, default={}): vol.Schema({
|
|
|
|
vol.Optional(CONF_ENTITIES, default=[]): cv.entity_ids,
|
|
|
|
vol.Optional(CONF_DOMAINS, default=[]):
|
|
|
|
vol.All(cv.ensure_list, [cv.string])
|
|
|
|
}),
|
2016-09-18 22:32:18 +00:00
|
|
|
vol.Optional(CONF_DB_NAME, default=DEFAULT_DATABASE): cv.string,
|
2016-12-06 07:39:22 +00:00
|
|
|
vol.Optional(CONF_PORT): cv.port,
|
|
|
|
vol.Optional(CONF_SSL): cv.boolean,
|
2017-11-24 00:58:18 +00:00
|
|
|
vol.Optional(CONF_RETRY_COUNT, default=0): cv.positive_int,
|
|
|
|
vol.Optional(CONF_RETRY_QUEUE, default=20): cv.positive_int,
|
2017-01-14 17:52:47 +00:00
|
|
|
vol.Optional(CONF_DEFAULT_MEASUREMENT): cv.string,
|
|
|
|
vol.Optional(CONF_OVERRIDE_MEASUREMENT): cv.string,
|
2016-09-18 22:32:18 +00:00
|
|
|
vol.Optional(CONF_TAGS, default={}):
|
|
|
|
vol.Schema({cv.string: cv.string}),
|
2017-08-03 14:26:01 +00:00
|
|
|
vol.Optional(CONF_TAGS_ATTRIBUTES, default=[]):
|
|
|
|
vol.All(cv.ensure_list, [cv.string]),
|
2016-09-18 22:32:18 +00:00
|
|
|
vol.Optional(CONF_VERIFY_SSL, default=DEFAULT_VERIFY_SSL): cv.boolean,
|
2017-08-03 14:26:01 +00:00
|
|
|
vol.Optional(CONF_COMPONENT_CONFIG, default={}):
|
|
|
|
vol.Schema({cv.entity_id: COMPONENT_CONFIG_SCHEMA_ENTRY}),
|
|
|
|
vol.Optional(CONF_COMPONENT_CONFIG_GLOB, default={}):
|
|
|
|
vol.Schema({cv.string: COMPONENT_CONFIG_SCHEMA_ENTRY}),
|
|
|
|
vol.Optional(CONF_COMPONENT_CONFIG_DOMAIN, default={}):
|
|
|
|
vol.Schema({cv.string: COMPONENT_CONFIG_SCHEMA_ENTRY}),
|
2016-09-18 22:32:18 +00:00
|
|
|
}),
|
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
2015-11-21 18:01:47 +00:00
|
|
|
|
2017-06-20 05:53:13 +00:00
|
|
|
RE_DIGIT_TAIL = re.compile(r'^[^\.]*\d+\.?\d+[^\.]*$')
|
|
|
|
RE_DECIMAL = re.compile(r'[^\d.]+')
|
|
|
|
|
2015-11-21 18:01:47 +00:00
|
|
|
|
|
|
|
def setup(hass, config):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Set up the InfluxDB component."""
|
2015-11-25 21:47:00 +00:00
|
|
|
from influxdb import InfluxDBClient, exceptions
|
2015-11-21 18:01:47 +00:00
|
|
|
|
|
|
|
conf = config[DOMAIN]
|
|
|
|
|
2016-12-06 07:39:22 +00:00
|
|
|
kwargs = {
|
|
|
|
'database': conf[CONF_DB_NAME],
|
|
|
|
'verify_ssl': conf[CONF_VERIFY_SSL],
|
|
|
|
'timeout': TIMEOUT
|
|
|
|
}
|
|
|
|
|
|
|
|
if CONF_HOST in conf:
|
|
|
|
kwargs['host'] = conf[CONF_HOST]
|
|
|
|
|
|
|
|
if CONF_PORT in conf:
|
|
|
|
kwargs['port'] = conf[CONF_PORT]
|
|
|
|
|
|
|
|
if CONF_USERNAME in conf:
|
|
|
|
kwargs['username'] = conf[CONF_USERNAME]
|
|
|
|
|
|
|
|
if CONF_PASSWORD in conf:
|
|
|
|
kwargs['password'] = conf[CONF_PASSWORD]
|
|
|
|
|
|
|
|
if CONF_SSL in conf:
|
|
|
|
kwargs['ssl'] = conf[CONF_SSL]
|
|
|
|
|
2017-04-26 19:14:52 +00:00
|
|
|
include = conf.get(CONF_INCLUDE, {})
|
|
|
|
exclude = conf.get(CONF_EXCLUDE, {})
|
|
|
|
whitelist_e = set(include.get(CONF_ENTITIES, []))
|
|
|
|
whitelist_d = set(include.get(CONF_DOMAINS, []))
|
|
|
|
blacklist_e = set(exclude.get(CONF_ENTITIES, []))
|
|
|
|
blacklist_d = set(exclude.get(CONF_DOMAINS, []))
|
2016-09-18 22:32:18 +00:00
|
|
|
tags = conf.get(CONF_TAGS)
|
2017-08-03 14:26:01 +00:00
|
|
|
tags_attributes = conf.get(CONF_TAGS_ATTRIBUTES)
|
2017-01-14 17:52:47 +00:00
|
|
|
default_measurement = conf.get(CONF_DEFAULT_MEASUREMENT)
|
|
|
|
override_measurement = conf.get(CONF_OVERRIDE_MEASUREMENT)
|
2017-08-03 14:26:01 +00:00
|
|
|
component_config = EntityValues(
|
|
|
|
conf[CONF_COMPONENT_CONFIG],
|
|
|
|
conf[CONF_COMPONENT_CONFIG_DOMAIN],
|
|
|
|
conf[CONF_COMPONENT_CONFIG_GLOB])
|
2017-11-24 00:58:18 +00:00
|
|
|
max_tries = conf.get(CONF_RETRY_COUNT)
|
|
|
|
queue_limit = conf.get(CONF_RETRY_QUEUE)
|
2015-11-21 18:01:47 +00:00
|
|
|
|
|
|
|
try:
|
2016-12-06 07:39:22 +00:00
|
|
|
influx = InfluxDBClient(**kwargs)
|
2017-06-08 10:26:37 +00:00
|
|
|
influx.query("SHOW SERIES LIMIT 1;", database=conf[CONF_DB_NAME])
|
2017-11-09 19:17:01 +00:00
|
|
|
except (exceptions.InfluxDBClientError,
|
|
|
|
requests.exceptions.ConnectionError) as exc:
|
2016-01-20 21:43:35 +00:00
|
|
|
_LOGGER.error("Database host is not accessible due to '%s', please "
|
2017-11-09 19:17:01 +00:00
|
|
|
"check your entries in the configuration file (host, "
|
|
|
|
"port, etc.) and verify that the database exists and is "
|
|
|
|
"READ/WRITE.", exc)
|
2015-11-25 21:47:00 +00:00
|
|
|
return False
|
2015-11-21 18:01:47 +00:00
|
|
|
|
2015-12-06 17:45:58 +00:00
|
|
|
def influx_event_listener(event):
|
2016-02-26 22:52:54 +00:00
|
|
|
"""Listen for new messages on the bus and sends them to Influx."""
|
2015-12-06 17:45:58 +00:00
|
|
|
state = event.data.get('new_state')
|
2017-01-14 17:52:47 +00:00
|
|
|
if state is None or state.state in (
|
|
|
|
STATE_UNKNOWN, '', STATE_UNAVAILABLE) or \
|
2017-04-26 19:14:52 +00:00
|
|
|
state.entity_id in blacklist_e or \
|
|
|
|
state.domain in blacklist_d:
|
2017-01-04 21:36:54 +00:00
|
|
|
return
|
2016-08-04 15:35:01 +00:00
|
|
|
|
2017-01-04 21:36:54 +00:00
|
|
|
try:
|
2017-04-26 19:14:52 +00:00
|
|
|
if (whitelist_e and state.entity_id not in whitelist_e) or \
|
|
|
|
(whitelist_d and state.domain not in whitelist_d):
|
2017-01-14 17:52:47 +00:00
|
|
|
return
|
|
|
|
|
2017-11-19 22:49:49 +00:00
|
|
|
_include_state = _include_value = False
|
|
|
|
|
|
|
|
_state_as_value = float(state.state)
|
|
|
|
_include_value = True
|
2016-02-11 17:13:57 +00:00
|
|
|
except ValueError:
|
2017-11-19 22:49:49 +00:00
|
|
|
try:
|
|
|
|
_state_as_value = float(state_helper.state_as_number(state))
|
|
|
|
_include_state = _include_value = True
|
|
|
|
except ValueError:
|
|
|
|
_include_state = True
|
2017-01-14 17:52:47 +00:00
|
|
|
|
2017-11-19 20:30:47 +00:00
|
|
|
include_uom = True
|
2017-08-03 14:26:01 +00:00
|
|
|
measurement = component_config.get(state.entity_id).get(
|
|
|
|
CONF_OVERRIDE_MEASUREMENT)
|
|
|
|
if measurement in (None, ''):
|
|
|
|
if override_measurement:
|
|
|
|
measurement = override_measurement
|
|
|
|
else:
|
|
|
|
measurement = state.attributes.get('unit_of_measurement')
|
|
|
|
if measurement in (None, ''):
|
|
|
|
if default_measurement:
|
|
|
|
measurement = default_measurement
|
|
|
|
else:
|
|
|
|
measurement = state.entity_id
|
2017-11-19 20:30:47 +00:00
|
|
|
else:
|
|
|
|
include_uom = False
|
2015-11-25 21:47:00 +00:00
|
|
|
|
|
|
|
json_body = [
|
|
|
|
{
|
2017-01-14 17:52:47 +00:00
|
|
|
'measurement': measurement,
|
2015-11-25 21:47:00 +00:00
|
|
|
'tags': {
|
|
|
|
'domain': state.domain,
|
|
|
|
'entity_id': state.object_id,
|
|
|
|
},
|
2015-12-06 17:45:58 +00:00
|
|
|
'time': event.time_fired,
|
2015-11-25 21:47:00 +00:00
|
|
|
'fields': {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
]
|
2017-11-19 22:49:49 +00:00
|
|
|
if _include_state:
|
|
|
|
json_body[0]['fields']['state'] = state.state
|
|
|
|
if _include_value:
|
|
|
|
json_body[0]['fields']['value'] = _state_as_value
|
2015-11-21 18:01:47 +00:00
|
|
|
|
2016-09-21 05:20:05 +00:00
|
|
|
for key, value in state.attributes.items():
|
2017-08-03 14:26:01 +00:00
|
|
|
if key in tags_attributes:
|
|
|
|
json_body[0]['tags'][key] = value
|
2017-11-19 20:30:47 +00:00
|
|
|
elif key != 'unit_of_measurement' or include_uom:
|
2017-01-14 17:52:47 +00:00
|
|
|
# If the key is already in fields
|
|
|
|
if key in json_body[0]['fields']:
|
|
|
|
key = key + "_"
|
|
|
|
# Prevent column data errors in influxDB.
|
|
|
|
# For each value we try to cast it as float
|
|
|
|
# But if we can not do it we store the value
|
|
|
|
# as string add "_str" postfix to the field key
|
|
|
|
try:
|
|
|
|
json_body[0]['fields'][key] = float(value)
|
|
|
|
except (ValueError, TypeError):
|
|
|
|
new_key = "{}_str".format(key)
|
2017-06-20 05:53:13 +00:00
|
|
|
new_value = str(value)
|
|
|
|
json_body[0]['fields'][new_key] = new_value
|
|
|
|
|
|
|
|
if RE_DIGIT_TAIL.match(new_value):
|
Make percentage string values as floats/ints in InfluxDB (#7879)
* Make percentage string values as floats in InfluxDB
Currently Z-wave and other compontents report an attributes battery
level as an integer, for example
```yaml
{
"is_awake": false,
"battery_level": 61,
}
```
However, some other components like Vera add the battery level as a
string
```yaml
{
"Vera Device Id": 25,
"device_armed": "False",
"battery_level": "63%",
"device_tripped": "False",
}
```
By removing any % signs in the field, this will send the value to
InfluxDB as an int, which can then be used to plot the data in graphs
correctly, like other percentage fields.
* Add tests and remove all trailing non digits
Adds tests and now removes all trailing non-numeric characters for
better use
* Update variable name for InfluxDB digit checks
Updates the variable used for the regex to remove trailing non digits
* Fix linting errors for InfluxDB component
Fixes a small linting error on the InfluxDB component
2017-06-13 22:42:55 +00:00
|
|
|
json_body[0]['fields'][key] = float(
|
2017-06-20 05:53:13 +00:00
|
|
|
RE_DECIMAL.sub('', new_value))
|
2016-09-21 05:20:05 +00:00
|
|
|
|
2017-01-14 17:52:47 +00:00
|
|
|
json_body[0]['tags'].update(tags)
|
2016-07-26 06:01:57 +00:00
|
|
|
|
2017-11-24 00:58:18 +00:00
|
|
|
_write_data(json_body)
|
|
|
|
|
|
|
|
@RetryOnError(hass, retry_limit=max_tries, retry_delay=20,
|
|
|
|
queue_limit=queue_limit)
|
|
|
|
def _write_data(json_body):
|
2015-12-06 17:45:58 +00:00
|
|
|
try:
|
|
|
|
influx.write_points(json_body)
|
|
|
|
except exceptions.InfluxDBClientError:
|
2017-04-24 03:41:09 +00:00
|
|
|
_LOGGER.exception("Error saving event %s to InfluxDB", json_body)
|
2015-11-21 18:01:47 +00:00
|
|
|
|
2015-12-06 17:45:58 +00:00
|
|
|
hass.bus.listen(EVENT_STATE_CHANGED, influx_event_listener)
|
2015-11-21 18:01:47 +00:00
|
|
|
|
|
|
|
return True
|
2017-11-24 00:58:18 +00:00
|
|
|
|
|
|
|
|
|
|
|
class RetryOnError(object):
|
|
|
|
"""A class for retrying a failed task a certain amount of tries.
|
|
|
|
|
|
|
|
This method decorator makes a method retrying on errors. If there was an
|
|
|
|
uncaught exception, it schedules another try to execute the task after a
|
|
|
|
retry delay. It does this up to the maximum number of retries.
|
|
|
|
|
|
|
|
It can be used for all probable "self-healing" problems like network
|
|
|
|
outages. The task will be rescheduled using HAs scheduling mechanism.
|
|
|
|
|
|
|
|
It takes a Hass instance, a maximum number of retries and a retry delay
|
|
|
|
in seconds as arguments.
|
|
|
|
|
|
|
|
The queue limit defines the maximum number of calls that are allowed to
|
|
|
|
be queued at a time. If this number is reached, every new call discards
|
|
|
|
an old one.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, hass, retry_limit=0, retry_delay=20, queue_limit=100):
|
|
|
|
"""Initialize the decorator."""
|
|
|
|
self.hass = hass
|
|
|
|
self.retry_limit = retry_limit
|
|
|
|
self.retry_delay = timedelta(seconds=retry_delay)
|
|
|
|
self.queue_limit = queue_limit
|
|
|
|
|
|
|
|
def __call__(self, method):
|
|
|
|
"""Decorate the target method."""
|
|
|
|
from homeassistant.helpers.event import track_point_in_utc_time
|
|
|
|
|
|
|
|
@wraps(method)
|
|
|
|
def wrapper(*args, **kwargs):
|
|
|
|
"""Wrapped method."""
|
|
|
|
# pylint: disable=protected-access
|
|
|
|
if not hasattr(wrapper, "_retry_queue"):
|
|
|
|
wrapper._retry_queue = []
|
|
|
|
|
|
|
|
def scheduled(retry=0, untrack=None, event=None):
|
|
|
|
"""Call the target method.
|
|
|
|
|
|
|
|
It is called directly at the first time and then called
|
|
|
|
scheduled within the Hass mainloop.
|
|
|
|
"""
|
|
|
|
if untrack is not None:
|
|
|
|
wrapper._retry_queue.remove(untrack)
|
|
|
|
|
|
|
|
# pylint: disable=broad-except
|
|
|
|
try:
|
|
|
|
method(*args, **kwargs)
|
|
|
|
except Exception as ex:
|
|
|
|
if retry == self.retry_limit:
|
|
|
|
raise
|
|
|
|
if len(wrapper._retry_queue) >= self.queue_limit:
|
|
|
|
last = wrapper._retry_queue.pop(0)
|
|
|
|
if 'remove' in last:
|
|
|
|
func = last['remove']
|
|
|
|
func()
|
|
|
|
if 'exc' in last:
|
|
|
|
_LOGGER.error(
|
|
|
|
"Retry queue overflow, drop oldest entry: %s",
|
|
|
|
str(last['exc']))
|
|
|
|
|
|
|
|
target = utcnow() + self.retry_delay
|
|
|
|
tracking = {'target': target}
|
|
|
|
remove = track_point_in_utc_time(self.hass,
|
|
|
|
partial(scheduled,
|
|
|
|
retry + 1,
|
|
|
|
tracking),
|
|
|
|
target)
|
|
|
|
tracking['remove'] = remove
|
|
|
|
tracking["exc"] = ex
|
|
|
|
wrapper._retry_queue.append(tracking)
|
|
|
|
|
|
|
|
scheduled()
|
|
|
|
return wrapper
|