After having added a decent amount of entities to my Home Assistant setup, the user interface—especially the history tab, but also the history for each entity—became very sluggish. Having to wait for up to 30 seconds to see the history graph for a single entity became the norm with an sqlite file weighing in at 6GB and millions of entries in the “events” and “states” tables.
I wanted to keep my acquired data, so I had to migrate it from the current sqlite file to a MySQL database instead of starting anew. I’ve done this using Home Assistant version 0.81 and MariaDB 10. Some parts might change in the future.
Make sure you have sqlite3
installed and in your path—e.g. via installing it using apt-get install sqlite3
.
Then, stop home assistant to have a non-changing database file.
Change to your homeassitant data folder and then, using
sqlite3 home-assistant_v2.db .dump > ha-database_sqlite.dump.sql
you can dump your current sqlite Home Assistant database into an sql file. Depending on your database size and I/O speed of your device, this might take many minutes, even hours.
Out of the box, the sqlite dump doesn’t work in a MySQL setup.
For conversion, I found https://github.com/athlite/sqlite3-to-mysql/blob/master/sqlite3-to-mysql but it doesn’t completely work, so I had to make some changes.
I replaced
-e 's/"/`/g' \
with
-e 's/"events"/`events`/g' \
-e 's/"recorder_runs"/`recorder_runs`/g' \
-e 's/"schema_changes"/`schema_changes`/g' \
-e 's/"states"/`states`/g' \
-e 's/"end"/`end`/g' \
as otherwise, the json strings within the tables were broken.
See the whole content in sqlite-to-mysql.sh.
You might need to make the shell file executable using chmod +x sqlite3-to-mysql.sh
Afterwards, you start the conversion process using
sudo ./sqlite3-to-mysql.sh ha-database_sqlite.dump.sql > mysql_import_me.sql
Like the initial dump, this may take quite a while to finish. And you won’t have a progress bar either.
The file won’t have a database creation query in it, so you start by adding one to your MySQL database yourself.
Connect to your local mysql instance, enter the root password when prompted:
mysql -u root -p
Create a database called “homeassistant”:
CREATE DATABASE homeassistant;
Create a user “homeassistant” with access to the database so that you don’t have to use your root account:
GRANT ALL PRIVILEGES ON homeassistant.* to 'homeassistant' IDENTIFIED BY '<yourpassword>'
Now you are ready to import.
To speed up the process, I first set autocommit
to false and afterwards manually commited. I found that this increased the speed of the import. YMMV.
SET autocommit=0;
source <full path to your mysql_import_me.sql file>;
commit;
Again, this will probably take some time.
Find out current max of event_id
to set AUTO_INCREMENT
:
MariaDB [homeassistant]> SELECT MAX(event_id) AS Count FROM events;
+---------+
| Count |
+---------+
| 3189954 |
+---------+
1 row in set (0.000 sec)
Then alter the events
table, use the previous max, incremented by 1 (=> +1), and set it as the starting point for AUTO_INCREMENT
:
ALTER TABLE events MODIFY COLUMN event_id INT NOT NULL AUTO_INCREMENT, AUTO_INCREMENT=3189955;
Repeat this process for the three other tables. Especially events
and states
will take some time (half an hour was the longest I saw so far).
MariaDB [homeassistant]> SELECT MAX(state_id) AS Count FROM states;
+---------+
| Count |
+---------+
| 3189396 |
+---------+
1 row in set (0.000 sec)
MariaDB [homeassistant]> ALTER TABLE states MODIFY COLUMN state_id INT NOT NULL AUTO_INCREMENT, AUTO_INCREMENT=3189397;
MariaDB [homeassistant]> SELECT MAX(run_id) as Count FROM recorder_runs;
+-------+
| Count |
+-------+
| 192 |
+-------+
1 row in set (0.042 sec)
MariaDB [homeassistant]> ALTER TABLE recorder_runs MODIFY COLUMN run_id INT NOT NULL AUTO_INCREMENT, AUTO_INCREMENT=193;
MariaDB [homeassistant]> SELECT MAX(change_id) AS Count FROM schema_changes;
+-------+
| Count |
+-------+
| 3 |
+-------+
1 row in set (0.043 sec)
MariaDB [homeassistant]> ALTER TABLE schema_changes MODIFY COLUMN change_id INT NOT NULL AUTO_INCREMENT, AUTO_INCREMENT=4;
Then, drop the foreign key of states
, should it exist, so we can recreate it on our own with the right parameters.
ALTER TABLE states DROP FOREIGN KEY states_ibfk_1;
ALTER TABLE states ADD CONSTRAINT `states_ibfk_1` FOREIGN KEY (`event_id`) REFERENCES `events` (`event_id`)
You’re almost done! Only task left is to add MySQL to your HA configuration.
See https://www.home-assistant.io/components/recorder/#custom-database-engines for your specific setup.
For MariaDB 10, I had to set
recorder:
db_url: mysql://homeassistant:<YOURCHOSENPASSWORD>@<MARIADB10HOST>:3607/homeassistant
And that’s it!
Awesome gist! Thank you, i could kiss you!
I had to disable foreign key checks to be able to
ALTER TABLE
:And i guess more tables have been introduced, as i had to add
auto_increment
to the following tables too, finding the value by the same method as above:As a note, it requires about 3-4x the size of the original database (Mine was 1.2GB) to migrate it, and on my 10500k on an NVMe SSD especially the 'source ' took 35 minutes, and several other steps took more than a couple of minutes too.
The energy-dashboard have been transferred and i'm now rocking MariaDB!