Skip to content

Commit

Permalink
Merge branch 'dev' into feat/ANT-583-import-output-zipped-that-are-alone
Browse files Browse the repository at this point in the history
  • Loading branch information
MartinBelthle committed Mar 6, 2024
2 parents bb713f8 + f7f082a commit 8ba012a
Show file tree
Hide file tree
Showing 73 changed files with 3,515 additions and 641 deletions.
13 changes: 13 additions & 0 deletions .github/workflows/commitlint.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
name: Lint Commit Messages
on: [pull_request]

permissions:
contents: read
pull-requests: read

jobs:
commitlint:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- uses: wagoid/commitlint-github-action@v5
Original file line number Diff line number Diff line change
@@ -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 ###
Original file line number Diff line number Diff line change
@@ -0,0 +1,123 @@
"""
Populate `tag` and `study_tag` tables from `patch` field in `study_additional_data` table
Revision ID: dae93f1d9110
Revises: 3c70366b10ea
Create Date: 2024-02-08 10:30:20.590919
"""
import collections
import itertools
import json
import secrets
import typing as t

import sqlalchemy as sa # type: ignore
from alembic import op
from sqlalchemy.engine import Connection # type: ignore

from antarest.study.css4_colors import COLOR_NAMES

# revision identifiers, used by Alembic.
revision = "dae93f1d9110"
down_revision = "3c70366b10ea"
branch_labels = None
depends_on = None


def _avoid_duplicates(tags: t.Iterable[str]) -> t.Sequence[str]:
"""Avoid duplicate tags (case insensitive)"""

upper_tags = {tag.upper(): tag for tag in tags}
return list(upper_tags.values())


def _load_patch_obj(patch: t.Optional[str]) -> t.MutableMapping[str, t.Any]:
"""Load the patch object from the `patch` field in the `study_additional_data` table."""

obj: t.MutableMapping[str, t.Any] = json.loads(patch or "{}")
obj["study"] = obj.get("study") or {}
obj["study"]["tags"] = _avoid_duplicates(obj["study"].get("tags") or [])
return obj


def upgrade() -> None:
"""
Populate `tag` and `study_tag` tables from `patch` field in `study_additional_data` table
Four steps to proceed:
- Retrieve study-tags pairs from patches in `study_additional_data`.
- Delete all rows in `tag` and `study_tag`, as tag updates between revised 3c70366b10ea and this version,
do modify the data in patches alongside the two previous tables.
- Populate `tag` table using unique tag-labels and by randomly generating their associated colors.
- Populate `study_tag` using study-tags pairs.
"""

# create connexion to the db
connexion: Connection = op.get_bind()

# retrieve the tags and the study-tag pairs from the db
study_tags = connexion.execute("SELECT study_id, patch FROM study_additional_data")
tags_by_ids: t.MutableMapping[str, t.Set[str]] = {}
for study_id, patch in study_tags:
obj = _load_patch_obj(patch)
tags_by_ids[study_id] = obj["study"]["tags"]

# delete rows in tables `tag` and `study_tag`
connexion.execute("DELETE FROM study_tag")
connexion.execute("DELETE FROM tag")

# insert the tags in the `tag` table
all_labels = {lbl.upper(): lbl for lbl in itertools.chain.from_iterable(tags_by_ids.values())}
bulk_tags = [{"label": label, "color": secrets.choice(COLOR_NAMES)} for label in all_labels.values()]
if bulk_tags:
sql = sa.text("INSERT INTO tag (label, color) VALUES (:label, :color)")
connexion.execute(sql, *bulk_tags)

# Create relationships between studies and tags in the `study_tag` table
bulk_study_tags = [
# fmt: off
{"study_id": id_, "tag_label": all_labels[lbl.upper()]}
for id_, tags in tags_by_ids.items()
for lbl in tags
# fmt: on
]
if bulk_study_tags:
sql = sa.text("INSERT INTO study_tag (study_id, tag_label) VALUES (:study_id, :tag_label)")
connexion.execute(sql, *bulk_study_tags)


def downgrade() -> None:
"""
Restore `patch` field in `study_additional_data` from `tag` and `study_tag` tables
Three steps to proceed:
- Retrieve study-tags pairs from `study_tag` table.
- Update patches study-tags in `study_additional_data` using these pairs.
- Delete all rows from `tag` and `study_tag`.
"""
# create a connection to the db
connexion: Connection = op.get_bind()

# Creating the `tags_by_ids` mapping from data in the `study_tags` table
tags_by_ids: t.MutableMapping[str, t.Set[str]] = collections.defaultdict(set)
study_tags = connexion.execute("SELECT study_id, tag_label FROM study_tag")
for study_id, tag_label in study_tags:
tags_by_ids[study_id].add(tag_label)

