2019-02-14 04:35:12 +00:00
|
|
|
"""Support for sending data to an Influx database."""
|
2020-06-30 18:02:25 +00:00
|
|
|
from dataclasses import dataclass
|
2015-11-21 18:01:47 +00:00
|
|
|
import logging
|
2019-12-06 12:05:35 +00:00
|
|
|
import math
|
2018-02-08 11:25:26 +00:00
|
|
|
import queue
|
|
|
|
import threading
|
|
|
|
import time
|
2020-06-30 18:02:25 +00:00
|
|
|
from typing import Any, Callable, Dict, List
|
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
|
|
|
|
2019-12-06 12:05:35 +00:00
|
|
|
from influxdb import InfluxDBClient, exceptions
|
2020-06-12 19:29:46 +00:00
|
|
|
from influxdb_client import InfluxDBClient as InfluxDBClientV2
|
|
|
|
from influxdb_client.client.write_api import ASYNCHRONOUS, SYNCHRONOUS
|
|
|
|
from influxdb_client.rest import ApiException
|
2017-11-09 19:17:01 +00:00
|
|
|
import requests.exceptions
|
2020-06-26 22:01:32 +00:00
|
|
|
import urllib3.exceptions
|
2016-09-18 22:32:18 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.const import (
|
2020-06-30 18:02:25 +00:00
|
|
|
CONF_DOMAIN,
|
|
|
|
CONF_ENTITY_ID,
|
|
|
|
CONF_TIMEOUT,
|
|
|
|
CONF_UNIT_OF_MEASUREMENT,
|
2020-06-12 19:29:46 +00:00
|
|
|
CONF_URL,
|
2019-07-31 19:25:30 +00:00
|
|
|
EVENT_HOMEASSISTANT_STOP,
|
2019-12-06 12:05:35 +00:00
|
|
|
EVENT_STATE_CHANGED,
|
2019-07-31 19:25:30 +00:00
|
|
|
STATE_UNAVAILABLE,
|
|
|
|
STATE_UNKNOWN,
|
|
|
|
)
|
2020-10-23 15:38:46 +00:00
|
|
|
from homeassistant.core import callback
|
2019-12-06 12:05:35 +00:00
|
|
|
from homeassistant.helpers import event as event_helper, state as state_helper
|
2018-01-21 06:35:38 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2017-08-03 14:26:01 +00:00
|
|
|
from homeassistant.helpers.entity_values import EntityValues
|
2020-06-26 22:01:32 +00:00
|
|
|
from homeassistant.helpers.entityfilter import (
|
|
|
|
INCLUDE_EXCLUDE_BASE_FILTER_SCHEMA,
|
|
|
|
convert_include_exclude_filter,
|
|
|
|
)
|
2016-09-18 22:32:18 +00:00
|
|
|
|
2020-06-29 15:31:49 +00:00
|
|
|
from .const import (
|
|
|
|
API_VERSION_2,
|
|
|
|
BATCH_BUFFER_SIZE,
|
|
|
|
BATCH_TIMEOUT,
|
2020-06-30 18:02:25 +00:00
|
|
|
CATCHING_UP_MESSAGE,
|
|
|
|
CLIENT_ERROR_V1,
|
|
|
|
CLIENT_ERROR_V2,
|
|
|
|
CODE_INVALID_INPUTS,
|
2020-06-29 15:31:49 +00:00
|
|
|
COMPONENT_CONFIG_SCHEMA_CONNECTION,
|
|
|
|
CONF_API_VERSION,
|
|
|
|
CONF_BUCKET,
|
|
|
|
CONF_COMPONENT_CONFIG,
|
|
|
|
CONF_COMPONENT_CONFIG_DOMAIN,
|
|
|
|
CONF_COMPONENT_CONFIG_GLOB,
|
|
|
|
CONF_DB_NAME,
|
|
|
|
CONF_DEFAULT_MEASUREMENT,
|
|
|
|
CONF_HOST,
|
2020-07-16 07:42:02 +00:00
|
|
|
CONF_IGNORE_ATTRIBUTES,
|
2020-10-14 20:49:57 +00:00
|
|
|
CONF_MEASUREMENT_ATTR,
|
2020-06-29 15:31:49 +00:00
|
|
|
CONF_ORG,
|
|
|
|
CONF_OVERRIDE_MEASUREMENT,
|
|
|
|
CONF_PASSWORD,
|
|
|
|
CONF_PATH,
|
|
|
|
CONF_PORT,
|
2020-09-19 02:42:03 +00:00
|
|
|
CONF_PRECISION,
|
2020-06-29 15:31:49 +00:00
|
|
|
CONF_RETRY_COUNT,
|
|
|
|
CONF_SSL,
|
2021-02-01 22:29:31 +00:00
|
|
|
CONF_SSL_CA_CERT,
|
2020-06-29 15:31:49 +00:00
|
|
|
CONF_TAGS,
|
|
|
|
CONF_TAGS_ATTRIBUTES,
|
|
|
|
CONF_TOKEN,
|
|
|
|
CONF_USERNAME,
|
|
|
|
CONF_VERIFY_SSL,
|
2020-06-30 18:02:25 +00:00
|
|
|
CONNECTION_ERROR,
|
2020-06-29 15:31:49 +00:00
|
|
|
DEFAULT_API_VERSION,
|
|
|
|
DEFAULT_HOST_V2,
|
2020-10-14 20:49:57 +00:00
|
|
|
DEFAULT_MEASUREMENT_ATTR,
|
2020-06-29 15:31:49 +00:00
|
|
|
DEFAULT_SSL_V2,
|
|
|
|
DOMAIN,
|
2020-06-30 18:02:25 +00:00
|
|
|
EVENT_NEW_STATE,
|
|
|
|
INFLUX_CONF_FIELDS,
|
|
|
|
INFLUX_CONF_MEASUREMENT,
|
|
|
|
INFLUX_CONF_ORG,
|
|
|
|
INFLUX_CONF_STATE,
|
|
|
|
INFLUX_CONF_TAGS,
|
|
|
|
INFLUX_CONF_TIME,
|
|
|
|
INFLUX_CONF_VALUE,
|
|
|
|
QUERY_ERROR,
|
2020-06-29 15:31:49 +00:00
|
|
|
QUEUE_BACKLOG_SECONDS,
|
|
|
|
RE_DECIMAL,
|
|
|
|
RE_DIGIT_TAIL,
|
2020-06-30 18:02:25 +00:00
|
|
|
RESUMED_MESSAGE,
|
2020-06-29 15:31:49 +00:00
|
|
|
RETRY_DELAY,
|
|
|
|
RETRY_INTERVAL,
|
2020-06-30 18:02:25 +00:00
|
|
|
RETRY_MESSAGE,
|
|
|
|
TEST_QUERY_V1,
|
|
|
|
TEST_QUERY_V2,
|
2020-06-29 15:31:49 +00:00
|
|
|
TIMEOUT,
|
|
|
|
WRITE_ERROR,
|
2020-06-30 18:02:25 +00:00
|
|
|
WROTE_MESSAGE,
|
2020-06-29 15:31:49 +00:00
|
|
|
)
|
2015-11-21 18:01:47 +00:00
|
|
|
|
2020-06-29 15:31:49 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2020-06-12 19:29:46 +00:00
|
|
|
|
|
|
|
|
|
|
|
def create_influx_url(conf: Dict) -> Dict:
|
|
|
|
"""Build URL used from config inputs and default when necessary."""
|
|
|
|
if conf[CONF_API_VERSION] == API_VERSION_2:
|
|
|
|
if CONF_SSL not in conf:
|
|
|
|
conf[CONF_SSL] = DEFAULT_SSL_V2
|
|
|
|
if CONF_HOST not in conf:
|
|
|
|
conf[CONF_HOST] = DEFAULT_HOST_V2
|
|
|
|
|
|
|
|
url = conf[CONF_HOST]
|
|
|
|
if conf[CONF_SSL]:
|
|
|
|
url = f"https://{url}"
|
|
|
|
else:
|
|
|
|
url = f"http://{url}"
|
|
|
|
|
|
|
|
if CONF_PORT in conf:
|
|
|
|
url = f"{url}:{conf[CONF_PORT]}"
|
|
|
|
|
|
|
|
if CONF_PATH in conf:
|
|
|
|
url = f"{url}{conf[CONF_PATH]}"
|
|
|
|
|
|
|
|
conf[CONF_URL] = url
|
|
|
|
|
|
|
|
return conf
|
|
|
|
|
|
|
|
|
|
|
|
def validate_version_specific_config(conf: Dict) -> Dict:
|
|
|
|
"""Ensure correct config fields are provided based on API version used."""
|
|
|
|
if conf[CONF_API_VERSION] == API_VERSION_2:
|
|
|
|
if CONF_TOKEN not in conf:
|
|
|
|
raise vol.Invalid(
|
|
|
|
f"{CONF_TOKEN} and {CONF_BUCKET} are required when {CONF_API_VERSION} is {API_VERSION_2}"
|
|
|
|
)
|
|
|
|
|
|
|
|
if CONF_USERNAME in conf:
|
|
|
|
raise vol.Invalid(
|
|
|
|
f"{CONF_USERNAME} and {CONF_PASSWORD} are only allowed when {CONF_API_VERSION} is {DEFAULT_API_VERSION}"
|
|
|
|
)
|
|
|
|
|
|
|
|
else:
|
|
|
|
if CONF_TOKEN in conf:
|
|
|
|
raise vol.Invalid(
|
|
|
|
f"{CONF_TOKEN} and {CONF_BUCKET} are only allowed when {CONF_API_VERSION} is {API_VERSION_2}"
|
|
|
|
)
|
|
|
|
|
|
|
|
return conf
|
|
|
|
|
|
|
|
|
2020-06-30 18:02:25 +00:00
|
|
|
_CUSTOMIZE_ENTITY_SCHEMA = vol.Schema(
|
2020-07-16 07:42:02 +00:00
|
|
|
{
|
|
|
|
vol.Optional(CONF_OVERRIDE_MEASUREMENT): cv.string,
|
|
|
|
vol.Optional(CONF_IGNORE_ATTRIBUTES): vol.All(cv.ensure_list, [cv.string]),
|
|
|
|
}
|
2020-06-30 18:02:25 +00:00
|
|
|
)
|
2020-06-12 19:29:46 +00:00
|
|
|
|
2020-06-30 18:02:25 +00:00
|
|
|
_INFLUX_BASE_SCHEMA = INCLUDE_EXCLUDE_BASE_FILTER_SCHEMA.extend(
|
2020-06-12 19:29:46 +00:00
|
|
|
{
|
|
|
|
vol.Optional(CONF_RETRY_COUNT, default=0): cv.positive_int,
|
|
|
|
vol.Optional(CONF_DEFAULT_MEASUREMENT): cv.string,
|
2020-10-14 20:49:57 +00:00
|
|
|
vol.Optional(CONF_MEASUREMENT_ATTR, default=DEFAULT_MEASUREMENT_ATTR): vol.In(
|
|
|
|
["unit_of_measurement", "domain__device_class", "entity_id"]
|
|
|
|
),
|
2020-06-12 19:29:46 +00:00
|
|
|
vol.Optional(CONF_OVERRIDE_MEASUREMENT): cv.string,
|
|
|
|
vol.Optional(CONF_TAGS, default={}): vol.Schema({cv.string: cv.string}),
|
|
|
|
vol.Optional(CONF_TAGS_ATTRIBUTES, default=[]): vol.All(
|
|
|
|
cv.ensure_list, [cv.string]
|
|
|
|
),
|
2020-07-16 07:42:02 +00:00
|
|
|
vol.Optional(CONF_IGNORE_ATTRIBUTES, default=[]): vol.All(
|
|
|
|
cv.ensure_list, [cv.string]
|
|
|
|
),
|
2020-06-12 19:29:46 +00:00
|
|
|
vol.Optional(CONF_COMPONENT_CONFIG, default={}): vol.Schema(
|
2020-06-30 18:02:25 +00:00
|
|
|
{cv.entity_id: _CUSTOMIZE_ENTITY_SCHEMA}
|
2020-06-12 19:29:46 +00:00
|
|
|
),
|
|
|
|
vol.Optional(CONF_COMPONENT_CONFIG_GLOB, default={}): vol.Schema(
|
2020-06-30 18:02:25 +00:00
|
|
|
{cv.string: _CUSTOMIZE_ENTITY_SCHEMA}
|
2020-06-12 19:29:46 +00:00
|
|
|
),
|
|
|
|
vol.Optional(CONF_COMPONENT_CONFIG_DOMAIN, default={}): vol.Schema(
|
2020-06-30 18:02:25 +00:00
|
|
|
{cv.string: _CUSTOMIZE_ENTITY_SCHEMA}
|
2020-06-12 19:29:46 +00:00
|
|
|
),
|
|
|
|
}
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
|
|
|
|
2020-06-30 18:02:25 +00:00
|
|
|
INFLUX_SCHEMA = vol.All(
|
|
|
|
_INFLUX_BASE_SCHEMA.extend(COMPONENT_CONFIG_SCHEMA_CONNECTION),
|
|
|
|
validate_version_specific_config,
|
|
|
|
create_influx_url,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
|
|
|
|
2020-08-27 11:56:20 +00:00
|
|
|
CONFIG_SCHEMA = vol.Schema(
|
|
|
|
{DOMAIN: INFLUX_SCHEMA},
|
|
|
|
extra=vol.ALLOW_EXTRA,
|
|
|
|
)
|
2015-11-21 18:01:47 +00:00
|
|
|
|
2016-12-06 07:39:22 +00:00
|
|
|
|
2020-06-30 18:02:25 +00:00
|
|
|
def _generate_event_to_json(conf: Dict) -> Callable[[Dict], str]:
|
|
|
|
"""Build event to json converter and add to config."""
|
2020-06-26 22:01:32 +00:00
|
|
|
entity_filter = convert_include_exclude_filter(conf)
|
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)
|
2020-10-14 20:49:57 +00:00
|
|
|
measurement_attr = conf.get(CONF_MEASUREMENT_ATTR)
|
2017-01-14 17:52:47 +00:00
|
|
|
override_measurement = conf.get(CONF_OVERRIDE_MEASUREMENT)
|
2020-07-16 07:42:02 +00:00
|
|
|
global_ignore_attributes = set(conf[CONF_IGNORE_ATTRIBUTES])
|
2017-08-03 14:26:01 +00:00
|
|
|
component_config = EntityValues(
|
|
|
|
conf[CONF_COMPONENT_CONFIG],
|
|
|
|
conf[CONF_COMPONENT_CONFIG_DOMAIN],
|
2019-07-31 19:25:30 +00:00
|
|
|
conf[CONF_COMPONENT_CONFIG_GLOB],
|
|
|
|
)
|
2015-11-21 18:01:47 +00:00
|
|
|
|
2020-06-30 18:02:25 +00:00
|
|
|
def event_to_json(event: Dict) -> str:
|
|
|
|
"""Convert event into json in format Influx expects."""
|
|
|
|
state = event.data.get(EVENT_NEW_STATE)
|
2019-07-31 19:25:30 +00:00
|
|
|
if (
|
|
|
|
state is None
|
|
|
|
or state.state in (STATE_UNKNOWN, "", STATE_UNAVAILABLE)
|
2020-06-26 22:01:32 +00:00
|
|
|
or not entity_filter(state.entity_id)
|
2019-07-31 19:25:30 +00:00
|
|
|
):
|
2018-03-04 05:22:31 +00:00
|
|
|
return
|
2016-08-04 15:35:01 +00:00
|
|
|
|
2017-01-04 21:36:54 +00:00
|
|
|
try:
|
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
|
2020-10-14 20:49:57 +00:00
|
|
|
include_dc = True
|
2020-07-16 07:42:02 +00:00
|
|
|
entity_config = component_config.get(state.entity_id)
|
|
|
|
measurement = entity_config.get(CONF_OVERRIDE_MEASUREMENT)
|
2019-07-31 19:25:30 +00:00
|
|
|
if measurement in (None, ""):
|
2017-08-03 14:26:01 +00:00
|
|
|
if override_measurement:
|
|
|
|
measurement = override_measurement
|
|
|
|
else:
|
2020-10-14 20:49:57 +00:00
|
|
|
if measurement_attr == "entity_id":
|
|
|
|
measurement = state.entity_id
|
|
|
|
elif measurement_attr == "domain__device_class":
|
|
|
|
device_class = state.attributes.get("device_class")
|
|
|
|
if device_class is None:
|
|
|
|
# This entity doesn't have a device_class set, use only domain
|
|
|
|
measurement = state.domain
|
|
|
|
else:
|
|
|
|
measurement = f"{state.domain}__{device_class}"
|
|
|
|
include_dc = False
|
|
|
|
else:
|
|
|
|
measurement = state.attributes.get(measurement_attr)
|
2019-07-31 19:25:30 +00:00
|
|
|
if measurement in (None, ""):
|
2017-08-03 14:26:01 +00:00
|
|
|
if default_measurement:
|
|
|
|
measurement = default_measurement
|
|
|
|
else:
|
|
|
|
measurement = state.entity_id
|
2017-11-19 20:30:47 +00:00
|
|
|
else:
|
2020-10-14 20:49:57 +00:00
|
|
|
include_uom = measurement_attr != "unit_of_measurement"
|
2015-11-25 21:47:00 +00:00
|
|
|
|
2018-03-04 05:22:31 +00:00
|
|
|
json = {
|
2020-06-30 18:02:25 +00:00
|
|
|
INFLUX_CONF_MEASUREMENT: measurement,
|
|
|
|
INFLUX_CONF_TAGS: {
|
|
|
|
CONF_DOMAIN: state.domain,
|
|
|
|
CONF_ENTITY_ID: state.object_id,
|
|
|
|
},
|
|
|
|
INFLUX_CONF_TIME: event.time_fired,
|
|
|
|
INFLUX_CONF_FIELDS: {},
|
2018-03-04 05:22:31 +00:00
|
|
|
}
|
2017-11-19 22:49:49 +00:00
|
|
|
if _include_state:
|
2020-06-30 18:02:25 +00:00
|
|
|
json[INFLUX_CONF_FIELDS][INFLUX_CONF_STATE] = state.state
|
2017-11-19 22:49:49 +00:00
|
|
|
if _include_value:
|
2020-06-30 18:02:25 +00:00
|
|
|
json[INFLUX_CONF_FIELDS][INFLUX_CONF_VALUE] = _state_as_value
|
2015-11-21 18:01:47 +00:00
|
|
|
|
2020-07-16 07:42:02 +00:00
|
|
|
ignore_attributes = set(entity_config.get(CONF_IGNORE_ATTRIBUTES, []))
|
|
|
|
ignore_attributes.update(global_ignore_attributes)
|
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:
|
2020-06-30 18:02:25 +00:00
|
|
|
json[INFLUX_CONF_TAGS][key] = value
|
2020-07-16 07:42:02 +00:00
|
|
|
elif (
|
2020-10-14 20:49:57 +00:00
|
|
|
(key != CONF_UNIT_OF_MEASUREMENT or include_uom)
|
|
|
|
and (key != "device_class" or include_dc)
|
|
|
|
and key not in ignore_attributes
|
|
|
|
):
|
2017-01-14 17:52:47 +00:00
|
|
|
# If the key is already in fields
|
2020-06-30 18:02:25 +00:00
|
|
|
if key in json[INFLUX_CONF_FIELDS]:
|
2020-04-04 23:32:58 +00:00
|
|
|
key = f"{key}_"
|
2017-01-14 17:52:47 +00:00
|
|
|
# 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:
|
2020-06-30 18:02:25 +00:00
|
|
|
json[INFLUX_CONF_FIELDS][key] = float(value)
|
2017-01-14 17:52:47 +00:00
|
|
|
except (ValueError, TypeError):
|
2019-09-03 15:27:14 +00:00
|
|
|
new_key = f"{key}_str"
|
2017-06-20 05:53:13 +00:00
|
|
|
new_value = str(value)
|
2020-06-30 18:02:25 +00:00
|
|
|
json[INFLUX_CONF_FIELDS][new_key] = new_value
|
2017-06-20 05:53:13 +00:00
|
|
|
|
|
|
|
if RE_DIGIT_TAIL.match(new_value):
|
2020-06-30 18:02:25 +00:00
|
|
|
json[INFLUX_CONF_FIELDS][key] = float(
|
|
|
|
RE_DECIMAL.sub("", new_value)
|
|
|
|
)
|
2016-09-21 05:20:05 +00:00
|
|
|
|
2018-05-09 00:54:38 +00:00
|
|
|
# Infinity and NaN are not valid floats in InfluxDB
|
|
|
|
try:
|
2020-06-30 18:02:25 +00:00
|
|
|
if not math.isfinite(json[INFLUX_CONF_FIELDS][key]):
|
|
|
|
del json[INFLUX_CONF_FIELDS][key]
|
2018-05-09 00:54:38 +00:00
|
|
|
except (KeyError, TypeError):
|
|
|
|
pass
|
2018-03-04 20:01:16 +00:00
|
|
|
|
2020-06-30 18:02:25 +00:00
|
|
|
json[INFLUX_CONF_TAGS].update(tags)
|
2016-07-26 06:01:57 +00:00
|
|
|
|
2018-03-04 05:22:31 +00:00
|
|
|
return json
|
2015-11-21 18:01:47 +00:00
|
|
|
|
2020-06-30 18:02:25 +00:00
|
|
|
return event_to_json
|
|
|
|
|
|
|
|
|
|
|
|
@dataclass
|
|
|
|
class InfluxClient:
|
|
|
|
"""An InfluxDB client wrapper for V1 or V2."""
|
2020-06-12 19:29:46 +00:00
|
|
|
|
2020-07-08 19:37:43 +00:00
|
|
|
data_repositories: List[str]
|
2020-06-30 18:02:25 +00:00
|
|
|
write: Callable[[str], None]
|
|
|
|
query: Callable[[str, str], List[Any]]
|
|
|
|
close: Callable[[], None]
|
|
|
|
|
|
|
|
|
|
|
|
def get_influx_connection(conf, test_write=False, test_read=False):
|
|
|
|
"""Create the correct influx connection for the API version."""
|
|
|
|
kwargs = {
|
|
|
|
CONF_TIMEOUT: TIMEOUT,
|
|
|
|
}
|
2020-09-19 02:42:03 +00:00
|
|
|
precision = conf.get(CONF_PRECISION)
|
2020-06-30 18:02:25 +00:00
|
|
|
|
|
|
|
if conf[CONF_API_VERSION] == API_VERSION_2:
|
|
|
|
kwargs[CONF_URL] = conf[CONF_URL]
|
|
|
|
kwargs[CONF_TOKEN] = conf[CONF_TOKEN]
|
|
|
|
kwargs[INFLUX_CONF_ORG] = conf[CONF_ORG]
|
2021-02-01 22:29:31 +00:00
|
|
|
kwargs[CONF_VERIFY_SSL] = conf[CONF_VERIFY_SSL]
|
|
|
|
if CONF_SSL_CA_CERT in conf:
|
|
|
|
kwargs[CONF_SSL_CA_CERT] = conf[CONF_SSL_CA_CERT]
|
2020-06-30 18:02:25 +00:00
|
|
|
bucket = conf.get(CONF_BUCKET)
|
|
|
|
influx = InfluxDBClientV2(**kwargs)
|
|
|
|
query_api = influx.query_api()
|
|
|
|
initial_write_mode = SYNCHRONOUS if test_write else ASYNCHRONOUS
|
|
|
|
write_api = influx.write_api(write_options=initial_write_mode)
|
|
|
|
|
|
|
|
def write_v2(json):
|
|
|
|
"""Write data to V2 influx."""
|
2020-10-16 14:44:50 +00:00
|
|
|
data = {"bucket": bucket, "record": json}
|
|
|
|
|
|
|
|
if precision is not None:
|
|
|
|
data["write_precision"] = precision
|
|
|
|
|
2020-06-30 18:02:25 +00:00
|
|
|
try:
|
2020-10-16 14:44:50 +00:00
|
|
|
write_api.write(**data)
|
2020-06-30 18:02:25 +00:00
|
|
|
except (urllib3.exceptions.HTTPError, OSError) as exc:
|
2020-08-28 11:50:32 +00:00
|
|
|
raise ConnectionError(CONNECTION_ERROR % exc) from exc
|
2020-06-30 18:02:25 +00:00
|
|
|
except ApiException as exc:
|
|
|
|
if exc.status == CODE_INVALID_INPUTS:
|
2020-08-28 11:50:32 +00:00
|
|
|
raise ValueError(WRITE_ERROR % (json, exc)) from exc
|
|
|
|
raise ConnectionError(CLIENT_ERROR_V2 % exc) from exc
|
2020-06-30 18:02:25 +00:00
|
|
|
|
|
|
|
def query_v2(query, _=None):
|
|
|
|
"""Query V2 influx."""
|
|
|
|
try:
|
|
|
|
return query_api.query(query)
|
|
|
|
except (urllib3.exceptions.HTTPError, OSError) as exc:
|
2020-08-28 11:50:32 +00:00
|
|
|
raise ConnectionError(CONNECTION_ERROR % exc) from exc
|
2020-06-30 18:02:25 +00:00
|
|
|
except ApiException as exc:
|
|
|
|
if exc.status == CODE_INVALID_INPUTS:
|
2020-08-28 11:50:32 +00:00
|
|
|
raise ValueError(QUERY_ERROR % (query, exc)) from exc
|
|
|
|
raise ConnectionError(CLIENT_ERROR_V2 % exc) from exc
|
2020-06-30 18:02:25 +00:00
|
|
|
|
|
|
|
def close_v2():
|
|
|
|
"""Close V2 influx client."""
|
|
|
|
influx.close()
|
|
|
|
|
2020-07-08 19:37:43 +00:00
|
|
|
buckets = []
|
2020-06-30 18:02:25 +00:00
|
|
|
if test_write:
|
2020-07-10 14:56:36 +00:00
|
|
|
# Try to write b"" to influx. If we can connect and creds are valid
|
2020-06-30 18:02:25 +00:00
|
|
|
# Then invalid inputs is returned. Anything else is a broken config
|
|
|
|
try:
|
2020-07-10 14:56:36 +00:00
|
|
|
write_v2(b"")
|
2020-06-30 18:02:25 +00:00
|
|
|
except ValueError:
|
|
|
|
pass
|
|
|
|
write_api = influx.write_api(write_options=ASYNCHRONOUS)
|
|
|
|
|
|
|
|
if test_read:
|
2020-07-08 19:37:43 +00:00
|
|
|
tables = query_v2(TEST_QUERY_V2)
|
|
|
|
if tables and tables[0].records:
|
|
|
|
buckets = [bucket.values["name"] for bucket in tables[0].records]
|
|
|
|
else:
|
|
|
|
buckets = []
|
2020-06-30 18:02:25 +00:00
|
|
|
|
2020-07-08 19:37:43 +00:00
|
|
|
return InfluxClient(buckets, write_v2, query_v2, close_v2)
|
2020-06-30 18:02:25 +00:00
|
|
|
|
|
|
|
# Else it's a V1 client
|
2021-02-01 22:29:31 +00:00
|
|
|
if CONF_SSL_CA_CERT in conf and conf[CONF_VERIFY_SSL]:
|
|
|
|
kwargs[CONF_VERIFY_SSL] = conf[CONF_SSL_CA_CERT]
|
|
|
|
else:
|
|
|
|
kwargs[CONF_VERIFY_SSL] = conf[CONF_VERIFY_SSL]
|
2020-06-30 18:02:25 +00:00
|
|
|
|
|
|
|
if CONF_DB_NAME in conf:
|
|
|
|
kwargs[CONF_DB_NAME] = conf[CONF_DB_NAME]
|
|
|
|
|
|
|
|
if CONF_USERNAME in conf:
|
|
|
|
kwargs[CONF_USERNAME] = conf[CONF_USERNAME]
|
|
|
|
|
|
|
|
if CONF_PASSWORD in conf:
|
|
|
|
kwargs[CONF_PASSWORD] = conf[CONF_PASSWORD]
|
|
|
|
|
|
|
|
if CONF_HOST in conf:
|
|
|
|
kwargs[CONF_HOST] = conf[CONF_HOST]
|
|
|
|
|
|
|
|
if CONF_PATH in conf:
|
|
|
|
kwargs[CONF_PATH] = conf[CONF_PATH]
|
|
|
|
|
|
|
|
if CONF_PORT in conf:
|
|
|
|
kwargs[CONF_PORT] = conf[CONF_PORT]
|
|
|
|
|
|
|
|
if CONF_SSL in conf:
|
|
|
|
kwargs[CONF_SSL] = conf[CONF_SSL]
|
|
|
|
|
|
|
|
influx = InfluxDBClient(**kwargs)
|
|
|
|
|
|
|
|
def write_v1(json):
|
|
|
|
"""Write data to V1 influx."""
|
|
|
|
try:
|
2020-09-19 02:42:03 +00:00
|
|
|
influx.write_points(json, time_precision=precision)
|
2020-06-30 18:02:25 +00:00
|
|
|
except (
|
|
|
|
requests.exceptions.RequestException,
|
|
|
|
exceptions.InfluxDBServerError,
|
|
|
|
OSError,
|
|
|
|
) as exc:
|
2020-08-28 11:50:32 +00:00
|
|
|
raise ConnectionError(CONNECTION_ERROR % exc) from exc
|
2020-06-30 18:02:25 +00:00
|
|
|
except exceptions.InfluxDBClientError as exc:
|
|
|
|
if exc.code == CODE_INVALID_INPUTS:
|
2020-08-28 11:50:32 +00:00
|
|
|
raise ValueError(WRITE_ERROR % (json, exc)) from exc
|
|
|
|
raise ConnectionError(CLIENT_ERROR_V1 % exc) from exc
|
2020-06-30 18:02:25 +00:00
|
|
|
|
|
|
|
def query_v1(query, database=None):
|
|
|
|
"""Query V1 influx."""
|
|
|
|
try:
|
|
|
|
return list(influx.query(query, database=database).get_points())
|
|
|
|
except (
|
|
|
|
requests.exceptions.RequestException,
|
|
|
|
exceptions.InfluxDBServerError,
|
|
|
|
OSError,
|
|
|
|
) as exc:
|
2020-08-28 11:50:32 +00:00
|
|
|
raise ConnectionError(CONNECTION_ERROR % exc) from exc
|
2020-06-30 18:02:25 +00:00
|
|
|
except exceptions.InfluxDBClientError as exc:
|
|
|
|
if exc.code == CODE_INVALID_INPUTS:
|
2020-08-28 11:50:32 +00:00
|
|
|
raise ValueError(QUERY_ERROR % (query, exc)) from exc
|
|
|
|
raise ConnectionError(CLIENT_ERROR_V1 % exc) from exc
|
2020-06-30 18:02:25 +00:00
|
|
|
|
|
|
|
def close_v1():
|
|
|
|
"""Close the V1 Influx client."""
|
|
|
|
influx.close()
|
|
|
|
|
2020-07-08 19:37:43 +00:00
|
|
|
databases = []
|
2020-06-30 18:02:25 +00:00
|
|
|
if test_write:
|
2020-07-08 19:37:43 +00:00
|
|
|
write_v1([])
|
2020-06-30 18:02:25 +00:00
|
|
|
|
|
|
|
if test_read:
|
2020-07-08 19:37:43 +00:00
|
|
|
databases = [db["name"] for db in query_v1(TEST_QUERY_V1)]
|
2020-06-30 18:02:25 +00:00
|
|
|
|
2020-07-08 19:37:43 +00:00
|
|
|
return InfluxClient(databases, write_v1, query_v1, close_v1)
|
2020-06-30 18:02:25 +00:00
|
|
|
|
|
|
|
|
|
|
|
def setup(hass, config):
|
|
|
|
"""Set up the InfluxDB component."""
|
|
|
|
conf = config[DOMAIN]
|
|
|
|
try:
|
|
|
|
influx = get_influx_connection(conf, test_write=True)
|
|
|
|
except ConnectionError as exc:
|
|
|
|
_LOGGER.error(RETRY_MESSAGE, exc)
|
|
|
|
event_helper.call_later(hass, RETRY_INTERVAL, lambda _: setup(hass, config))
|
|
|
|
return True
|
|
|
|
|
|
|
|
event_to_json = _generate_event_to_json(conf)
|
|
|
|
max_tries = conf.get(CONF_RETRY_COUNT)
|
|
|
|
instance = hass.data[DOMAIN] = InfluxThread(hass, influx, event_to_json, max_tries)
|
2018-02-08 11:25:26 +00:00
|
|
|
instance.start()
|
2017-11-24 00:58:18 +00:00
|
|
|
|
2018-02-08 11:25:26 +00:00
|
|
|
def shutdown(event):
|
|
|
|
"""Shut down the thread."""
|
|
|
|
instance.queue.put(None)
|
|
|
|
instance.join()
|
2018-03-04 20:01:16 +00:00
|
|
|
influx.close()
|
2017-11-24 00:58:18 +00:00
|
|
|
|
2018-02-08 11:25:26 +00:00
|
|
|
hass.bus.listen_once(EVENT_HOMEASSISTANT_STOP, shutdown)
|
2017-11-24 00:58:18 +00:00
|
|
|
|
2018-02-08 11:25:26 +00:00
|
|
|
return True
|
2017-11-24 00:58:18 +00:00
|
|
|
|
|
|
|
|
2018-02-08 11:25:26 +00:00
|
|
|
class InfluxThread(threading.Thread):
|
|
|
|
"""A threaded event handler class."""
|
2017-11-24 00:58:18 +00:00
|
|
|
|
2020-06-30 18:02:25 +00:00
|
|
|
def __init__(self, hass, influx, event_to_json, max_tries):
|
2018-02-08 11:25:26 +00:00
|
|
|
"""Initialize the listener."""
|
2020-06-30 18:02:25 +00:00
|
|
|
threading.Thread.__init__(self, name=DOMAIN)
|
2018-02-08 11:25:26 +00:00
|
|
|
self.queue = queue.Queue()
|
2018-03-04 05:22:31 +00:00
|
|
|
self.influx = influx
|
|
|
|
self.event_to_json = event_to_json
|
2018-02-08 11:25:26 +00:00
|
|
|
self.max_tries = max_tries
|
2018-03-04 05:22:31 +00:00
|
|
|
self.write_errors = 0
|
|
|
|
self.shutdown = False
|
2018-02-08 11:25:26 +00:00
|
|
|
hass.bus.listen(EVENT_STATE_CHANGED, self._event_listener)
|
2017-11-24 00:58:18 +00:00
|
|
|
|
2020-10-23 15:38:46 +00:00
|
|
|
@callback
|
2018-02-08 11:25:26 +00:00
|
|
|
def _event_listener(self, event):
|
|
|
|
"""Listen for new messages on the bus and queue them for Influx."""
|
|
|
|
item = (time.monotonic(), event)
|
|
|
|
self.queue.put(item)
|
2017-11-24 00:58:18 +00:00
|
|
|
|
2018-03-04 05:22:31 +00:00
|
|
|
@staticmethod
|
|
|
|
def batch_timeout():
|
|
|
|
"""Return number of seconds to wait for more events."""
|
|
|
|
return BATCH_TIMEOUT
|
|
|
|
|
|
|
|
def get_events_json(self):
|
|
|
|
"""Return a batch of events formatted for writing."""
|
2019-07-31 19:25:30 +00:00
|
|
|
queue_seconds = QUEUE_BACKLOG_SECONDS + self.max_tries * RETRY_DELAY
|
2017-11-24 00:58:18 +00:00
|
|
|
|
2018-03-04 05:22:31 +00:00
|
|
|
count = 0
|
|
|
|
json = []
|
2017-11-24 00:58:18 +00:00
|
|
|
|
2018-03-04 05:22:31 +00:00
|
|
|
dropped = 0
|
2017-11-24 00:58:18 +00:00
|
|
|
|
2018-03-04 05:22:31 +00:00
|
|
|
try:
|
|
|
|
while len(json) < BATCH_BUFFER_SIZE and not self.shutdown:
|
|
|
|
timeout = None if count == 0 else self.batch_timeout()
|
|
|
|
item = self.queue.get(timeout=timeout)
|
|
|
|
count += 1
|
|
|
|
|
|
|
|
if item is None:
|
|
|
|
self.shutdown = True
|
|
|
|
else:
|
|
|
|
timestamp, event = item
|
|
|
|
age = time.monotonic() - timestamp
|
|
|
|
|
|
|
|
if age < queue_seconds:
|
|
|
|
event_json = self.event_to_json(event)
|
|
|
|
if event_json:
|
|
|
|
json.append(event_json)
|
|
|
|
else:
|
|
|
|
dropped += 1
|
2017-11-24 00:58:18 +00:00
|
|
|
|
2018-03-04 05:22:31 +00:00
|
|
|
except queue.Empty:
|
|
|
|
pass
|
|
|
|
|
|
|
|
if dropped:
|
2020-06-30 18:02:25 +00:00
|
|
|
_LOGGER.warning(CATCHING_UP_MESSAGE, dropped)
|
2018-03-04 05:22:31 +00:00
|
|
|
|
|
|
|
return count, json
|
|
|
|
|
|
|
|
def write_to_influxdb(self, json):
|
|
|
|
"""Write preprocessed events to influxdb, with retry."""
|
2019-07-31 19:25:30 +00:00
|
|
|
for retry in range(self.max_tries + 1):
|
2018-03-04 05:22:31 +00:00
|
|
|
try:
|
2020-06-30 18:02:25 +00:00
|
|
|
self.influx.write(json)
|
2018-03-04 05:22:31 +00:00
|
|
|
|
|
|
|
if self.write_errors:
|
2020-06-30 18:02:25 +00:00
|
|
|
_LOGGER.error(RESUMED_MESSAGE, self.write_errors)
|
2018-03-04 05:22:31 +00:00
|
|
|
self.write_errors = 0
|
|
|
|
|
2020-06-30 18:02:25 +00:00
|
|
|
_LOGGER.debug(WROTE_MESSAGE, len(json))
|
|
|
|
break
|
|
|
|
except ValueError as err:
|
|
|
|
_LOGGER.error(err)
|
2018-03-04 05:22:31 +00:00
|
|
|
break
|
2020-06-30 18:02:25 +00:00
|
|
|
except ConnectionError as err:
|
2018-03-04 05:22:31 +00:00
|
|
|
if retry < self.max_tries:
|
|
|
|
time.sleep(RETRY_DELAY)
|
|
|
|
else:
|
|
|
|
if not self.write_errors:
|
2020-06-30 18:02:25 +00:00
|
|
|
_LOGGER.error(err)
|
2018-03-04 05:22:31 +00:00
|
|
|
self.write_errors += len(json)
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
"""Process incoming events."""
|
|
|
|
while not self.shutdown:
|
|
|
|
count, json = self.get_events_json()
|
|
|
|
if json:
|
|
|
|
self.write_to_influxdb(json)
|
|
|
|
for _ in range(count):
|
|
|
|
self.queue.task_done()
|
2018-02-08 11:25:26 +00:00
|
|
|
|
|
|
|
def block_till_done(self):
|
|
|
|
"""Block till all events processed."""
|
|
|
|
self.queue.join()
|