For most of the releases database schema differs between versions. To easily migrate/deploy database schema we are using EntityFramework migrations.
For further reading, visit the official Microsoft documentation on Entity Framework migration https://learn.microsoft.com/en-us/ef/core/managing-schemas/migrations/?tabs=dotnet-core-cli
There are two supported methods of applying migrations, discussed below:
- Using programmatic Entity Framework migrations (for simplicity they can be applied using docker image)
- Executing an Idempotent SQL script
For each release, we publish a docker image which is responsible for migrating the database. Similarly to the idempotent SQL script, it takes care of applying each migration only once and applying only missing migrations.
The image can be found here: https://hub.docker.com/r/radixdlt/babylon-ng-database-migrations
To migrate the database you just need to run that image providing a connection string to the database. To do that you need to set an environment variable for the docker container ConnectionStrings__NetworkGatewayMigrations
.
In the current version of NpgSQL, if using a schema other than public
, the migrations may fail to execute with an error message such as "__EFMigrationsHistory already exists on dbContext.Database.Migrate();". This is caused by this underlying issue.
We have implemented a workaround for this, which requires specifying the SearchPath=schema_name
parameter to the connection string as described here in the Ngpsql docs.
Providing the schema ONLY at the user level i.e ALTER user someuser in database mainnet SET search_path TO schema_name, public;
is not enough to activate the work-around and will result in errors.
This is a Raw SQL script that has to be executed on the database. It takes care of applying each migration only once and applying only missing migrations. It can be found as IdempotentApplyMigrations.sql in the migrations directory.
Before running the script, may need to ensure that the user running the script has SUPERUSER privileges, for example with:
ALTER USER db_dev_superuser WITH SUPERUSER;
For more information, see the Microsoft documentation on the idempotent SQL scripts.
Because of a bug in CLI responsible for generating these SQL scripts, it is possible that it will generate invalid SQL scripts (missing semicolon at the end of each command). We are trying to make sure it does not happen but since we are internally using docker images to migrate the database there is a chance that it might slip through. To fix that, simply add a semicolon at the end of the invalid command.
There are two scenarios that might happen between gateway versions.
If the situation allows us to migrate existing schemas without losing data, there will be database migrations that take care of that. To migrate to the new version you will only need to execute migrations on top of the existing database.
There are situations in which it is impossible to migrate the database schema and migrate existing data to the new schema.
In such cases, we are making it clear in release notes that it has to be deployed on a clean database and requires syncing the gateway database from the first transaction.
To do that:
- Make sure your database is empty.
- there is no
__EFMigrationsHistory
table in the database. - there are no tables in that database.
- there is no
- Execute migrations using the preferred method (idempotent SQL script or docker image).
- Run DataAggregator and process the entire transaction stream.
Keep in mind that processing the entire transaction stream might take a significant amount of time. It is dependent on multiple factors (i.e. machine resources, the latency on connecting to node/database, number of transactions on the network), As of now (1st June 2024) foundation gateway is able to sync within around 30 hours. Gateway API will not return up-to-date information before the entire transaction stream is processed.