From 2f1e6d91020d399dfec12e3cc57642b34a05b584 Mon Sep 17 00:00:00 2001 From: Laurent LAPORTE Date: Wed, 21 Feb 2024 16:40:59 +0100 Subject: [PATCH 1/3] test(study-db): add unit test to delete studies with additional data --- tests/storage/repository/test_study.py | 46 +++++++++++++++++++++++++- 1 file changed, 45 insertions(+), 1 deletion(-) diff --git a/tests/storage/repository/test_study.py b/tests/storage/repository/test_study.py index bb4ea795e3..5535f4d074 100644 --- a/tests/storage/repository/test_study.py +++ b/tests/storage/repository/test_study.py @@ -1,3 +1,4 @@ +import json from datetime import datetime from sqlalchemy.orm import Session # type: ignore @@ -5,7 +6,7 @@ from antarest.core.cache.business.local_chache import LocalCache from antarest.core.model import PublicMode from antarest.login.model import Group, User -from antarest.study.model import DEFAULT_WORKSPACE_NAME, RawStudy, Study, StudyContentStatus +from antarest.study.model import DEFAULT_WORKSPACE_NAME, RawStudy, Study, StudyAdditionalData, StudyContentStatus from antarest.study.repository import StudyMetadataRepository from antarest.study.storage.variantstudy.model.dbmodel import VariantStudy @@ -77,6 +78,49 @@ def test_lifecycle(db_session: Session) -> None: assert repo.get(a_id) is None +def test_study__additional_data(db_session: Session) -> None: + repo = StudyMetadataRepository(LocalCache(), session=db_session) + + user = User(id=0, name="admin") + group = Group(id="my-group", name="group") + + patch = {"foo": "bar"} + a = RawStudy( + name="a", + version="820", + author="John Smith", + created_at=datetime.utcnow(), + updated_at=datetime.utcnow(), + public_mode=PublicMode.FULL, + owner=user, + groups=[group], + workspace=DEFAULT_WORKSPACE_NAME, + path="study", + content_status=StudyContentStatus.WARNING, + additional_data=StudyAdditionalData(author="John Smith", horizon="2024-2050", patch=json.dumps(patch)), + ) + + repo.save(a) + a_id = a.id + + # Check that the additional data is correctly saved + additional_data = repo.get_additional_data(a_id) + assert additional_data.author == "John Smith" + assert additional_data.horizon == "2024-2050" + assert json.loads(additional_data.patch) == patch + + # Check that the additional data is correctly updated + new_patch = {"foo": "baz"} + a.additional_data.patch = json.dumps(new_patch) + repo.save(a) + additional_data = repo.get_additional_data(a_id) + assert json.loads(additional_data.patch) == new_patch + + # Check that the additional data is correctly deleted when the study is deleted + repo.delete(a_id) + assert repo.get_additional_data(a_id) is None + + def test_study_inheritance(db_session: Session) -> None: repo = StudyMetadataRepository(LocalCache(), session=db_session) From 8b0e5f3cb9a07d57beff3185e1ce3380d5a365ef Mon Sep 17 00:00:00 2001 From: Laurent LAPORTE Date: Wed, 21 Feb 2024 17:39:17 +0100 Subject: [PATCH 2/3] fix(variant-db): add cascade delete constraints on foreign keys --- antarest/study/storage/variantstudy/model/dbmodel.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/antarest/study/storage/variantstudy/model/dbmodel.py b/antarest/study/storage/variantstudy/model/dbmodel.py index 9272eb797f..f4a99d74e9 100644 --- a/antarest/study/storage/variantstudy/model/dbmodel.py +++ b/antarest/study/storage/variantstudy/model/dbmodel.py @@ -21,7 +21,7 @@ class VariantStudySnapshot(Base): # type: ignore id: str = Column( String(36), - ForeignKey("variantstudy.id"), + ForeignKey("variantstudy.id", ondelete="CASCADE"), primary_key=True, ) created_at: datetime.date = Column(DateTime) @@ -48,7 +48,7 @@ class CommandBlock(Base): # type: ignore default=lambda: str(uuid.uuid4()), unique=True, ) - study_id: str = Column(String(36), ForeignKey("variantstudy.id")) + study_id: str = Column(String(36), ForeignKey("variantstudy.id", ondelete="CASCADE")) index: int = Column(Integer) command: str = Column(String(255)) version: int = Column(Integer) From fd6a21f78bd84caa5309db04163e5c0eb317f264 Mon Sep 17 00:00:00 2001 From: Laurent LAPORTE Date: Wed, 21 Feb 2024 17:40:42 +0100 Subject: [PATCH 3/3] fix(variant-db): add a migration script to fix foreign key constraints --- ...4861_add_delete_cascade_variant_studies.py | 51 +++++++++++++++++++ scripts/rollback.sh | 2 +- 2 files changed, 52 insertions(+), 1 deletion(-) create mode 100644 alembic/versions/c0c4aaf84861_add_delete_cascade_variant_studies.py diff --git a/alembic/versions/c0c4aaf84861_add_delete_cascade_variant_studies.py b/alembic/versions/c0c4aaf84861_add_delete_cascade_variant_studies.py new file mode 100644 index 0000000000..cc1100eeba --- /dev/null +++ b/alembic/versions/c0c4aaf84861_add_delete_cascade_variant_studies.py @@ -0,0 +1,51 @@ +""" +Add delete cascade constraint to variant study foreign keys + +Revision ID: c0c4aaf84861 +Revises: fd73601a9075 +Create Date: 2024-02-21 17:29:48.736664 +""" +from alembic import op # type: ignore + +# revision identifiers, used by Alembic. +revision = "c0c4aaf84861" +down_revision = "fd73601a9075" +branch_labels = None +depends_on = None + +COMMAND_BLOCK_FK = "commandblock_study_id_fkey" +SNAPSHOT_FK = "variant_study_snapshot_id_fkey" + + +def upgrade() -> None: + dialect_name: str = op.get_context().dialect.name + + # SQLite doesn't support dropping foreign keys, so we need to ignore it here + if dialect_name == "postgresql": + # ### commands auto generated by Alembic - please adjust! ### + with op.batch_alter_table("commandblock", schema=None) as batch_op: + batch_op.drop_constraint(COMMAND_BLOCK_FK, type_="foreignkey") + batch_op.create_foreign_key(COMMAND_BLOCK_FK, "variantstudy", ["study_id"], ["id"], ondelete="CASCADE") + + with op.batch_alter_table("variant_study_snapshot", schema=None) as batch_op: + batch_op.drop_constraint(SNAPSHOT_FK, type_="foreignkey") + batch_op.create_foreign_key(SNAPSHOT_FK, "variantstudy", ["id"], ["id"], ondelete="CASCADE") + + # ### end Alembic commands ### + + +def downgrade() -> None: + dialect_name: str = op.get_context().dialect.name + + # SQLite doesn't support dropping foreign keys, so we need to ignore it here + if dialect_name == "postgresql": + # ### commands auto generated by Alembic - please adjust! ### + with op.batch_alter_table("variant_study_snapshot", schema=None) as batch_op: + batch_op.drop_constraint(SNAPSHOT_FK, type_="foreignkey") + batch_op.create_foreign_key(SNAPSHOT_FK, "variantstudy", ["id"], ["id"]) + + with op.batch_alter_table("commandblock", schema=None) as batch_op: + batch_op.drop_constraint(COMMAND_BLOCK_FK, type_="foreignkey") + batch_op.create_foreign_key(COMMAND_BLOCK_FK, "variantstudy", ["study_id"], ["id"]) + + # ### end Alembic commands ### diff --git a/scripts/rollback.sh b/scripts/rollback.sh index 46d04a7966..974ca59422 100755 --- a/scripts/rollback.sh +++ b/scripts/rollback.sh @@ -12,5 +12,5 @@ CUR_DIR=$(cd "$(dirname "$0")" && pwd) BASE_DIR=$(dirname "$CUR_DIR") cd "$BASE_DIR" -alembic downgrade 3c70366b10ea +alembic downgrade fd73601a9075 cd -