# Then, we read objects from the `patch` field of the `study_additional_data` table
objects_by_ids = {}
study_tags = connexion.execute("SELECT study_id, patch FROM study_additional_data")
for study_id, patch in study_tags:
obj = _load_patch_obj(patch)
obj["study"]["tags"] = _avoid_duplicates(tags_by_ids[study_id] | set(obj["study"]["tags"]))
objects_by_ids[study_id] = obj

# Updating objects in the `study_additional_data` table
bulk_patches = [{"study_id": id_, "patch": json.dumps(obj)} for id_, obj in objects_by_ids.items()]
if bulk_patches:
sql = sa.text("UPDATE study_additional_data SET patch = :patch WHERE study_id = :study_id")
connexion.execute(sql, *bulk_patches)

# Deleting study_tags and tags
connexion.execute("DELETE FROM study_tag")
connexion.execute("DELETE FROM tag")
86 changes: 86 additions & 0 deletions alembic/versions/fd73601a9075_add_delete_cascade_studies.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,86 @@
"""
Add delete cascade constraint to study foreign keys
Revision ID: fd73601a9075
Revises: 3c70366b10ea
Create Date: 2024-02-12 17:27:37.314443
"""
import sqlalchemy as sa # type: ignore
from alembic import op

# revision identifiers, used by Alembic.
revision = "fd73601a9075"
down_revision = "dae93f1d9110"
branch_labels = None
depends_on = None

# noinspection SpellCheckingInspection
RAWSTUDY_FK = "rawstudy_id_fkey"

# noinspection SpellCheckingInspection
VARIANTSTUDY_FK = "variantstudy_id_fkey"

# noinspection SpellCheckingInspection
STUDY_ADDITIONAL_DATA_FK = "study_additional_data_study_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":
with op.batch_alter_table("rawstudy", schema=None) as batch_op:
batch_op.drop_constraint(RAWSTUDY_FK, type_="foreignkey")
batch_op.create_foreign_key(RAWSTUDY_FK, "study", ["id"], ["id"], ondelete="CASCADE")

with op.batch_alter_table("study_additional_data", schema=None) as batch_op:
batch_op.drop_constraint(STUDY_ADDITIONAL_DATA_FK, type_="foreignkey")
batch_op.create_foreign_key(STUDY_ADDITIONAL_DATA_FK, "study", ["study_id"], ["id"], ondelete="CASCADE")

with op.batch_alter_table("variantstudy", schema=None) as batch_op:
batch_op.drop_constraint(VARIANTSTUDY_FK, type_="foreignkey")
batch_op.create_foreign_key(VARIANTSTUDY_FK, "study", ["id"], ["id"], ondelete="CASCADE")

with op.batch_alter_table("group_metadata", schema=None) as batch_op:
batch_op.alter_column("group_id", existing_type=sa.VARCHAR(length=36), nullable=False)
batch_op.alter_column("study_id", existing_type=sa.VARCHAR(length=36), nullable=False)
batch_op.create_index(batch_op.f("ix_group_metadata_group_id"), ["group_id"], unique=False)
batch_op.create_index(batch_op.f("ix_group_metadata_study_id"), ["study_id"], unique=False)
if dialect_name == "postgresql":
batch_op.drop_constraint("group_metadata_group_id_fkey", type_="foreignkey")
batch_op.drop_constraint("group_metadata_study_id_fkey", type_="foreignkey")
batch_op.create_foreign_key(
"group_metadata_group_id_fkey", "groups", ["group_id"], ["id"], ondelete="CASCADE"
)
batch_op.create_foreign_key(
"group_metadata_study_id_fkey", "study", ["study_id"], ["id"], ondelete="CASCADE"
)


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":
with op.batch_alter_table("rawstudy", schema=None) as batch_op:
batch_op.drop_constraint(RAWSTUDY_FK, type_="foreignkey")
batch_op.create_foreign_key(RAWSTUDY_FK, "study", ["id"], ["id"])

with op.batch_alter_table("study_additional_data", schema=None) as batch_op:
batch_op.drop_constraint(STUDY_ADDITIONAL_DATA_FK, type_="foreignkey")
batch_op.create_foreign_key(STUDY_ADDITIONAL_DATA_FK, "study", ["study_id"], ["id"])

with op.batch_alter_table("variantstudy", schema=None) as batch_op:
batch_op.drop_constraint(VARIANTSTUDY_FK, type_="foreignkey")
batch_op.create_foreign_key(VARIANTSTUDY_FK, "study", ["id"], ["id"])

