Taming the Recorder

I run occasionally into an excessive database file size. Grooming the community provided a wealth of insights in scattered places. This what I learned while taming the recorder.

Problem Description: I operate a Raspberry Pie 3 with Home Assistant OS, build-in database sqlite, managing 1000+ entities. I run into an excessive use of the Recorder while performing some tests, resulting in a >1 GB file size for home-assistant_v2.db. This led to the following problems: HA crashes while compiling ESPhome, and creating a full snapshot is not possible.

To understand the reasons and to find a remedy I made use of the following tools: Template Editor, Sensors (filesize, systemmonitor, sql, infludb), Plug-Ins (Samba Server, SQLite Web Plug-In, InfluxDB Plug-In), Linux tools (top, du), and Win10 tools (sqlite3_analyzer.exe).

I learned to differentiate between three time frames: runtime, short term recording, and log term storage.

To understand the runtime load, I measured the number of states and events with the template editor ( in my case the result was 1066 and 437):

Current number of events= {{states | count}}, and states= {{states.sensor | count }}

short term recording. My recorder is set to purge every day, thus storing every not filtered state change and event for at least 24 hours. The SQLite Web plug-in gives answers about the load: The first two queries return the total number of states and events.

select count('*') from states  
select count('*') from events  

To identify the most load consuming entity_id:

SELECT entity_id, COUNT(*) as count FROM states GROUP BY entity_id ORDER BY count DESC LIMIT 10;

A list of all monitored sensors:

select distinct entity_id from states WHERE domain = 'sensor'  order by entity_id   

Show the latest 10 entries for a specific sensor. Looking at the difference between timestamps, one can determine its updating interval.

select state, created from states WHERE entity_id = 'sensor.ferres_timestamp_delta' ORDER by state_id DESC LIMIT 10;

long term monitoring : I found influxDB very useful. The following experiment compares the internal load (purple) and the database file size (blue), covering a period of two days. The peak load was 2M states and 1M events, consuming 1.4GB space. Two morning purges (at 4:00) can be seen. However, this does not reduce the size of the data base. Only a service call recorder.purge with the option repack: true reduced the size to 0.3 MB (at 11:00).


My conclusions:

  • Separate short and long time requirements. Deploy different database machines.

  • Identify the most harmful recorder contributors in both states and events tables with respect to the number of involved entities, their update frequency, and keep duration.
    This led me to follow the following strategy:
    a) Reduce the recorder candidates by very stringent whitelist filtering,
    b) Reduce the update frequency of sensors at the origin (workaround: automate saving sensor data to an input_number entity at suitable intervals),
    c) Reduce the recorder keep time to a minimum.

  • Erase intermediate recordings (i.e. for test purposes) manually with purge/repack.

  • Consider using notify (platform file) as an alternative.

  • Keep a fresh database for later usage. In a emergency, replace databases with file system commands, while HA is frozen under the system call homeassistant.stop.

[edit] I ended up reducing the initial database size by 99%, without loosing any of my desired data.

5 Likes

Great post! This is something I also wanted to dive into. However, I’m using MariaDB and don’t really have a lot of experience using SQL. Do you have any advise on how to tackle this problem in MariaDB?

Try wrapping the database commands into a sql sensor. Or use a CLI or web interface to your favorite database.

I’m in agreement. My database size is horrendous. I have 1.6GB database and I’ve set a 5 day purge. I want more than 5-days of history but I think it records entirely too much. I think there is room for improvement. Here’s some working criteria which I came up with based on your research and my experience.

  1. Don’t touch boolean entries
  2. Keep all of today’s, yesterday’s, and the day before’s entries.
  3. Remove redundant, matching-value events after 72 hours.
  4. Remove 1/2 of intermediate entries (defined by min/max over the course of 15m/30m/1h/6h depending on user/sensor settings and activity) of each entry older than 72 hours, each 24 hours, during low processor activity time.
  5. Determine a point (maybe 4 weeks) where the data is too corrupt to be useful.

As an alternative to using influxdb as the long term monitoring solution there is TimescaleDB and the HA custom component LTSS (Custom component: Long Time State Storage (LTSS) utilizing TimescaleDB) for those preferring to rely on an sql-based solution for everything.

Keep a close look at the events recordings. If not excluded, any state change of a sensor produces a new entry in both the events and states table.

The count for all events was 21690. In this case, state_changed accounted for 2/3 of all recorded events.

I decided to exclude state_changed and service_call events.

Does anyone else think logging in general is a weakness in HA?

I tried to start a new thread along those lines, but all I got was a snarky reply suggesting that HA shouldn’t try to be easy to use, and that beginners aren’t wanted here.

Thanks for the help man! I’m going to post my journey for those who are also SQL-limited as I am.

I use the MariaDB plugin for Home Assistant. Since I didn’t really know how to connect to my database, I wanted to find a nice GUI-capable tool. For this, I have used dbForge.

