FHIR BLOBs
Hapi-fhir-jpaserver is used alongside OpenELIS to provide FHIR support. Older versions of the project made extensive use of the Postgres BLOB data type. This data type cannot be handled in the same way as "normal" Postgres data types. Because of this, upgrading a large DB with many BLOBs takes a long time, is very memory intensive, and is prone to failing if enough memory isn't available. Due to these reasons, it is recommended to upgrade your hapi-fhir-jpaserver to at least 6.6.0 before upgrading your DB, so you can run the reindex command which moves all resources (smaller than a configurable limit) into the normal table structure.
Instructions for updating FHIR BLOBs
Stop the existing containers besides the database
sudo docker stop openelisglobal-webapp autoheal-oe external-fhir-api
Collect metrics around the data to see what you're working with
sudo docker exec -it openelisglobal-database psql -Uclinlims
SELECT COUNT(*) FROM clinlims.hfj_res_ver;
SELECT pg_table_size('pg_largeobject');
Run the script for upgrading hapi-fhir-jpaserver data structure in the database
wget https://github.com/hapifhir/hapi-fhir/releases/download/v6.2.0/hapi-fhir-6.2.0-cli.tar.bz2
bzip2 -d hapi-fhir-6.2.0-cli.tar.bz2
tar xf hapi-fhir-6.2.0-cli.tar
./hapi-fhir-cli migrate-database -d POSTGRES_9_4 -u "jdbc:postgresql://localhost:15432/clinlims currentSchema=clinlims" -n "clinlims" --no-column-shrink -p <password>
Collect metrics around the data to see that data loss has not occurred
sudo docker exec -it openelisglobal-database psql -Uclinlims
SELECT COUNT(*) FROM clinlims.hfj_res_ver;
SELECT pg_table_size('pg_largeobject');
Run 2.7 OE installer
wget <online-path-to-2.7-installer>
tar -xzf <installer-tar.gz>
cd <installer-dir>
sudo python3 OpenELIS.py
No
to logical db backup, docker cleans, and backup script
Modify the
docker-compose.yml
Change db container image from
14.4
to9.5
Start the containers. From this point until the db migration is started, these commands can be run in the background while normal use of the application(s) goes ahead
sudo docker-compose up -d
Create files to submit to the updated FHIR store to trigger a reindex of the various data types from BLOBs to regular column data. Resource types that are commonly used, leading to the bulk of the BLOBs are:
Task
,Patient
,ServiceRequest
,DiagnosticReport
,Observation
,Specimen
,Practitioner
,Organization
,Location
,QuestionnaireResponse
{ "resourceType": "Parameters", "parameter": [ { "name": "url", "valueString": "<ResourceType>?" }, { "name": "optimizeStorage", "valueString": "ALL_VERSIONS" } ] }
Send each optimize request as a POST request to the FHIR server
sudo curl -X POST -H "Content-Type: application/json" -d '@task-optimize.json' --cert /etc/openelis-global/cert.pem --key /etc/openelis-global/key.pem -k 'https://localhost:8444/fhir/$reindex'
Wait for the reindexes to succeed (checking the logs of the FHIR container should give some indication that they are running or not)
Run the vacuum lob command to clean up the database (first run is a test)
sudo docker exec -it openelisglobal-database vacuumlo -Uclinlims --dry-run
sudo docker exec -it openelisglobal-database vacuumlo -Uclinlims
Collect metrics around the data to see that data loss has not occurred.
The pg_largeobject
should be MUCH smaller, buthfj_res_ver
should be similar to before.sudo docker exec -it openelisglobal-database psql -Uclinlims
SELECT COUNT(*) FROM clinlims.hfj_res_ver;
SELECT pg_table_size('pg_largeobject');
Run the
pg_upgrade
as in the "Migrate with pg_upgrade" section belowCollect metrics around the data to see that data loss has not occurred.
sudo docker exec -it openelisglobal-database psql -Uclinlims
SELECT COUNT(*) FROM clinlims.hfj_res_ver;
SELECT pg_table_size('pg_largeobject');
Migrate with pg_upgrade
Much faster method of upgrading postgres dbs
If you are running these commands on a remote server, it is recommended to use a recoverable script session. For example, running screen
. If you disconnect, just reconnect and run screen -x
to recover active session.
NOTES:
It is paramount that data backups are up to date and recovery is tested before attempting a database migration.
This can be a very memory intensive process. It is recommended to increase memory on the server that this is running on to 128 GB or more. This can be accomplished without too much of a performance hit by adding an SSD as a swap drive.
This process doesn't delete the old database files until the new database is up, running, and tested. Ensure that OE is running properly and the data is there BEFORE deleting the old machine.
Because of point number 3, you will require room on the server for a copy of all the database files located at
/var/lib/openelis-global/data
Migrating OE 9.5 database to 14.4 database in dockerized environments
stop containers so no changes happen while migration is occurring
sudo docker stop autoheal-oe external-fhir-api openelisglobal-webapp openelisglobal-database
remove db container so auto restart doesn’t occur
sudo docker rm openelisglobal-database
1.
create folders for first step db migration to take place (this can be done on a separate machine with docker installed if the main server lacks resources to run the upgrade).
sudo mkdir /var/lib/openelis-global/db
sudo mkdir /var/lib/openelis-global/db/9.5
sudo mkdir /var/lib/openelis-global/db/14
sudo mkdir /var/lib/openelis-global/db/14/data
copy current db to the upgrade location (copy will preserve the old data so that we can more easily revert if something goes wrong)
If you are performing the upgrade on the machine where the database is installed:
sudo cp -r /var/lib/openelis-global/data /var/lib/openelis-global/db/9.5/data
If you are performing the upgrade on another machine where the database is not installed:
sudo tar cf /var/lib/openelis-global/data.tar.gz -C /var/lib/openelis-global/ data
sudo scp /var/lib/openelis-global/data.tar.gz username@destination:/var/lib/openelis-global/db/9.5/data.tar.gz
ssh username@destination
tar xzf /var/lib/openelis-global/db/9.5/data.tar.gz -C /var/lib/openelis-global/db/9.5/
run the 9.5 to 14 migration
sudo docker pull ctsteele/postgres-migration:9.5-14
sudo docker run -it --rm -v /var/lib/openelis-global/db/:/var/lib/postgresql/ ctsteele/postgres-migration:9.5-14 --link
replace old db with new db
sudo mv /var/lib/openelis-global/data /var/lib/openelis-global/data2
sudo mv /var/lib/openelis-global/db/14/data /var/lib/openelis-global/data
ensure file permissions and db access permissions are correct
sudo chown -R tomcat2:tomcat2 /var/lib/openelis-global/data
edit
/var/lib/openelis-global/data/pg_hba.conf
to include all the same entries that are in/var/lib/openelis-global/data2/pg_hba.conf
run the setup script for the new version with updated db, ignoring db backup couldn’t occur step
sudo python3 setup_OpenELIS.py
ensure systems start up and that data is present
optionally delete old db (or move to a secure backup server)
sudo rm /var/lib/openelis-global/db /var/lib/openelis-global/data2
Migrating OE 9.5 database to 14.4 database in non dockerized environment into dockerized environment (untested)
stop containers so no changes happen while migration is occurring
sudo docker stop autoheal-oe external-fhir-api openelisglobal-webapp
stop postgres instance
sudo -upostgres /usr/lib/postgresql/9/bin/pg_ctl -D /var/lib/postgresql/9/data -l logfile stop
check that upgrade can occur
time /usr/lib/postgresql/14/bin/pg_upgrade --old-bindir /usr/lib/postgresql/9/bin --new-bindir /usr/lib/postgresql/14/bin --old-datadir /var/lib/postgresql/9/data --new-datadir /var/lib/openelis-global/data --link --check
run the upgrade
time /usr/lib/postgresql/14/bin/pg_upgrade --old-bindir /usr/lib/postgresql/9/bin --new-bindir /usr/lib/postgresql/14/bin --old-datadir /var/lib/postgresql/9/data --new-datadir /var/lib/openelis-global/data --link
run the setup script for the new version with updated db, ignoring db backup couldn’t occur step
sudo setup_OpenELIS.py
ensure systems start up and that data is present
Migrate with pg_dump
This approach is mentioned as being the preferred option in postgres docs, but is VERY slow when restoring BLOBs
Migrating OE 9.5 database to 14.4 database in dockerized environments
Run the following commands to create the backup for restoring into OE 14, and the backup for 9.5 in case something goes wrong
sudo docker exec openelisglobal-database pg_dump -j 8 -d clinlims --verbose -U admin -F c -f /backups/95db.backup
sudo docker kill openelisglobal-database && sudo mv /var/lib/openelis-global/data /var/lib/openelis-global/data2
install new OE with setup script - ignore db missing when prompted
Run the following to pause non-db connections and restore the backup before bringing the containers up again (must run in OE installer directory). Restoring a db can take a long time for large dbs. To avoid a long downtime, this can be done in a separate container
sudo docker kill external-fhir-api openelisglobal-webapp && sudo docker rm external-fhir-api openelisglobal-webapp
sudo docker exec openelisglobal-database pg_restore -j 8 -d clinlims -U postgres -v -Fc -c /backups/95db.backup
sudo docker-compose up -d
confirm that OpenELIS is working and that the data is there by accessing the front end
optionally, delete old data
rm /var/lib/openelis-global/data2
rm /var/lib/openelis-global/backups/95db.backup
Migrating OE 9.5 database to 14.4 database in non dockerized environment into dockerized environment (untested)
Run the following commands to create the backup for restoring into OE 14
pg_dump -d clinlims -h localhost -p 5432 –verbose -U clinlims -F c -f /var/lib/openelis-global/backups/95db.backup
[enter password for clinlims]
Modify
setup.ini
to have docker db on port other than5432
and use docker dbsudo vi /etc/openelis-global/setup.ini
install new OE with setup script
Run the following to pause non-db connections and restore the backup before bringing the containers up again (must run in OE installer directory)
sudo docker kill external-fhir-api openelisglobal-webapp && sudo docker rm external-fhir-api openelisglobal-webapp
sudo docker exec openelisglobal-database pg_restore -d clinlims -U postgres -v -Fc -c /backups/95db.backup
sudo docker-compose up -d
confirm that OpenELIS is working and that the data is there by accessing the front end
optionally, delete old data
sudo rm /var/lib/openelis-global/backups/95db.backup
uninstall native postgres if you want to move the docker db onto port 5432
0 Comments