openhab-addons/bundles/org.openhab.binding.dbquery
Jacob Laursen a357f7f0cb
Upgrade json to 20231013 (#15745)
Signed-off-by: Jacob Laursen <jacob-github@vindvejr.dk>
2023-10-27 20:13:03 +02:00
..
src Improve javadoc for all addons (#15667) 2023-09-30 21:49:12 +02:00
NOTICE [dbquery] Create NOTICE file for dbquery addon (#11415) 2021-10-19 18:43:08 +02:00
pom.xml Upgrade json to 20231013 (#15745) 2023-10-27 20:13:03 +02:00
README.md [Documentation] Markdown improvements for bindings a to e (#13859) 2022-12-07 21:09:32 +01:00

DBQuery Binding

This binding allows creating items from the result of native database queries. It currently only supports InfluxDB 2.X.

You can use the addon in any situation where you want to create an item from a native query. The source of the query can be any supported database, and doesn't need to be the one you use as the persistence service in openHAB. Some use cases can be:

  • Integrate a device that stores its data in a database
  • Query derived data from you openHAB persistence, for example with Influx2 tasks you can process your data to create a new one
  • Bypass limitations of current openHAB persistence queries

Supported Things

There are two types of supported things: influxdb2 and a query. For each different database you want to connect to, you must define a Bridge thing for that database. Then each Bridge can define as many Query things that you want to execute.

Thing Configuration

Bridges

influxdb2

Defines a connection to an Influx2 database and allows creating queries on it.

Parameter Required Description
url Yes database url
user Yes name of the database user
token Yes token to authenticate to the database (Intructions about how to create one)
organization Yes database organization name
bucket Yes database bucket name

query

The Query thing defines a native query that provides several channels that you can bind to items.

Query parameters

The query items support the following parameters:

Parameter Required Default Description
query true Query string in native syntax
interval false 0 Interval in seconds in which the query is automatically executed
hasParameters false false True if the query has parameters, false otherwise
timeout false 0 Query execution timeout in seconds
scalarResult false true If query always returns a single value or not
scalarColumn false In case of multiple columns, it indicates which to use for scalarResult

These are described further in the following subsections.

query

The query the items represents in the native language of your database:

  • Flux for influxdb2

hasParameters

If hasParameters=true you can use parameters in the query string that can be dynamically set with the setQueryParameters action.

For InfluxDB use the ${paramName} syntax for each parameter, and keep in mind that the values from that parameters must be from a trusted source as current parameter substitution is subject to query injection attacks.

timeout

A time-out in seconds to wait for the query result, if it's exceeded, the result will be discarded and the addon will do its best to cancel the query. Currently it's ignored and it will be implemented in a future version.

scalarResult

If true the query is expected to return a single scalar value that will be available to result channels as string, number, boolean,... If the query can return several rows and/or several columns per row then it needs to be set to false and the result can be retrieved in resultString channel as JSON or using the getLastQueryResult action.

scalarColumn

In case scalarResult is true and the select returns multiple columns you can use that parameter to choose which column to use to extract the result.

Channels

Query items offer the following channels to be able to query / bind them to items:

Channel Type ID Item Type Description
execute Switch Send ON to execute the query manually. It also indicates if query is currently running (ON) or not running (OFF)
resultString String Result of last executed query as a String
resultNumber Number Result of last executed query as a Number, query must have scalarResult=true
resultDateTime DateTime Result of last executed query as a DateTime, query must have scalarResult=true
resultContact Contact Result of last executed query as Contact, query must have scalarResult=true
resultSwitch Switch Result of last executed query as Switch, query must have scalarResult=true
parameters String Contains parameters of last executed query as JSON
correct Switch ON if the last executed query completed successfully, OFF if the query failed.

All the channels, except execute, are updated when the query execution finishes, and while there is a query in execution they have the values from last previous executed query.

The resultString channel is the only valid one if scalarResult=false, and in that case it contains the query result serialized to JSON in that format:

{
    correct : true,
    data : [
        { 
            column1 : value,
            column2 : value
        },
        { ... }, //row2
        { ... }  //row3
    ]
}

Channel Triggers

calculateParameters

Triggers when there's a need to calculate parameters before query execution. When a query has hasParameters=true it fires the calculateParameters channel trigger and pauses the execution until setQueryParameters action is call in that query.

In the case a query has parameters, it's expected that there is a rule that catches the calculateParameters trigger, calculate the parameters with the corresponding logic and then calls the setQueryParameters action, after that the query will be executed.

Actions

For DatabaseBridge

executeQuery

It allows executing a query synchronously from a script/rule without defining it in a Thing.

To execute the action you need to pass the following parameters:

  • String query: The query to execute
  • Map<String,Object>: Query parameters (empty map if not needed)
  • int timeout: Query timeout in seconds

And it returns an ActionQueryResult that has the following properties:

  • correct (boolean) : True if the query was executed correctly, false otherwise
  • data (List<Map<String,Object>>): A list where each element is a row that is stored in a map with (columnName,value) entries
  • isScalarResult: It returns if the result is scalar one (only one row with one column)
  • resultAsScalar: It returns the result as a scalar if possible, if not returns null

Example (using Jython script):

from core.log import logging, LOG_PREFIX 
log = logging.getLogger("{}.action_example".format(LOG_PREFIX))
map = {"time" : "-2h"}
influxdb = actions.get("dbquery","dbquery:influxdb2:sampleQuery") //Get bridge thing
result = influxdb.executeQuery("from(bucket: \"default\") |> range(start:-2h)  |> filter(fn: (r) => r[\"_measurement\"] == \"go_memstats_frees_total\")  |> filter(fn: (r) => r[\"_field\"] == \"counter\")  |> mean()",{},5)
log.info("execute query result is "+str(result.data))

Use this action with care, because as the query is executed synchronously, it is not good to execute long-running queries that can block script execution.

For Queries

setQueryParameters

It's used for queries with parameters to set them. To execute the action you need to pass the parameters as a Map.

Example (using Jython script):

params = {"time" : "-2h"}
dbquery = actions.get("dbquery","dbquery:query:queryWithParams")  //Get query thing
dbquery.setQueryParameters(params)

getLastQueryResult

It can be used in scripts to get the last query result. It doesn't have any parameters and returns an ActionQueryResult as defined in executeQuery action.

Example (using Jython script):

dbquery = actions.get("dbquery","dbquery:query:queryWithParams")  //Get query thing
result = dbquery.getLastQueryResult()

Examples

The Simplest case

Define an InfluxDB2 database thing and a query with an interval execution. That executes the query every 15 seconds and punts the result in myItem.

# Bridge Thing definition
Bridge dbquery:influxdb2:mydatabase "InfluxDB2 Bridge" [ bucket="default", user="admin", url="http://localhost:8086", organization="openhab", token="*******" ]

# Query Thing definition
Thing dbquery:query:myquery "My Query" [ interval=15, hasParameters=false, scalarResult=true, timeout=0, query="from(bucket: \"default\") |> range(start:-1h) |> filter(fn: (r) => r[\"_measurement\"] == \"go_memstats_frees_total\")  |> filter(fn: (r) => r[\"_field\"] == \"counter\")  |> mean()", scalarColumn="_value" ]

# Item definition
Number myItem "QueryResult" {channel="dbquery:query:myquery:resultNumber"}

A query with parameters

Using the previous example you change the range(start:-1h) for range(start:${time})

Create a rule that is fired

  • When calculateParameters is triggered in myquery
  • Then executes the following script action (in that example Jython):
map = {"time" : "-2h"}   
dbquery = actions.get("dbquery","dbquery:query:myquery")   
dbquery.setQueryParameters(map)