Stepwise:

  • download and install dbForge.
  • Go to the MariaDB config page, and in the Network section, enter your port number. In my case the Container column reads 3306/tcp and I have entered 3306 below Host.
  • Open dbForge and use this tutorial to enter your database details
    • Use type TCP/IP
    • As Host enter your HA IP address and as port enter the port you used (3306 in my case)
    • As User and Password, use the entries in your MariaDB plugin config under logins
    • As Database, use the database name set in your MariaDB plugin config (default is homeassistant)
    • Press Test Connection or Connect if you’re feeling lucky!
  • Press ctrl+n to start a new SQL
  • Enter the commands stated above by @heinrich

Some useful commands I have used:
To find out which entities use the most data (updated 2025-01-06 based on this post by @jeedewee )

SELECT
  COUNT(*) AS cnt,
  COUNT(*) * 100 / (SELECT COUNT(*) FROM states) AS cnt_pct,
  states_meta.entity_id
FROM states
INNER JOIN states_meta ON states.metadata_id=states_meta.metadata_id
GROUP BY states_meta.entity_id
ORDER BY cnt DESC

To remove entities from the database directly using regular expressions:

-- first test if the regular expression works. I'm looking for e.g.: sensor.blitzwolf_NUC_energy_voltage
SELECT entity_id, COUNT(*) as count FROM states WHERE entity_id LIKE 'sensor.blitzwolf%status' GROUP BY entity_id ORDER BY count DESC LIMIT 10;
-- then remove the entities. This is final!
DELETE FROM states WHERE entity_id LIKE 'sensor.blitzwolf%energy\_voltage';

To find out how much data each table (I think it’s called a table) uses (credit goes to mbuscher)

SELECT
    table_name AS `Table Name`,
	table_rows AS `Row Count`,
	ROUND(SUM(data_length)/(1024*1024*1024), 3) AS `Table Size [GB]`,
	ROUND(SUM(index_length)/(1024*1024*1024), 3) AS `Index Size [GB]`,
	ROUND(SUM(data_length+index_length)/(1024*1024*1024), 3) `Total Size [GB]`
FROM information_schema.TABLES
WHERE table_schema = 'homeassistant'
GROUP BY table_name
ORDER BY table_name 

Another thing I found useful was to plot the first 1000 entities of the first query using Excel and then calculate the sum of all counts up until that entity. That way I found out I could reduce the size of my database by a factor of 10, simply by removing the first 100 entities from the database.

EDIT 2025-01-06:
The above query has not been working for a long time, and many have already found workarounds. However, if you end up googling this later, I want to refer to the correct queries, which can be found here: How to keep your recorder database size under control - #126 by jeedewee

1 Like

How do you delete when you get a foreign key error?

#1451 - Cannot delete or update a parent row: a foreign key constraint fails (homeassistant.states, CONSTRAINT states_ibfk_2 FOREIGN KEY (old_state_id) REFERENCES states (state_id))

You first delete the record the foreign key is pointing to.

Get disk storage size of influxDB data.

To learn more about my influxDB storage consumption, I checked the running system and compared it with various snapshots. Motivation: I paid not enough attention to my influxDB setup. Basically, I forgot to set excludes. As a result, the storage size grew in excess of 10 GB, resulting in an instable system, and in difficulties with snapshots.

To find the influxDB data storage size in a Snapshot, open the tar file in 7zip and navigated into the influxdb.tar. In my case:

To find the influxDB data storage size in the live filesystem, start a root shell (e.g. HDMI Monitor and “login”):
• cd /mnt/data/supervisor/addons/data/
• du -h *influxdb

Result: 410 MB. Looking deeper rendered again the two databases, with 347 MB for Homeassistant and 40 MB for _internal

Note: Data is not stored in a single file, but instead distributed in “shard” subdirectories. To navigate through container use “docker images, docker ps, docker inspect”

And how do you transfer that size information into a constantly updated template or command line sensor?

Try the SQL integration. It offers database sensors, including size. In my case:

  - platform: sql
    db_url: !secret mariadb_url
    scan_interval: 600
    queries:
      - name: sql_maria_size_total
        query: 'SELECT table_schema "database", Round(Sum(data_length + index_length) / 1048576, 2) "value" FROM information_schema.tables WHERE table_schema="homeassistant" GROUP BY table_schema'
        column: 'value'
        unit_of_measurement: MB

This does not seem to work anymore starting with HA Core 2023.4 where the new table states_meta has been introduced. How to adapt this statement so it works again?

Edit: found a working version, thanks to @jeedewee, see How to keep your recorder database size under control - #126 by jeedewee

1 Like

You’re right. I removed the queries a long time ago. I found that the database size sensor works, there is a working example in the SQL sensor page. Could you share your yaml code for the most recorded entities sensor?

I don’t have a sensor for the most recorded entities. No use-case for such a sensor.

SQL is at How to keep your recorder database size under control - #126 by jeedewee

DB noob here.
Got a db file ~8GB on a test HA setup.
Due to some internal issues with templates got this chain reaction:
python got crazy → huge CPU load → DB is flooded by some unknown data → dbsize increased from ~1GB to ~8GB.

Now I need to know which integration managed to flood the DB.

HA in Docker (Debian), so cannot use HASS addons for sql.
The db file is stored on some HDD in Win machine.
May be moved to some Debian machine if needed.

What shall I do?

Update:
Answering my own question.
Managed to open db-file by DB Browser for SQLite from Win-machine.
The HA config folder was opened as a SMB folder.