Persistency storage
Data model
We have following data model, implemented in yaptide/persistence/models.py
:
Simulation model and dependent classes:
classDiagram
class SimulationModel {
id: int
job_id: str
user_id: int
start_time: datetime
end_time: datetime
title: str
platform: str
input_type: str
sim_type: str
job_state: str
tasks
estimators
}
class CelerySimulationModel {
id: int
merge_id: str
}
class BatchSimulationModel {
id: int
cluster_id: int
job_dir: str
array_id: int
collect_id: int
}
class TaskModel {
id: int
simulation_id: int
task_id: int
requested_primaries: int
simulated_primaries: int
task_state: str
estimated_time: int
start_time: datetime
end_time: datetime
platform: str
last_update_time: datetime
}
class CeleryTaskModel {
id: int
celery_id: str
}
class BatchTaskModel {
id: int
}
class InputModel {
id: int
simulation_id: int
compressed_data: bytes
data
}
class EstimatorModel {
id: int
simulation_id: int
name: str
file_name: str
compressed_data: bytes
data
}
class PageModel {
id: int
page_name: str
estimator_id: int
page_number: int
page_dimension: int
compressed_data: bytes
data
}
class LogfilesModel {
id: int
simulation_id: int
compressed_data: bytes
data
}
SimulationModel <|-- CelerySimulationModel
SimulationModel <|-- BatchSimulationModel
TaskModel <|-- CeleryTaskModel
TaskModel <|-- BatchTaskModel
SimulationModel "1" *-- "0..*" TaskModel
SimulationModel "1" *-- "0..*" EstimatorModel
EstimatorModel "1" *-- "0..*" PageModel
SimulationModel "1" *-- "0..*" LogfilesModel
SimulationModel *-- InputModel
other classes we use are:
classDiagram
class UserModel {
id: int
username: str
auth_provider: str
simulations
}
class YaptideUserModel {
id: int
password_hash: str
}
class KeycloakUserModel {
id: int
cert: str
private_key: str
}
class ClusterModel {
id: int
cluster_name: str
simulations
}
UserModel <|-- YaptideUserModel
UserModel <|-- KeycloakUserModel
We've been too lazy to write down the mermaid code for these diagrams, but ChatGPT nowadays does a good job on that.
Whenever you need to update the diagrams, just copy the code from the yaptide/persistence/models.py
file and ask ChatGPT to generate the diagram for you.
Database
Production version uses PostgreSQL database, while in the unit tests suite we use SQLite in-memory database.
Sometimes it may be convenient to connect to the production DB from outside the container, e.g. to check the content of the database. Then you can use the following command to get the DB URL.
docker exec -it yaptide_flask bash -c "cd /usr/local/app && python -c 'from yaptide.application import create_app; app = create_app(); app.app_context().push() or print(app.extensions[\"sqlalchemy\"].engine.url.render_as_string(hide_password=False))'"
The code above is implemented as a handy onliner, the code may look tricky, especially the app.app_context().push() or
part.
The reason for that hacking is simple. Regular methods to get the DB URL require the application context. This is usually achieved using with app.app_context():
construct, which is not possible in the oneliner.
Knowing the DB URL, you can connect to the DB using any DB client, e.g. psql
or pgadmin
. You can also use the db_manage.py
script from the yaptide/admin
directory. For example, to list all users in the DB, you can use the following command from outside the container:
FLASK_SQLALCHEMY_DATABASE_URI=postgresql+psycopg://yaptide_user:yaptide_password@localhost:5432/yaptide_db ./yaptide/admin/db_manage.py list-users
This is equivalent to the following command executed inside the container:
docker exec -it yaptide_flask ./yaptide/admin/db_manage.py list-users
Developing model
In Yaptide flask-migrate is responsible for modyfing database after each change to models.py
and keeping track of versions of database (new version comes after each modification of models.py).
Development steps
For development - running yaptide_postgres in docker is required (Flask-migrate can be used on sqlite daatbase we use in development but it's postgres database on production we want to migrate). It's recommended to do development on local machine.
- Make sure all poetry dependencies are installed. Run
poetry shell
in terminal. -
Calling
flask db
commands will requireFLASK_SQLALCHEMY_DATABASE_URI
variable to be defined before each execution:- The general pattern for
FLASK_SQLALCHEMY_DATABASE_URI
is taken from docker-compose (there is onlypostgres
changed to localhost or 127.0.0.1 ):
FLASK_SQLALCHEMY_DATABASE_URI=postgresql+psycopg://${POSTGRES_USER:-yaptide_user}:${POSTGRES_PASSWORD:-yaptide_password}@localhost:5432/${POSTGRES_DB:-yaptide_db}
e.g. for local development
FLASK_SQLALCHEMY_DATABASE_URI=postgresql+psycopg://yaptide_user:yaptide_password@localhost:5432/yaptide_db
will be put before eachflask db
call. For local development it can be exported as variable but it's not recommended for environments where username, password are sensitive information.From now each command in this docummentation containing
flask db
should be called withFLASK_SQLALCHEMY_DATABASE_URI
. - The general pattern for
-
Now it's time to prepare local/development database for development of models.py and creation of migration script.
- In
docker-compose.yml
edit database service to use volume with different name, this will create new volume and old one won't get deleted. Runscripts/start_with_docker.sh
. This will create database that for sure reflects what's in models.py. Then to mark database with version from migrations/versions, runflask --app yaptide.application db stamp head
. This will save id of newest version of database in alembic_version table. Be cautios as this option is only for development on local machine.
- In
-
Do modifications in
models.py
. - Run
flask --app yaptide.application db migrate
. - There will be generated migration file in migrations/versions. Name of the newest file is displayed in output of above command.
- IMPORTANT! Check the file carefully. For example there might be some
None
values which needs to be changed.- script that adds CASCADE option to foreign constraint at first looks like this
In this case change None to 'Task_simulation_id_fkey'.
def upgrade(): # ### commands auto generated by Alembic - please adjust! ### with op.batch_alter_table('Task', schema=None) as batch_op: batch_op.drop_constraint('Task_simulation_id_fkey', type_='foreignkey') batch_op.create_foreign_key(None, 'Simulation', ['simulation_id'], ['id'], ondelete='CASCADE')
- script that adds CASCADE option to foreign constraint at first looks like this
- Run
flask --app yaptide.application db upgrade
to apply migration script. - To undo changes and go back to previous version run
flask --app yaptide.application db downgrade
. - Commit and push script and modification to models.py.
Testing migration script with copy of production volume
- If there is testing environment other than local - pull changes.
- Copy volume ("data") of production postgres database and save it under different name.
- In
docker-compose.yml
modify database configuration to use this volume. - Run
scripts/start_with_docker_develop.sh
, to run backend and additionaly pgadminer tool (see section Using pgadminer). - Again prepare
FLASK_SQLALCHEMY_DATABASE_URI
like above and use it together with eachflask db
command. - Run
flask --app yaptide.application db upgrade
. - Check also
flask --app yaptide.application db downgrade
, then do upgrade again. All should execute without errors. - Do manual testing. Check functionalities, some might be unnecessary to check depending which part in
models.py
was changed:- logging in and out
- Loading simulation results, input files, logs.
- Submiting new simulation
- operations contained in
admin/db_manage.py
Migrating production
- Run git pull on master.
-
Backup the production database Before applying any migrations, create a backup of the live database:
pg_dump -U <db_user> -h <db_host> -d <db_name> > backup.sql
Alterniative is making copy of volume: -
docker run --rm -v yaptide_data:/var/lib/postgresql/data -v /home/ubuntu/backup:/backup busybox tar czf /backup/yaptide_data_backup.tar.gz -C /var/lib/postgresql/data .
3. Again prepareFLASK_SQLALCHEMY_DATABASE_URI
like above and use it together with eachflask db
command. 4. Applying the migration in production There are two options for applying the migration:Option 1: Execute from outside the Docker container
FLASK_SQLALCHEMY_DATABASE_URI=postgresql+psycopg://<user>:<password>@<db_container_ip>:5432/db_name flask --app yaptide.application db upgrade
Option 2: Execute from inside the Flask container
Access the container and run the upgrade: -
docker exec -it <flask_container> bash
-FLASK_SQLALCHEMY_DATABASE_URI=postgresql+psycopg://<user>:<password>@<db_container_ip>:5432/db_name flask --app yaptide.application db upgrade
5. Rollback strategy In case of any issues, you can revert the changes by running:flask --app yaptide.application db downgrade
Post-migration testing Perform manual tests on the production system:- Verify logging in and out.
- Check simulation submissions.
- Ensure any functionality affected by the migration is working.
-
In case of restoring database from backup, run:
-
psql -U <db_user> -h <db_host> -c "DROP DATABASE IF EXISTS <db_name>;"
-
psql -U <db_user> -h <db_host> -c "CREATE DATABASE <db_name>;"
-
psql -U <db_user> -h <db_host> -d <db_name> -f backup.sql
If copy of volume was made instead of backup.sql, run: -
docker run --rm -v yaptide_data:/var/lib/postgresql/data -v /home/ubuntu/backup:/backup busybox tar xzf /backup/yaptide_data_backup.tar.gz -C /var/lib/postgresql/data
-
Using pgadminer
Pgadminer is tool that lets user browse database through graphical interface. It can help with veryfication, testing and troubleshooting during migration. To run pgadminer alongside other containers run script: scripts/start_with_docker_develop.sh
. If executed locally it can be accessed from browser with address: localhost:9999
. When running from remote the tunnel connection is requred. Run:
ssh -L 9999:localhost:9999 <remote_host>
commands in db_manage.py
The db_manage.py
script provides several commands to manage the database. Below is a list of available commands along with their arguments and options:
- list_users
- Printed columns:
username
,auth_provider
-
Options:
-v
,--verbose
-
add_user
- Arguments:
name
-
Options:
--password
(default: '')-v
,--verbose
-
update_user
- Arguments:
name
-
Options:
--password
(default: '')-v
,--verbose
-
remove_user
-
Arguments:
name
auth_provider
-
list_tasks
- Printed columns:
simulation_id
,task_id
,task_state
,username
-
Options:
--user
--auth-provider
-
remove_task
- Arguments:
simulation_id
task_id
-
Options:
-v
,--verbose
-
list_simulations
- Printed columns:
id
,job_id
,start_time
,end_time
,username
- Printed columns:
-
Options:
-v
,--verbose
--user
--auth-provider
-
remove_simulation
- Arguments:
simulation_id
-
Options:
-v
,--verbose
-
add_cluster
- Arguments:
cluster_name
-
Options:
-v
,--verbose
-
list_clusters
- Columns:
id
,cluster_name