mirror of
https://github.com/openhab/openhab-addons.git
synced 2025-01-25 14:55:55 +01:00
cb1bd8c21f
Signed-off-by: Leo Siepel <leosiepel@gmail.com> Signed-off-by: Ciprian Pascu <contact@ciprianpascu.ro>
267 lines
14 KiB
Markdown
267 lines
14 KiB
Markdown
# rrd4j Persistence
|
|
|
|
The [rrd4j](https://github.com/rrd4j/rrd4j) persistence service is based on a round-robin database.
|
|
|
|
In contrast to a "normal" database such as db4o, a round-robin database does not grow in size - it has a fixed allocated size.
|
|
This is accomplished by saving a fixed amount of datapoints and by doing data compression, which means that the older the data is, the less values are available.
|
|
The data is kept in several "archives", each holding the data for its set timeframe at a defined level of granularity.
|
|
The starting point for all archives is the actually saved data sample (Item value).
|
|
So while you might store a sample value every minute for the last 8 hours, you might store the average per day for the last year.
|
|
|
|
This service cannot be directly queried, because of its data compression, which means that it cannot provide precise answers to all queries.
|
|
|
|
NOTE: rrd4j is for storing numerical data only.
|
|
It cannot store complex data types.
|
|
The supported item types are therefore only `Switch` (internally mapped to 0/1), `Dimmer`, `Number`, `Contact` (internally mapped to 0/1), `Rollershutter` and `Color` (only brightness component stored).
|
|
|
|
## Configuration
|
|
|
|
The rrd4j persistence services comes with a default persistence strategy which persists every Item on every state change and at least once a minute.
|
|
Additionally, it restores the last stored value at system startup.
|
|
|
|
If you want to define a custom behavior, you will need to create a `rrd4j.persist` file in the `persistence` configuration folder.
|
|
|
|
## Persistence Process
|
|
|
|
Round-robin databases (RRDs) have fixed length so called "archives" for storing values.
|
|
Think of an archive as a "drawer" with a fixed number of "storage boxes" in it.
|
|
|
|
The persistence service reads data "samples" from the openHAB core at regular intervals, and these are then put into the storage boxes.
|
|
Either a) the samples are stored singly directly into a box, or b) multiple samples are consolidated (using a consolidation function) into a box.
|
|
|
|
The service starts by storing samples in the leftmost box in the drawer.
|
|
Once the leftmost box is full, the service starts filling the next box to the right; and so on.
|
|
Once the rightmost box in the drawer is full, the leftmost box is emptied, the content of all boxes is moved one box to the left, and new content is added to the rightmost box.
|
|
|
|
## Datasources
|
|
|
|
For every persisted Item, a separate database file is created in the `userdata/persistence/rrd4j` folder.
|
|
These database files contain the archives of different granularities.
|
|
|
|
By default, if `services/rrd4j.cfg` does not exist, or if an Item is not explicitly listed in a `<dsName>.items` property value in it, then the respective Item will be persisted according to the [default datasource settings](#default-datasource).
|
|
|
|
Other datasources **may** be configured in addition.
|
|
This is done in a `services/rrd4j.cfg` configuration file.
|
|
|
|
If an Item **is** explicitly listed in a `<dsName>.items` property value, then it is persisted according to those respective datasource settings.
|
|
|
|
Each datasource is defined by three property values (`def`, `archives`, `items`), where each `archives` property can comprise settings for one or more archives.
|
|
|
|
The various datasource property values are explained in the table below.
|
|
|
|
| Property | Description |
|
|
|---------------------|-------------|
|
|
| `<dsName>`.def | Definition of the range of sample values to be taken, and when. The format is `<dsType>,<heartBeat>,<minValue>,<maxValue>,<sampleInterval>` |
|
|
| `<dsName>`.archives | List of archives to be created. Each archive defines which subset of data samples shall be archived, and for how long. Consists of one or more archive entries separated by a ":" character. The format for one archive entry is `<consolidationFunction>,<xff>,<samplesPerBox>,<boxCount>` |
|
|
| `<dsName>`.items | List of Items whose values shall be sampled and stored in the archive. The format is `Item1,Item2` _**Note: the same Item is not allowed to be listed in more than one datasource!**_ |
|
|
|
|
For example..
|
|
|
|
```ini
|
|
ctr24h.def=COUNTER,900,0,U,60
|
|
ctr24h.archives=AVERAGE,0.5,1,480:AVERAGE,0.5,10,144
|
|
ctr24h.items=Item1,Item2
|
|
```
|
|
|
|
The description of the various datasource property elements is as follows:
|
|
|
|
### `<dsName>` (Datasource Name)
|
|
|
|
The name of the datasource.
|
|
It must be an alphanumeric string.
|
|
|
|
### `<dsType>` (Datasource Type)
|
|
|
|
Defines the type of data to be stored.
|
|
It must be one of the following string values:
|
|
|
|
- **COUNTER** represents an ever-incrementing value (historically this was used for packet counters or traffic counters on network interfaces, a typical home-automation application would be your electricity meter). If you store the values of this counter in a simple database and make a chart of that, you'll most likely see a nearly flat line, because the increments per time are small compared to the absolute value (e.g. your electricity meter reads 60567 kWh, and you add 0.5 kWh per hour, than your chart over the whole day will show 60567 at the start and 60579 at the end of your chart. That is nearly invisible. RRD4J helps you out and will display the difference from one stored value to the other (depending on the selected size). Please note that the persistence extensions will return difference instead of the actual values if you use this type; this especially leads to wrong values if you try to restoreOnStartup!
|
|
- **GAUGE** represents the reading of e.g. a temperature sensor. You'll see only small deviation over the day and your values will be within a small range, clearly visible within a chart.
|
|
- **ABSOLUTE** is like a counter, but RRD4J assumes that the counter is reset when the value is read. So these are basically the delta values between the reads.
|
|
- **DERIVE** is like a counter, but it can also decrease and therefore have a negative delta.
|
|
|
|
### `<heartBeat>` (Heart Beat)
|
|
|
|
The heartbeat parameter helps the database to detect missing values.
|
|
i.e. if no new sample is stored after "heartBeat" seconds, the value is considered missing when charting.
|
|
|
|
It must be a positive integer value.
|
|
|
|
### `<minValue> / <maxValue>` (Minimum resp. Maximum Value)
|
|
|
|
These parameters define the range of acceptable sample values for that datasource.
|
|
They must be either:
|
|
|
|
- A numeric value, or
|
|
- The letter "U" (unlimited)
|
|
|
|
### `<sampleInterval>` (Sample Interval)
|
|
|
|
The time interval (seconds) between reading consecutive samples from the openHAB core.
|
|
|
|
It must be a positive integer value.
|
|
|
|
Relation between the sample intervall and openHABs persistence strategy.
|
|
The persistence strategy determines on which events (time, change, startUp) a new value is provided to the persistence service, the sample intervall determines at which time steps the actually provided value is being read (i.e an everyMinute strategy provides a new value at every full minute, a sample intervall of 10 seconds would use such a provided value 6 times).
|
|
|
|
### `<consolidationFunction>` (Consolidation Function)
|
|
|
|
Determines the type of data compression to be used when more than one sample is to be stored in a single "storage box".
|
|
So if you use the `AVERAGE` function, and two samples of `20.0` and `21.0` are to be stored, then the value `20.5` would be stored in the box.
|
|
|
|
It must be one of the following strings:
|
|
|
|
- **AVERAGE** the average of all the samples is stored in the box
|
|
- **MIN** the lowest sample is stored in the box
|
|
- **MAX** the highest sample is stored in the box
|
|
- **LAST** the last sample is stored in the box
|
|
- **FIRST** the first sample is stored in the box
|
|
- **TOTAL** the sum of all samples is stored in the box
|
|
|
|
All archives of a datasource must use the same `<consolidationFunction>`.
|
|
|
|
### `<xff>` (X-files Factor)
|
|
|
|
Defines the maximum allowed proportion of data samples that are stored as NaN ("Not a Number") relative to the set number of `<samplesPerBox>`. In case this proportion is above the set value, NaN will be persisted instead of the consolidated value. Using 0.5 would require at least 50 percent of the data samples to hold a value other than NaN.
|
|
|
|
It must be a value between 0 and 1.
|
|
|
|
### `<samplesPerBox>` (Samples Per Box)
|
|
|
|
The number of consecutive data samples that will be consolidated to create a single entry ("storage box") in the database.
|
|
If `<samplesPerBox>` is greater than 1 then the samples will be consolidated into the "storage box" by means of the `<consolidationFunction>` described above.
|
|
The time span covered by a single "storage box" is therefore (`<sampleInterval>` x `<samplesPerBox>`) seconds.
|
|
|
|
It must be a positive integer value.
|
|
|
|
### `<boxCount>` (Box Count)
|
|
|
|
The number of "storage boxes" in the archive.
|
|
The time span covered by a full archive is therefore (`<sampleInterval>` x `<samplesPerBox>` x `<boxCount>`) seconds.
|
|
|
|
It must be a positive integer value.
|
|
|
|
### Multiple Possible Archives
|
|
|
|
As already said, each datasource can have one or more archives.
|
|
The purpose of having several archives is that it allows a different granularity of data storage over different timespans.
|
|
|
|
In the example below..
|
|
|
|
```ini
|
|
ctr24h.def=COUNTER,900,0,U,60
|
|
ctr24h.archives=AVERAGE,0.5,1,480:AVERAGE,0.5,10,144
|
|
ctr24h.items=Item1,Item2
|
|
```
|
|
|
|
The `ctr24.def` defines a datasource which is using a COUNTER, a `<hearBeat>` of 900 seconds, a `<minValue>` of 0, a `<maxValue>` of unlimited and a `<sampleInterval>` of 60 seconds.
|
|
|
|
The first archive entry in the `ctr24.archives` parameter has `480` boxes each containing `1` sample (or to be exact the `AVERAGE` of `1` sample).
|
|
So it covers `480 X 60` seconds of data (8 hours) at a granularity of one minute.
|
|
As a general rule the first archive (and maybe the only one) should have `<samplesPerBox> = 1` so that each sample is stored in one box.
|
|
|
|
And the second archive entry has `144` boxes each containing the `AVERAGE` of `10` samples.
|
|
So it covers `144 X 10 X 60` seconds of data (24 hours) at a granularity of ten minutes.
|
|
|
|
## Default Datasource
|
|
|
|
The service automatically creates three default datasources with the properties below.
|
|
|
|
There is no `.items` parameter for the default datasources.
|
|
This means that any Item with an allocated strategy in the `rrd4j.persist` file is persisted using one the default settings, unless the Item is explicitly listed in an `.items` property value of a datasource in the `rrd4j.cfg` file.
|
|
|
|
#### default_numeric
|
|
|
|
This datasource is used for plain `Number` items.
|
|
It does not build averages over values, so that it is ensured that discrete values are kept when being read (e.g. an Item which has only states 0 and 1 will not be set to 0.5).
|
|
|
|
```ini
|
|
default_numeric.def=GAUGE,600,U,U,10
|
|
default_numeric.archives=LAST,0.5,1,360:LAST,0.5,6,10080:LAST,0.5,90,36500:LAST,0.5,360,43800:LAST,0.5,8640,3650
|
|
```
|
|
|
|
It uses 10 seconds as a step size for numeric values and allows a 10 minute silence between updates.
|
|
|
|
It defines 5 archives:
|
|
|
|
1. granularity of 10s for the last hour
|
|
2. granularity of 1m for the last week
|
|
3. granularity of 15m for the last year
|
|
4. granularity of 1h for the last 5 years
|
|
5. granularity of 1d for the last 10 years
|
|
|
|
#### default_quantifiable
|
|
|
|
This datasource is used for `Number` items with dimensions - it is therefore assumed that the values are measurement values that exist on a continuum.
|
|
It thus builds averages over values, so that graphs can be smooth, even if there is only a coarse granularity available.
|
|
|
|
```ini
|
|
default_quantifiable.def=GAUGE,600,U,U,10
|
|
default_quantifiable.archives=AVERAGE,0.5,1,360:AVERAGE,0.5,6,10080:AVERAGE,0.5,90,36500:AVERAGE,0.5,360,43800:AVERAGE,0.5,8640,3650
|
|
```
|
|
|
|
It uses 10 seconds as a step size for numeric values and allows a 10 minute silence between updates.
|
|
|
|
It defines 5 archives:
|
|
|
|
1. granularity of 10s for the last hour
|
|
2. granularity of 1m for the last week
|
|
3. granularity of 15m for the last year
|
|
4. granularity of 1h for the last 5 years
|
|
5. granularity of 1d for the last 10 years
|
|
|
|
#### default_other
|
|
|
|
This datasource is used for any other items.
|
|
Their values are considered to be discrete, similar to the `default_numeric` datasource, but it keeps the data in more fine-granular archives.
|
|
|
|
```ini
|
|
default_other.def=GAUGE,3600,U,U,5
|
|
default_other.archives=LAST,0.5,1,720:LAST,0.5,12,10080:LAST,0.5,180,35040:LAST,0.5,2880,21900
|
|
```
|
|
|
|
It uses 5 seconds as a step size for discrete values and allows a 1h silence between updates.
|
|
|
|
It defines 4 archives:
|
|
|
|
1. granularity of 5s for the last hour
|
|
2. granularity of 1m for the last week
|
|
3. granularity of 15m for the last year
|
|
4. granularity of 4h for the last 10 years
|
|
|
|
## Examples
|
|
|
|
### `rrd4j.cfg` file
|
|
|
|
```ini
|
|
ctr24h.def=COUNTER,900,0,U,60
|
|
ctr24h.archives=AVERAGE,0.5,1,480:AVERAGE,0.5,10,144
|
|
ctr24h.items=Item1,Item2
|
|
ctr7d.def=COUNTER,900,0,U,60
|
|
ctr7d.archives=AVERAGE,0.5,1,480:AVERAGE,0.5,10,144:AVERAGE,0.5,60,672
|
|
ctr7d.items=Item3,Item4
|
|
```
|
|
|
|
### `rrd4j.persist` Example
|
|
|
|
```java
|
|
Strategies {
|
|
// for rrd charts, we need a cron strategy
|
|
everyMinute : "0 * * * * ?"
|
|
}
|
|
|
|
Items {
|
|
// persist items on every change and every minute
|
|
* : strategy = everyChange, everyMinute
|
|
}
|
|
```
|
|
|
|
**IMPORTANT:**
|
|
When creating a custom datasource in the `rrd4j.cfg` file the used [sample interval](#sampleinterval-sample-interval) should be 20 seconds or less in order to keep the granularity. The selection of the used strategy has no effect on the granularity.
|
|
|
|
## Troubleshooting
|
|
|
|
From time to time, you may find that if you change the Item type of a persisted data point, you may experience charting or other problems. To resolve this issue, remove the old `<item_name>`.rrd file in the `${openhab_home}/userdata/persistence/rrd4j` folder or `/var/lib/openhab/persistence/rrd4j` folder for apt-get installed openHABs.
|
|
|
|
Restoring Item values after startup takes some time. Rules may already have started to run in parallel. Especially in rules that are started via the "System started" trigger, it may happen that the restore has not yet completed resulting in non-defined Item values. In these cases the use of restored Item values should be delayed by a couple of seconds. This delay has to be determined experimentally.
|