with op.batch_alter_table("group_metadata", schema=None) as batch_op:
# SQLite doesn't support dropping foreign keys, so we need to ignore it here
if dialect_name == "postgresql":
batch_op.drop_constraint("group_metadata_study_id_fkey", type_="foreignkey")
batch_op.drop_constraint("group_metadata_group_id_fkey", type_="foreignkey")
batch_op.create_foreign_key("group_metadata_study_id_fkey", "study", ["study_id"], ["id"])
batch_op.create_foreign_key("group_metadata_group_id_fkey", "groups", ["group_id"], ["id"])
batch_op.drop_index(batch_op.f("ix_group_metadata_study_id"))
batch_op.drop_index(batch_op.f("ix_group_metadata_group_id"))
batch_op.alter_column("study_id", existing_type=sa.VARCHAR(length=36), nullable=True)
batch_op.alter_column("group_id", existing_type=sa.VARCHAR(length=36), nullable=True)
4 changes: 2 additions & 2 deletions antarest/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,9 +7,9 @@

# Standard project metadata

__version__ = "2.16.3"
__version__ = "2.16.6"
__author__ = "RTE, Antares Web Team"
__date__ = "2024-01-17"
__date__ = "2024-03-04"
# noinspection SpellCheckingInspection
__credits__ = "(c) Réseau de Transport de l’Électricité (RTE)"

Expand Down
8 changes: 8 additions & 0 deletions antarest/core/exceptions.py
Original file line number Diff line number Diff line change
Expand Up @@ -253,6 +253,14 @@ def __init__(self, *area_ids: str) -> None:
super().__init__(HTTPStatus.NOT_FOUND, msg)


class DuplicateAreaName(HTTPException):
"""Exception raised when trying to create an area with an already existing name."""

def __init__(self, area_name: str) -> None:
msg = f"Area '{area_name}' already exists and could not be created"
super().__init__(HTTPStatus.CONFLICT, msg)


class DistrictNotFound(HTTPException):
def __init__(self, *district_ids: str) -> None:
count = len(district_ids)
Expand Down
42 changes: 24 additions & 18 deletions antarest/core/filetransfer/service.py
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,8 @@ def request_download(
filename: str,
name: Optional[str] = None,
owner: Optional[JWTUser] = None,
use_notification: bool = True,
expiration_time_in_minutes: int = 0,
) -> FileDownload:
fh, path = tempfile.mkstemp(dir=self.tmp_dir, suffix=filename)
os.close(fh)
Expand All @@ -55,36 +57,40 @@ def request_download(
path=str(tmpfile),
owner=owner.impersonator if owner is not None else None,
expiration_date=datetime.datetime.utcnow()
+ datetime.timedelta(minutes=self.download_default_expiration_timeout_minutes),
+ datetime.timedelta(
minutes=expiration_time_in_minutes or self.download_default_expiration_timeout_minutes
),
)
self.repository.add(download)
self.event_bus.push(
Event(
type=EventType.DOWNLOAD_CREATED,
payload=download.to_dto(),
permissions=PermissionInfo(owner=owner.impersonator)
if owner
else PermissionInfo(public_mode=PublicMode.READ),
if use_notification:
self.event_bus.push(
Event(
type=EventType.DOWNLOAD_CREATED,
payload=download.to_dto(),
permissions=PermissionInfo(owner=owner.impersonator)
if owner
else PermissionInfo(public_mode=PublicMode.READ),
)
)
)
return download

def set_ready(self, download_id: str) -> None:
def set_ready(self, download_id: str, use_notification: bool = True) -> None:
download = self.repository.get(download_id)
if not download:
raise FileDownloadNotFound()

download.ready = True
self.repository.save(download)
self.event_bus.push(
Event(
type=EventType.DOWNLOAD_READY,
payload=download.to_dto(),
permissions=PermissionInfo(owner=download.owner)
if download.owner
else PermissionInfo(public_mode=PublicMode.READ),
if use_notification:
self.event_bus.push(
Event(
type=EventType.DOWNLOAD_READY,
payload=download.to_dto(),
permissions=PermissionInfo(owner=download.owner)
if download.owner
else PermissionInfo(public_mode=PublicMode.READ),
)
)
)

def fail(self, download_id: str, reason: str = "") -> None:
download = self.repository.get(download_id)
Expand Down
8 changes: 6 additions & 2 deletions antarest/launcher/service.py
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,7 @@
from antarest.launcher.repository import JobResultRepository
from antarest.launcher.ssh_client import calculates_slurm_load
from antarest.launcher.ssh_config import SSHConfigDTO
from antarest.study.repository import StudyFilter
from antarest.study.repository import AccessPermissions, StudyFilter
from antarest.study.service import StudyService
from antarest.study.storage.utils import assert_permission, extract_output_name, retrieve_output_path

Expand Down Expand Up @@ -313,7 +313,11 @@ def _filter_from_user_permission(self, job_results: List[JobResult], user: Optio
if study_ids:
studies = {
study.id: study
for study in self.study_service.repository.get_all(study_filter=StudyFilter(study_ids=study_ids))
for study in self.study_service.repository.get_all(
study_filter=StudyFilter(
study_ids=study_ids, access_permissions=AccessPermissions.from_params(user)
)
)
}
else:
studies = {}
Expand Down
Loading

0 comments on commit 8ba012a

Please sign in to comment.