This page is part of archived documentation for openHAB 4.0. Go to the current stable version
# rrd4j Persistence
The rrd4j (opens new window) 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.
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..
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..
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).
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:
- granularity of 10s for the last hour
- granularity of 1m for the last week
- granularity of 15m for the last year
- granularity of 1h for the last 5 years
- 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.
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:
- granularity of 10s for the last hour
- granularity of 1m for the last week
- granularity of 15m for the last year
- granularity of 1h for the last 5 years
- 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.
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:
- granularity of 5s for the last hour
- granularity of 1m for the last week
- granularity of 15m for the last year
- granularity of 4h for the last 10 years
# Examples
# rrd4j.cfg
file
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
file:
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 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.