This adapter saves state history into InfluxDB.
The Adapter supports InfluxDB 1.x and 2.x
This adapter uses Sentry libraries to automatically report exceptions and code errors to the developers. For more details and for information how to disable the error reporting see Sentry-Plugin Documentation! Sentry reporting is used starting with js-controller 3.0.
If you have an InfluxDB 1.x installation (preferably 1.8.x or 1.9.x) then you choose " 1.x" in the adapter configuration and enter the Host-IP and Port together with Username and Password for the Access. You can also define a database name. The default is "iobroker". On first adapter start this database is created.
When doing custom queries via the "query" message you can use InfluxQL to select the data you want.
Since 2.0 of the adapter also InfluxDB 2.x is supported which works a bit different. Here beside the Host-IP and Port the following data are required:
- Organization: You need to create an organization on the commandline and need to enter the name or ID of that organization here. If you created one Organization when doing the InfluxDB setup you have created an initial organization and can use this here, else use
influx org list
to see available organizations. - Authentication Token: You need to create an Authentication token that have sufficient rights to basically do all actions on the provided organization! Important: For now just use the initial owner auth token because we still struggle on how to create a token that has sufficient permissions. The Owner Token was generated on InfluxDB setup process. If you know how to create the right tokens let us now :-)
You can also define a database name - this is used as Bucket. The default is "iobroker". On first adapter start this bucket is created in the configured organization.
When doing custom queries via the "query" message you can use Flux queries to select the data you want. Details on Flux can be found at https://docs.influxdata.com/influxdb/v2.0/reference/flux/
For Influx 1.x state value, as well as associated metadata fields (q
, ack
and from
) are stored as fields within InfluxDB. When using Flux-commands to retrieve this data (instead of InfluxQL) the data is returned in separate tables, which makes it more difficult to view the data in a joined way when using external database clients or Influx CLI query
command. This is by design, as Influx only supports one field per data-point.
With Influx 2.x it is now also possible, to store this metadata-information besides the actual value of the state as Influx Tags. Tags are indexed and allow for faster search queries. In addition, they are closely linked to the measurement stored within the db, meaning they will be returned in one table with the queried measurement which makes them much easier to handle, when used outside this adapter. There are limitations with this however:
- The use of tags for metadata needs to be enabled under
Advanced Settings
. It is disabled by default. - If you decide to use tags, you cannot continue to use old measurements that were gathered with Influx 1.x, as they are stored in fields.
- Attempting to use an existing database that was set up without enabling the Tag-feature will cause the adapter to fail to initialize, and you will see an error message about it in the log.
- This also is valid the other way: Once you start using the Tag-feature in a new database, you cannot switch back to using fields for this database.
- This feature is currently only available, if you use Influx 2.x. And only if you use the new responsive GUI of Admin 5.
Please refer to the official guides on how to migrate from InfluxDB 1.x to 2.x. Especially the migration instructions for time series data have been verified to work during adapter testing. Please always create a backup of your data before performing the migration.
After the migration, the adapter is able to work with the old data (e.g. for history-queries) as well. This only applies, if you don't decide to use Tags for storing metadata fields.
While Influx 1.x supports the concept of multiple retention policies for one database, Influx 2 by design allows only one retention period per bucket. Therefore, it is only possible to set one policy for the whole database/bucket with this adapter via Default Settings -> Storage retention. The retention selected here will be applied on the fly and can be changed at any time. Retention policies set by the adapter will never be deleted, but instead altered if required, as otherwise Influx 1.x would delete all data that the policy applied to.
Please also read Understanding Retention Policies.
With the default configuration the adapter stores each single datapoint directly into the database and only use the internal buffer if the database is not available. If the database was not available the buffer is flushed at the given interval, so it can take the defined interval till the missing points are written!
By changing the configuration it is possible to cache new data points up to a defined count or a defined maximum interval after which all points are stored into the database. This also gives better performance and less system load compared to writing the data points directly. InfluxDB has a limitation of the maximum size for writes which is at around 2MB. It should be safe to have up to 15.000 data points as buffer maximum, maybe also 20.000, but this highly depends on the length of your datapoint-IDs.
On exit of the adapter the buffer is stored on disk and reinitialized with next adapter start, so no data points should be lost and will be written after next start.
InfluxDB is very strict on data types. The datatype for a measurement value is defined with it's first write. The adapter tries to write with the correct value, but if the datatype changes for the same state then there may be the write errors in the InfluxDB. The adapter detects this and will write these potential conflicting data points always directly, but write errors mean that the value is not written into the DB at all. So make sure to check the logs for such cases.
Additionally, InfluxDB do not support "null" values, so these are not written at all into the DB.
Please refer to the official InfluxDB pages for installation instructions depending on your OS.
- InfluxDB 1.x: https://docs.influxdata.com/influxdb/v1.8/introduction/install/
- InfluxDB 2.x: https://docs.influxdata.com/influxdb/v2.0/install/
NOTE: Influx DB V2.x relies on organization/token login, instead of username/password! This is only applicable for InfluxDB 1.x
If you use DB locally you may leave authentication disabled and skip this part.
- Start service:
service influxdb start
- Go to admin page: http://:8083
- Create users:
CREATE USER "admin" WITH PASSWORD '<adminpassword>' WITH ALL PRIVILEGES
CREATE USER "user" WITH PASSWORD '<userpassword>'
CREATE DATABASE "iobroker"
GRANT ALL ON "iobroker" TO "user"
Enable authentication, by editing /etc/influxdb/influxdb.conf:
[http]
enabled = true
bind-address = ":8086"
auth-enabled = true # ✨
log-enabled = true
write-tracing = false
pprof-enabled = false
https-enabled = false
https-certificate = "/etc/ssl/influxdb.pem"
- Restart service:
service influxdb restart
There is additional charting tool for InfluxDB - Grafana. It must be installed additionally.
Install a current version of Grafana 3.x+ because InfluxDB support is enhanced there in comparison to earlier Grafana versions.
Under debian you can install it as described at http://docs.grafana.org/installation/debian/ . For ARM platforms your can check vor v3.x at https://github.com/fg2it/grafana-on-raspberry.
Explanation for other OS can be found here.
After the Grafana is installed, follow this to create connection.
The user can execute custom queries on data from javascript adapter.
The multi-query feature is also supported. You can send multiple queries separated by a semicolon.
That's why the result is always an array with one numbered index for each query.
Example with one query:
sendTo('influxdb.0', 'query', 'SELECT * FROM iobroker.global."system.adapter.admin.0.memRss" LIMIT 100', function (result) {
if (result.error) {
console.error(result.error);
} else {
// show result
console.log('Rows: ' + JSON.stringify(result.result[0]));
}
});
Two queries:
sendTo('influxdb.0', 'query', 'SELECT * FROM iobroker.global."system.adapter.admin.0.memRss" LIMIT 100; SELECT * FROM iobroker.global."system.adapter.admin.0.memHeapUsed" LIMIT 100', function (result) {
if (result.error) {
console.error(result.error);
} else {
// show result
console.log('Rows First: ' + JSON.stringify(result.result[0]));
console.log('Rows Second: ' + JSON.stringify(result.result[1]));
}
});
NOTE: The values are coming back in the result array in filename "value" (instead of "val" as normal in ioBroker)
In InfluxDB v2.0 onwards the SQL-based query language InfluxQL is deprecated in favour of Flux. For more information please refer to the offical InfluxDB 2.0 documentation.
Example with one query:
sendTo('influxdb.0', 'query', 'from(bucket: "iobroker") |> range(start: -3h)', function (result) {
if (result.error) {
console.error(result.error);
} else {
// show result
console.log('Rows: ' + JSON.stringify(result));
}
});
Two queries: NOTE: By default you cannot execute 2 queries at once via Flux-language, as there is no delimiter available. This adapter emulates this behaviour by defining ; as delimiter, so you can still run two queries in one statement.
sendTo('influxdb.0', 'query', 'from(bucket: "iobroker") |> range(start: -3h); from(bucket: "iobroker") |> range(start: -1h)" LIMIT 100', function (result) {
if (result.error) {
console.error(result.error);
} else {
// show result
console.log('Rows First: ' + JSON.stringify(result.result[0])); //Values from last 3 hours
console.log('Rows Second: ' + JSON.stringify(result.result[1])); //Values from last hour
}
});
NOTE: The values are coming back in the result array in filename "value" (instead of "val" as normal in ioBroker)
Additional to custom queries, you can use build in system function getHistory to access the stored history for data points:
var end = new Date().getTime();
sendTo('influxdb.0', 'getHistory', {
id: 'system.adapter.admin.0.memRss',
options: {
start: end - 3600000,
end: end,
aggregate: 'average' // or 'none' to get raw values
}
}, function (result) {
for (var i = 0; i < result.result.length; i++) {
console.log(result.result[i].val + ' ' + new Date(result.result[i].ts).toISOString());
}
});
Possible options:
- start - (optional) time in ms - new Date().getTime()'
- end - (optional) time in ms - new Date().getTime()', by default is (now + 5000 seconds)
- step - (required if aggregate is not "none") used for aggregate functions (max, min, average, total, count) step in ms of intervals
- count - (optional) number of values if aggregate is 'onchange'/'none' or number of intervals if other aggregate method. Count will be ignored if step is set.
- limit - do not return more entries than limit (only used if aggregate is 'onchange'/'none')
- addId - if id field should be included in answer
- aggregate - aggregate method:
- minmax - Splice the whole time range in small intervals and find for every interval max value and min value and use them for this interval (nulls will be ignored).
- max - Splice the whole time range in small intervals and find for every interval max value and use it for this interval (nulls will be ignored).
- min - Same as max, but take minimal value.
- average - Same as max, but take average value.
- total - Same as max, but calculate total value.
- count - Same as max, but calculate number of values (nulls will be calculated).
- none/onchange - No aggregation at all. Only raw values in given period.
When raw data are selected without using 'step' the returned fields are ts, val, ack, q and from. As soon as step is used the returned fields are ts and val.
Interpolated values will be marked as i=true
, like: {i: true, val: 4.7384845, ts: 29892365723652}
.
Please hold in mind that InfluxDB aggregates on "rounded time boundaries" (see https://docs.influxdata.com/influxdb/v0.11/troubleshooting/frequently_encountered_issues/#understanding-the-time-intervals-returned-from-group-by-time-queries)
InfluxDB is very strict when it comes to data types. This has effects for aggregator functions, e.g.:
- average (MEAN) can not be used for boolean values (true/false), only MIN or MAX works here
- average (MEAN) can not be used for string values (text), no aggregator makes sense here at all
- ...
If you want to write other data into the InfluxDB you can use the build in system function storeState. This function can also be used to convert data from other History adapters like History or SQL.
The given ids are not checked against the ioBroker database and do not need to be set up there, but can only be accessed directly.
The Message can have one of the following three formats:
- one ID and one state object
- one ID and array of state objects
- array of multiple IDs with state objects
If you want to delete entry from the Database you can use the build in system function delete:
sendTo('sql.0', 'delete', [
{id: 'mbus.0.counter.xxx', state: {ts: 1589458809352},
{id: 'mbus.0.counter.yyy', state: {ts: 1589458809353}
], result => console.log('deleted'));
To delete ALL history data for some data point execute:
sendTo('sql.0', 'deleteAll', [
{id: 'mbus.0.counter.xxx'}
{id: 'mbus.0.counter.yyy'}
], result => console.log('deleted'));
To delete history data for some data point and for some range, execute:
sendTo('sql.0', 'deleteRange', [
{id: 'mbus.0.counter.xxx', start: '2019-01-01T00:00:00.000Z', end: '2019-12-31T23:59:59.999'},
{id: 'mbus.0.counter.yyy', start: 1589458809352, end: 1589458809353}
], result => console.log('deleted'));
Time could be ms since epoch or ans string, that could be converted by javascript Date object.
Values will be deleted including defined limits. ts >= start AND ts <= end
If you want to change entry's value, quality or acknowledge flag in the database you can use the build in system function update:
sendTo('sql.0', 'update', [
{id: 'mbus.0.counter.xxx', state: {ts: 1589458809352, val: 15, ack: true, q: 0},
{id: 'mbus.0.counter.yyy', state: {ts: 1589458809353, val: 16, ack: true, q: 0}
], result => console.log('deleted'));
ts
is mandatory. At least one other flags must be included in state object.
The adapter supports enabling and disabling of history logging via JavaScript and also retrieving the list of enabled data points with their settings.
The message requires to have the "id" of the datapoint. Additionally, optional "options" to define the datapoint specific settings:
sendTo('influxdb.0', 'enableHistory', {
id: 'system.adapter.influxdb.0.memRss',
options: {
changesOnly: true,
debounce: 0,
retention: 31536000,
maxLength: 3,
changesMinDelta: 0.5,
aliasId: ''
}
}, function (result) {
if (result.error) {
console.log(result.error);
}
if (result.success) {
// successfully enabled
}
});
The message requires to have the "id" of the datapoint.
sendTo('influxdb.0', 'disableHistory', {
id: 'system.adapter.influxdb.0.memRss',
}, function (result) {
if (result.error) {
console.log(result.error);
}
if (result.success) {
// successfully enabled
}
});
The message has no parameters.
sendTo('influxdb.0', 'getEnabledDPs', {}, function (result) {
//result is object like:
{
"system.adapter.influxdb.0.memRss": {
"changesOnly":true,
"debounce":0,
"retention":31536000,
"maxLength":3,
"changesMinDelta":0.5,
"enabled":true,
"changesRelogInterval":0,
"aliasId":""
}
...
}
});
- (Apollon77) Add flag returnNewestEntries for GetHistory to determine which records to return when more entries as "count" are existing for aggregate "none"
- (Apollon77) Flush the buffer for the datapoint which is queried by GetHistory before the query is done, so that all data are in the database
- (Apollon77) make sure id is always returned on GetHistory when addId is set
- (Apollon77) Fix potential crash cases (Sentry IOBROKER-INFLUXDB-4Q)
- (Apollon77) Correctly handle db names with hyphens for InfluxDB 1
- (Apollon77) Fix getHistory for non boolean fields
- (Apollon77) Adjust shard group duration to influxdb proposals
- (Excodibur) Address issue with updating retention policy for influxDB 1
- (Excodibur) Make request timeout configurable - defaults to 30s
- (Apollon77) Add option to also accept self signed ssl certificates for InfluxDB 2.x
- (bluefox) Marked interpolated data with
i=true
- (bluefox) Added new messages: 'update', 'delete', 'deleteRange', 'deleteAll' (only for influxdb v1)
- (bluefox) Corrected aggregation of data
- (Excodibur) Added expert settings to deactivate health checks for Influx 2 db (There are no health checks for Influx 1)
- (Excodibur) Hide settings not relevant when "log changes only" is not used
- (Apollon77) Allow all number values for debounce again
- (Apollon77) InfluxDB also do not like Infinite values, so ignore them too
- (bluefox) Support only
js.controller
>= 3.3.x - (bluefox) Used system/custom view for collecting the objects
- (bluefox) Implemented option to ignore zero- or/and below zero- values
- (Excodibur) Added option to store metadata (q, ack, from) as tags instead of fields for Influx 2.x - see README!
- (Excodibur) Failure to update/set retention policy will now cause warning instead of error/restart, to support more restrictive DB setups
- (Excodibur/Apollon77) Bug fixes and adjustments
- IMPORTANT: The adapter now requires Admin 5.1.15+ and js-controller 3.3+! For other admin or js-controller versions please use the former v1.9.5 of thi adapter.
- (Excodibur) Added InfluxDB 2.0 support
- (Excodibur) Adjust Retention handling on Database level to work for InfluxDB 1.x and 2.x
- (Excodibur) Removed retention options on datapoint level because never worked and also not supported really by InfluxDB anymore
- (bluefox) Added the support of Admin5
- (Apollon77) Optimize stop handling
- (Apollon77) Crash case prevented (Sentry IOBROKER-INFLUXDB-T, IOBROKER-INFLUXDB-Y)
- (Apollon77) crash prevented (Sentry IOBROKER-INFLUXDB-G)
- (Apollon77) crash prevented (Sentry IOBROKER-INFLUXDB-E)
- (Apollon77) Optimize handling in case of write errors and host unavailabilities
- (Apollon77) Some errors prevented
- (Apollon77) Set timeouts for influxdb connections to make sure connectioens do not run forever
- (bluefox) added command to read supported features
- (Apollon77) make sure disabling of data points while starting adapter do not crash adapter (Sentry IOBROKER-INFLUXDB-7)
- (Apollon77) Make sure all start values are processed correctly
- (Apollon77) More checks to make sure to not crash when states are disabled while data are processed (Sentry IOBROKER-INFLUXDB-8)
- (bluefox) set default history if not yet set
- (Apollon77) make sure disabling of data points do not crash adapter (Sentry IOBROKER-INFLUXDB-4)
- (Apollon77) Fix pot crash case when deleting objects while saving values (Sentry)
- Requires js-controller >= 2.0.0
- (Apollon77) removed usage of adapter.objects
- (Apollon77) check if objects have changed and ignore unchanged
- (Apollon77) Add Sentry for Error Reporting with js-controller 3.0
- (Apollon77) Make sure value undefined is ignored
- (Apollon77) Add option to define storage datatype per datapoint inclusing converting the value if needed
- (Apollon77) Small fix for older configurations
- (Apollon77) Enhance handling of DB outages with mass writes afterwards
- (Apollon77) Small fix for older configurations
- (Apollon77) Enhance Min-Delta logic for data points from type mixed
- (bluefox) Fix handling of float values in Adapter config and Datapoint config.
- (Apollon77) Add messages enableHistory/disableHistory
- (Apollon77) add support to log changes only if value differs a minimum value for numbers
- (Apollon77) small enhancements and fixes
- (Apollon77) support re-logging also for states that are not updated often (timed relog using relog-Interval)
- (Apollon77) try to solve easy type conflicts and convert float <--> boolean if needed
- (Apollon77) enhance getHistory to retrieve also boundary values better by selecting also points outside of the given time boundaries
- (Apollon77) small final change on custom dialog
- (Apollon77) optimizations and enhancements for high traffic situations
- (Apollon77) Bugfix if InfluxDB is unavailable
- (Apollon77) add functions getConflictingPoints and resetConflictingPoints
- (Apollon77) add option to re-log unchanged values to make it easier for visualization
- (Apollon77) changed storing values to asynchronous way
- (Apollon77) add support for „minmax“ aggregate type on getHistory
- (Apollon77) bulk write into DB
- (Apollon77) buffer values if no connection to DB
- (bluefox) connection indication
- (Apollon77) enhance and correct documentation
- (Apollon77) implement "step" calculation for aggregation if only "count" is set
- (Apollon77) change custom queries. Fix delete DB
- (Apollon77) Step is given as Milliseconds and not Seconds!
- (bluefox) Compatible only with new admin
- (bluefox) change name of object from "history" to "custom"
- (bluefox) better error handling
- (bluefox) no support of influxDB 0.8 (please update)
- (bluefox) make name of DB configurable
- (bluefox) support of milliseconds instead of seconds
- (bluefox) make onchange work
- (bluefox) retention policy for 0.9.x
- (bluefox) supported InfluxDB version 0.9.x and 0.8.x
- (bluefox) change supported InfluxDB version to 0.9.x
- (bluefox) initial commit
The MIT License (MIT)
Copyright (c) 2015-2022 bluefox, apollon77
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.