From d8df7e901361405a267f36dff2721a63a8ad16b7 Mon Sep 17 00:00:00 2001 From: MartinBelthle Date: Thu, 19 Dec 2024 11:49:03 +0100 Subject: [PATCH 1/7] fix(xpansion): fix several issues related to weights and constraints (#2273) --- .../study/business/xpansion_management.py | 44 +++++++------ .../storage/business/test_xpansion_manager.py | 61 +++++++++++++++++-- 2 files changed, 81 insertions(+), 24 deletions(-) diff --git a/antarest/study/business/xpansion_management.py b/antarest/study/business/xpansion_management.py index c86dd5cab7..3f19e7c8f3 100644 --- a/antarest/study/business/xpansion_management.py +++ b/antarest/study/business/xpansion_management.py @@ -203,7 +203,7 @@ def from_config(cls, config_obj: JSON) -> "GetXpansionSettings": try: return cls(**config_obj) except ValidationError: - return cls.construct(**config_obj) + return cls.model_construct(**config_obj) @all_optional_model @@ -270,11 +270,6 @@ def __init__(self) -> None: super().__init__(http.HTTPStatus.BAD_REQUEST) -class WrongTypeFormat(HTTPException): - def __init__(self, message: str) -> None: - super().__init__(http.HTTPStatus.BAD_REQUEST, message) - - class WrongLinkFormatError(HTTPException): def __init__(self, message: str) -> None: super().__init__(http.HTTPStatus.BAD_REQUEST, message) @@ -295,11 +290,6 @@ def __init__(self, message: str) -> None: super().__init__(http.HTTPStatus.NOT_FOUND, message) -class ConstraintsNotFoundError(HTTPException): - def __init__(self, message: str) -> None: - super().__init__(http.HTTPStatus.NOT_FOUND, message) - - class FileCurrentlyUsedInSettings(HTTPException): def __init__(self, message: str) -> None: super().__init__(http.HTTPStatus.CONFLICT, message) @@ -337,7 +327,10 @@ def create_xpansion_configuration(self, study: Study, zipped_config: t.Optional[ xpansion_settings = XpansionSettings() settings_obj = xpansion_settings.model_dump( - mode="json", by_alias=True, exclude_none=True, exclude={"sensitivity_config"} + mode="json", + by_alias=True, + exclude_none=True, + exclude={"sensitivity_config", "yearly_weights", "additional_constraints"}, ) if xpansion_settings.sensitivity_config: sensitivity_obj = xpansion_settings.sensitivity_config.model_dump( @@ -389,22 +382,33 @@ def update_xpansion_settings( file_study = self.study_storage_service.get_storage(study).get_raw(study) - # Specific handling of the additional constraints file: - # - If the file name is `None`, it means that the user does not want to select an additional constraints file. - # - If the file name is empty, it means that the user wants to deselect the additional constraints file, - # but he does not want to delete it from the expansion configuration folder. - # - If the file name is not empty, it means that the user wants to select an additional constraints file. + # Specific handling for yearly_weights and additional_constraints: + # - If the attributes are given, it means that the user wants to select a file. # It is therefore necessary to check that the file exists. - constraints_file = new_xpansion_settings.additional_constraints - if constraints_file: + # - Else, it means the user want to deselect the additional constraints file, + # but he does not want to delete it from the expansion configuration folder. + excludes = {"sensitivity_config"} + if constraints_file := new_xpansion_settings.additional_constraints: try: constraints_url = ["user", "expansion", "constraints", constraints_file] file_study.tree.get(constraints_url) except ChildNotFoundError: msg = f"Additional constraints file '{constraints_file}' does not exist" raise XpansionFileNotFoundError(msg) from None + else: + excludes.add("additional_constraints") + + if weights_file := new_xpansion_settings.yearly_weights: + try: + weights_url = ["user", "expansion", "weights", weights_file] + file_study.tree.get(weights_url) + except ChildNotFoundError: + msg = f"Additional weights file '{weights_file}' does not exist" + raise XpansionFileNotFoundError(msg) from None + else: + excludes.add("yearly_weights") - config_obj = updated_settings.model_dump(mode="json", by_alias=True, exclude={"sensitivity_config"}) + config_obj = updated_settings.model_dump(mode="json", by_alias=True, exclude=excludes) file_study.tree.save(config_obj, ["user", "expansion", "settings"]) if new_xpansion_settings.sensitivity_config: diff --git a/tests/storage/business/test_xpansion_manager.py b/tests/storage/business/test_xpansion_manager.py index 5b1f8d1abd..325ddf7e0c 100644 --- a/tests/storage/business/test_xpansion_manager.py +++ b/tests/storage/business/test_xpansion_manager.py @@ -127,8 +127,6 @@ def set_up_xpansion_manager(tmp_path: Path) -> t.Tuple[FileStudy, RawStudy, Xpan "log_level": 0, "separation_parameter": 0.5, "batch_size": 96, - "yearly-weights": "", - "additional-constraints": "", "timelimit": int(1e12), }, "weights": {}, @@ -228,8 +226,6 @@ def test_update_xpansion_settings(tmp_path: Path) -> None: "max_iteration": 123, "uc_type": UcType.EXPANSION_FAST, "master": Master.INTEGER, - "yearly-weights": "", - "additional-constraints": "", "relaxed_optimality_gap": "1.2%", # percentage "relative_gap": 1e-12, "batch_size": 4, @@ -463,6 +459,63 @@ def test_update_constraints(tmp_path: Path) -> None: assert actual_settings.additional_constraints == "" +@pytest.mark.unit_test +def test_update_constraints_via_the_front(tmp_path: Path) -> None: + empty_study, study, xpansion_manager = set_up_xpansion_manager(tmp_path) + empty_study.tree.save({"user": {"expansion": {"constraints": {"constraints.txt": b"0"}}}}) + + # asserts we can update a field without writing the field additional constraint in the file + front_settings = UpdateXpansionSettings(master="relaxed") + xpansion_manager.update_xpansion_settings(study, front_settings) + json_content = empty_study.tree.get(["user", "expansion", "settings"]) + assert "additional-constraints" not in json_content + assert json_content["master"] == "relaxed" + + # asserts the front-end can fill additional constraints + new_constraint = {"additional-constraints": "constraints.txt"} + front_settings = UpdateXpansionSettings.model_validate(new_constraint) + actual_settings = xpansion_manager.update_xpansion_settings(study, front_settings) + assert actual_settings.additional_constraints == "constraints.txt" + json_content = empty_study.tree.get(["user", "expansion", "settings"]) + assert json_content["additional-constraints"] == "constraints.txt" + + # asserts the front-end can unselect this constraint by not filling it + front_settings = UpdateXpansionSettings() + actual_settings = xpansion_manager.update_xpansion_settings(study, front_settings) + assert actual_settings.additional_constraints == "" + json_content = empty_study.tree.get(["user", "expansion", "settings"]) + assert "additional-constraints" not in json_content + + +@pytest.mark.unit_test +def test_update_weights_via_the_front(tmp_path: Path) -> None: + # Same test as the one for constraints + empty_study, study, xpansion_manager = set_up_xpansion_manager(tmp_path) + empty_study.tree.save({"user": {"expansion": {"weights": {"weights.txt": b"0"}}}}) + + # asserts we can update a field without writing the field yearly-weights in the file + front_settings = UpdateXpansionSettings(master="relaxed") + xpansion_manager.update_xpansion_settings(study, front_settings) + json_content = empty_study.tree.get(["user", "expansion", "settings"]) + assert "yearly-weights" not in json_content + assert json_content["master"] == "relaxed" + + # asserts the front-end can fill yearly weights + new_constraint = {"yearly-weights": "weights.txt"} + front_settings = UpdateXpansionSettings.model_validate(new_constraint) + actual_settings = xpansion_manager.update_xpansion_settings(study, front_settings) + assert actual_settings.yearly_weights == "weights.txt" + json_content = empty_study.tree.get(["user", "expansion", "settings"]) + assert json_content["yearly-weights"] == "weights.txt" + + # asserts the front-end can unselect this weight by not filling it + front_settings = UpdateXpansionSettings() + actual_settings = xpansion_manager.update_xpansion_settings(study, front_settings) + assert actual_settings.yearly_weights == "" + json_content = empty_study.tree.get(["user", "expansion", "settings"]) + assert "yearly-weights" not in json_content + + @pytest.mark.unit_test def test_add_resources(tmp_path: Path) -> None: empty_study, study, xpansion_manager = set_up_xpansion_manager(tmp_path) From 4b68b036d0aa26d04276d140c6746503b2d5b214 Mon Sep 17 00:00:00 2001 From: Samir Kamal <1954121+skamril@users.noreply.github.com> Date: Fri, 20 Dec 2024 17:10:09 +0100 Subject: [PATCH 2/7] Revert "feat(ui-studies): add on click fetch and display list of non studies folder (#2224)" This reverts commit 763f3707a969da706d3b5b004103a1676e3b02ba. --- antarest/study/model.py | 17 +- antarest/study/storage/explorer_service.py | 6 +- antarest/study/web/explorer_blueprint.py | 6 +- .../explorer_blueprint/test_explorer.py | 6 +- .../storage/business/test_explorer_service.py | 12 +- webapp/public/locales/en/main.json | 5 - webapp/public/locales/fr/main.json | 6 +- webapp/src/components/App/Studies/SideNav.tsx | 2 +- .../App/Studies/StudiesList/index.tsx | 44 +--- .../src/components/App/Studies/StudyTree.tsx | 77 +++++++ .../Studies/StudyTree/__test__/fixtures.ts | 157 ------------- .../Studies/StudyTree/__test__/utils.test.tsx | 113 ---------- .../App/Studies/StudyTree/index.tsx | 154 ------------- .../components/App/Studies/StudyTree/utils.ts | 208 ------------------ webapp/src/components/App/Studies/utils.ts | 7 - webapp/src/redux/ducks/studies.ts | 2 +- webapp/src/services/api/study.ts | 38 +--- 17 files changed, 110 insertions(+), 750 deletions(-) create mode 100644 webapp/src/components/App/Studies/StudyTree.tsx delete mode 100644 webapp/src/components/App/Studies/StudyTree/__test__/fixtures.ts delete mode 100644 webapp/src/components/App/Studies/StudyTree/__test__/utils.test.tsx delete mode 100644 webapp/src/components/App/Studies/StudyTree/index.tsx delete mode 100644 webapp/src/components/App/Studies/StudyTree/utils.ts diff --git a/antarest/study/model.py b/antarest/study/model.py index 207662aeea..b8378aa356 100644 --- a/antarest/study/model.py +++ b/antarest/study/model.py @@ -19,7 +19,7 @@ from pathlib import Path from antares.study.version import StudyVersion -from pydantic import BeforeValidator, PlainSerializer, computed_field, field_validator +from pydantic import BeforeValidator, PlainSerializer, field_validator from sqlalchemy import ( # type: ignore Boolean, Column, @@ -323,7 +323,7 @@ class StudyFolder: groups: t.List[Group] -class NonStudyFolderDTO(AntaresBaseModel): +class NonStudyFolder(AntaresBaseModel): """ DTO used by the explorer to list directories that aren't studies directory, this will be usefull for the front so the user can navigate in the hierarchy @@ -333,19 +333,6 @@ class NonStudyFolderDTO(AntaresBaseModel): workspace: str name: str - @computed_field(alias="parentPath") - def parent_path(self) -> Path: - """ - This computed field is convenient for the front. - - This field is also aliased as parentPath to match the front-end naming convention. - - Returns: the parent path of the current directory. Starting with the workspace as a root directory (we want /workspafe/folder1/sub... and not workspace/folder1/fsub... ). - """ - workspace_path = Path(f"/{self.workspace}") - full_path = workspace_path.joinpath(self.path) - return full_path.parent - class WorkspaceMetadata(AntaresBaseModel): """ diff --git a/antarest/study/storage/explorer_service.py b/antarest/study/storage/explorer_service.py index fa9ef7fa30..5610f3e5f8 100644 --- a/antarest/study/storage/explorer_service.py +++ b/antarest/study/storage/explorer_service.py @@ -14,7 +14,7 @@ from typing import List from antarest.core.config import Config -from antarest.study.model import DEFAULT_WORKSPACE_NAME, NonStudyFolderDTO, WorkspaceMetadata +from antarest.study.model import DEFAULT_WORKSPACE_NAME, NonStudyFolder, WorkspaceMetadata from antarest.study.storage.utils import ( get_folder_from_workspace, get_workspace_from_config, @@ -33,7 +33,7 @@ def list_dir( self, workspace_name: str, workspace_directory_path: str, - ) -> List[NonStudyFolderDTO]: + ) -> List[NonStudyFolder]: """ return a list of all directories under workspace_directory_path, that aren't studies. """ @@ -44,7 +44,7 @@ def list_dir( if child.is_dir() and not is_study_folder(child) and not should_ignore_folder_for_scan(child): # we don't want to expose the full absolute path on the server child_rel_path = child.relative_to(workspace.path) - directories.append(NonStudyFolderDTO(path=child_rel_path, workspace=workspace_name, name=child.name)) + directories.append(NonStudyFolder(path=child_rel_path, workspace=workspace_name, name=child.name)) return directories def list_workspaces( diff --git a/antarest/study/web/explorer_blueprint.py b/antarest/study/web/explorer_blueprint.py index b453cab787..0981ba5214 100644 --- a/antarest/study/web/explorer_blueprint.py +++ b/antarest/study/web/explorer_blueprint.py @@ -18,7 +18,7 @@ from antarest.core.config import Config from antarest.core.jwt import JWTUser from antarest.login.auth import Auth -from antarest.study.model import NonStudyFolderDTO, WorkspaceMetadata +from antarest.study.model import NonStudyFolder, WorkspaceMetadata from antarest.study.storage.explorer_service import Explorer logger = logging.getLogger(__name__) @@ -40,13 +40,13 @@ def create_explorer_routes(config: Config, explorer: Explorer) -> APIRouter: @bp.get( "/explorer/{workspace}/_list_dir", summary="For a given directory, list sub directories that aren't studies", - response_model=List[NonStudyFolderDTO], + response_model=List[NonStudyFolder], ) def list_dir( workspace: str, path: str, current_user: JWTUser = Depends(auth.get_current_user), - ) -> List[NonStudyFolderDTO]: + ) -> List[NonStudyFolder]: """ Endpoint to list sub directories of a given directory Args: diff --git a/tests/integration/explorer_blueprint/test_explorer.py b/tests/integration/explorer_blueprint/test_explorer.py index 31990b1781..dbb6f83ebc 100644 --- a/tests/integration/explorer_blueprint/test_explorer.py +++ b/tests/integration/explorer_blueprint/test_explorer.py @@ -14,7 +14,7 @@ import pytest from starlette.testclient import TestClient -from antarest.study.model import NonStudyFolderDTO, WorkspaceMetadata +from antarest.study.model import NonStudyFolder, WorkspaceMetadata BAD_REQUEST_STATUS_CODE = 400 # Status code for directory listing with invalid parameters @@ -65,9 +65,9 @@ def test_explorer(client: TestClient, admin_access_token: str, study_tree: Path) ) res.raise_for_status() directories_res = res.json() - directories_res = [NonStudyFolderDTO(**d) for d in directories_res] + directories_res = [NonStudyFolder(**d) for d in directories_res] directorires_expected = [ - NonStudyFolderDTO( + NonStudyFolder( path=Path("folder/trash"), workspace="ext", name="trash", diff --git a/tests/storage/business/test_explorer_service.py b/tests/storage/business/test_explorer_service.py index 37a7c0c033..883e79cfca 100644 --- a/tests/storage/business/test_explorer_service.py +++ b/tests/storage/business/test_explorer_service.py @@ -15,7 +15,7 @@ import pytest from antarest.core.config import Config, StorageConfig, WorkspaceConfig -from antarest.study.model import DEFAULT_WORKSPACE_NAME, NonStudyFolderDTO, WorkspaceMetadata +from antarest.study.model import DEFAULT_WORKSPACE_NAME, NonStudyFolder, WorkspaceMetadata from antarest.study.storage.explorer_service import Explorer @@ -85,7 +85,8 @@ def test_list_dir_empty_string(config_scenario_a: Config): result = explorer.list_dir("diese", "") assert len(result) == 1 - assert result[0] == NonStudyFolderDTO(path=Path("folder"), workspace="diese", name="folder") + workspace_path = config_scenario_a.get_workspace_path(workspace="diese") + assert result[0] == NonStudyFolder(path=Path("folder"), workspace="diese", name="folder") @pytest.mark.unit_test @@ -94,10 +95,11 @@ def test_list_dir_several_subfolders(config_scenario_a: Config): result = explorer.list_dir("diese", "folder") assert len(result) == 3 + workspace_path = config_scenario_a.get_workspace_path(workspace="diese") folder_path = Path("folder") - assert NonStudyFolderDTO(path=(folder_path / "subfolder1"), workspace="diese", name="subfolder1") in result - assert NonStudyFolderDTO(path=(folder_path / "subfolder2"), workspace="diese", name="subfolder2") in result - assert NonStudyFolderDTO(path=(folder_path / "subfolder3"), workspace="diese", name="subfolder3") in result + assert NonStudyFolder(path=(folder_path / "subfolder1"), workspace="diese", name="subfolder1") in result + assert NonStudyFolder(path=(folder_path / "subfolder2"), workspace="diese", name="subfolder2") in result + assert NonStudyFolder(path=(folder_path / "subfolder3"), workspace="diese", name="subfolder3") in result @pytest.mark.unit_test diff --git a/webapp/public/locales/en/main.json b/webapp/public/locales/en/main.json index 1f13b6e551..250a75f2e6 100644 --- a/webapp/public/locales/en/main.json +++ b/webapp/public/locales/en/main.json @@ -642,9 +642,7 @@ "studies.studylaunched": "{{studyname}} launched!", "studies.copySuffix": "Copy", "studies.filters.strictfolder": "Show only direct folder children", - "studies.filters.showAllDescendants": "Show all children", "studies.scanFolder": "Scan folder", - "studies.requestDeepScan": "Recursive scan", "studies.moveStudy": "Move", "studies.movefolderplaceholder": "Path separated by '/'", "studies.importcopy": "Copy to database", @@ -676,9 +674,6 @@ "studies.exportOutputFilter": "Export filtered output", "studies.selectOutput": "Select an output", "studies.variant": "Variant", - "studies.tree.error.failToFetchWorkspace": "Failed to load workspaces", - "studies.tree.error.failToFetchFolder": "Failed to load subfolders for {{path}}", - "studies.tree.error.detailsInConsole": "Details logged in the console", "variants.createNewVariant": "Create new variant", "variants.newVariant": "New variant", "variants.newCommand": "Add new command", diff --git a/webapp/public/locales/fr/main.json b/webapp/public/locales/fr/main.json index 14dbcda923..98d6ee71b9 100644 --- a/webapp/public/locales/fr/main.json +++ b/webapp/public/locales/fr/main.json @@ -642,9 +642,7 @@ "studies.studylaunched": "{{studyname}} lancé(s) !", "studies.copySuffix": "Copie", "studies.filters.strictfolder": "Afficher uniquement les descendants directs", - "studies.filters.showAllDescendants": "Voir les sous-dossiers", "studies.scanFolder": "Scanner le dossier", - "studies.requestDeepScan": "Scan récursif", "studies.moveStudy": "Déplacer", "studies.movefolderplaceholder": "Chemin séparé par des '/'", "studies.importcopy": "Copier en base", @@ -675,9 +673,7 @@ "studies.exportOutput": "Exporter une sortie", "studies.exportOutputFilter": "Exporter une sortie filtrée", "studies.selectOutput": "Selectionnez une sortie", - "studies.tree.error.failToFetchWorkspace": "Échec lors de la récupération de l'espace de travail", - "studies.tree.error.failToFetchFolder": "Échec lors de la récupération des sous dossiers de {{path}}", - "studies.tree.error.detailsInConsole": "Détails de l'érreur dans la console", + "studies.variant": "Variante", "variants.createNewVariant": "Créer une nouvelle variante", "variants.newVariant": "Nouvelle variante", "variants.newCommand": "Ajouter une nouvelle commande", diff --git a/webapp/src/components/App/Studies/SideNav.tsx b/webapp/src/components/App/Studies/SideNav.tsx index b966f27fd6..c0009ce7d2 100644 --- a/webapp/src/components/App/Studies/SideNav.tsx +++ b/webapp/src/components/App/Studies/SideNav.tsx @@ -16,7 +16,7 @@ import { useNavigate } from "react-router"; import { Box, Typography, List, ListItem, ListItemText } from "@mui/material"; import { useTranslation } from "react-i18next"; import { STUDIES_SIDE_NAV_WIDTH } from "../../../theme"; -import StudyTree from "@/components/App/Studies/StudyTree"; +import StudyTree from "./StudyTree"; import useAppSelector from "../../../redux/hooks/useAppSelector"; import { getFavoriteStudies } from "../../../redux/selectors"; diff --git a/webapp/src/components/App/Studies/StudiesList/index.tsx b/webapp/src/components/App/Studies/StudiesList/index.tsx index b8b32f1646..6d90785471 100644 --- a/webapp/src/components/App/Studies/StudiesList/index.tsx +++ b/webapp/src/components/App/Studies/StudiesList/index.tsx @@ -33,8 +33,7 @@ import AutoSizer from "react-virtualized-auto-sizer"; import HomeIcon from "@mui/icons-material/Home"; import ArrowUpwardIcon from "@mui/icons-material/ArrowUpward"; import ArrowDownwardIcon from "@mui/icons-material/ArrowDownward"; -import FolderIcon from "@mui/icons-material/Folder"; -import AccountTreeIcon from "@mui/icons-material/AccountTree"; +import FolderOffIcon from "@mui/icons-material/FolderOff"; import RadarIcon from "@mui/icons-material/Radar"; import { FixedSizeGrid, GridOnScrollProps } from "react-window"; import { v4 as uuidv4 } from "uuid"; @@ -62,7 +61,6 @@ import RefreshButton from "../RefreshButton"; import { scanFolder } from "../../../../services/api/study"; import useEnqueueErrorSnackbar from "../../../../hooks/useEnqueueErrorSnackbar"; import ConfirmationDialog from "../../../common/dialogs/ConfirmationDialog"; -import CheckBoxFE from "@/components/common/fieldEditors/CheckBoxFE"; const CARD_TARGET_WIDTH = 500; const CARD_HEIGHT = 250; @@ -90,7 +88,6 @@ function StudiesList(props: StudiesListProps) { const [selectedStudies, setSelectedStudies] = useState([]); const [selectionMode, setSelectionMode] = useState(false); const [confirmFolderScan, setConfirmFolderScan] = useState(false); - const [isRecursiveScan, setIsRecursiveScan] = useState(false); useEffect(() => { setFolderList(folder.split("/")); @@ -159,18 +156,13 @@ function StudiesList(props: StudiesListProps) { try { // Remove "/root" from the path const folder = folderList.slice(1).join("/"); - await scanFolder(folder, isRecursiveScan); + await scanFolder(folder); setConfirmFolderScan(false); - setIsRecursiveScan(false); } catch (e) { enqueueErrorSnackbar(t("studies.error.scanFolder"), e as AxiosError); } }; - const handleRecursiveScan = () => { - setIsRecursiveScan(!isRecursiveScan); - }; - //////////////////////////////////////////////////////////////// // Utils //////////////////////////////////////////////////////////////// @@ -257,21 +249,13 @@ function StudiesList(props: StudiesListProps) { ({`${studyIds.length} ${t("global.studies").toLowerCase()}`}) - - {strictFolderFilter ? ( - - - - - - ) : ( - - - - - - )} - + + + + + {folder !== "root" && ( setConfirmFolderScan(true)}> @@ -282,20 +266,12 @@ function StudiesList(props: StudiesListProps) { {folder !== "root" && confirmFolderScan && ( { - setConfirmFolderScan(false); - setIsRecursiveScan(false); - }} + onCancel={() => setConfirmFolderScan(false)} onConfirm={handleFolderScan} alert="warning" open > {`${t("studies.scanFolder")} ${folder}?`} - )} diff --git a/webapp/src/components/App/Studies/StudyTree.tsx b/webapp/src/components/App/Studies/StudyTree.tsx new file mode 100644 index 0000000000..7208caaec4 --- /dev/null +++ b/webapp/src/components/App/Studies/StudyTree.tsx @@ -0,0 +1,77 @@ +/** + * Copyright (c) 2024, RTE (https://www.rte-france.com) + * + * See AUTHORS.txt + * + * This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * SPDX-License-Identifier: MPL-2.0 + * + * This file is part of the Antares project. + */ + +import { StudyTreeNode } from "./utils"; +import useAppSelector from "../../../redux/hooks/useAppSelector"; +import { getStudiesTree, getStudyFilters } from "../../../redux/selectors"; +import useAppDispatch from "../../../redux/hooks/useAppDispatch"; +import { updateStudyFilters } from "../../../redux/ducks/studies"; +import TreeItemEnhanced from "../../common/TreeItemEnhanced"; +import { SimpleTreeView } from "@mui/x-tree-view/SimpleTreeView"; +import { getParentPaths } from "../../../utils/pathUtils"; +import * as R from "ramda"; + +function StudyTree() { + const folder = useAppSelector((state) => getStudyFilters(state).folder, R.T); + const studiesTree = useAppSelector(getStudiesTree); + const dispatch = useAppDispatch(); + + //////////////////////////////////////////////////////////////// + // Event Handlers + //////////////////////////////////////////////////////////////// + + const handleTreeItemClick = (itemId: string) => { + dispatch(updateStudyFilters({ folder: itemId })); + }; + + //////////////////////////////////////////////////////////////// + // JSX + //////////////////////////////////////////////////////////////// + + const buildTree = (children: StudyTreeNode[], parentId?: string) => { + return children.map((child) => { + const id = parentId ? `${parentId}/${child.name}` : child.name; + + return ( + handleTreeItemClick(id)} + > + {buildTree(child.children, id)} + + ); + }); + }; + + return ( + + {buildTree([studiesTree])} + + ); +} + +export default StudyTree; diff --git a/webapp/src/components/App/Studies/StudyTree/__test__/fixtures.ts b/webapp/src/components/App/Studies/StudyTree/__test__/fixtures.ts deleted file mode 100644 index a6a4dbc3ad..0000000000 --- a/webapp/src/components/App/Studies/StudyTree/__test__/fixtures.ts +++ /dev/null @@ -1,157 +0,0 @@ -/** - * Copyright (c) 2024, RTE (https://www.rte-france.com) - * - * See AUTHORS.txt - * - * This Source Code Form is subject to the terms of the Mozilla Public - * License, v. 2.0. If a copy of the MPL was not distributed with this - * file, You can obtain one at http://mozilla.org/MPL/2.0/. - * - * SPDX-License-Identifier: MPL-2.0 - * - * This file is part of the Antares project. - */ - -export const FIXTURES = { - basicTree: { - name: "Basic tree with single level", - studyTree: { - name: "Root", - path: "/", - children: [ - { name: "a", path: "/a", children: [] }, - { name: "b", path: "/b", children: [] }, - ], - }, - folders: [ - { - name: "folder1", - path: "folder1", - workspace: "a", - parentPath: "/a", - }, - ], - expected: { - name: "Root", - path: "/", - children: [ - { - name: "a", - path: "/a", - children: [{ name: "folder1", path: "/a/folder1", children: [] }], - }, - { name: "b", path: "/b", children: [] }, - ], - }, - }, - nestedTree: { - name: "Nested tree structure", - studyTree: { - name: "Root", - path: "/", - children: [ - { - name: "a", - path: "/a", - children: [{ name: "suba", path: "/a/suba", children: [] }], - }, - ], - }, - folders: [ - { - name: "folder1", - path: "suba/folder1", - workspace: "a", - parentPath: "/a/suba", - }, - ], - expected: { - name: "Root", - path: "/", - children: [ - { - name: "a", - path: "/a", - children: [ - { - name: "suba", - path: "/a/suba", - children: [ - { name: "folder1", path: "/a/suba/folder1", children: [] }, - ], - }, - ], - }, - ], - }, - }, - duplicateCase: { - name: "Tree with potential duplicates", - studyTree: { - name: "Root", - path: "/", - children: [ - { - name: "a", - path: "/a", - children: [{ name: "folder1", path: "/a/folder1", children: [] }], - }, - ], - }, - folders: [ - { - name: "folder1", - path: "/folder1", - workspace: "a", - parentPath: "/a", - }, - ], - expected: { - name: "Root", - path: "/", - children: [ - { - name: "a", - path: "/a", - children: [{ name: "folder1", path: "/a/folder1", children: [] }], - }, - ], - }, - }, - multipleFolders: { - name: "Multiple folders merge", - studyTree: { - name: "Root", - path: "/", - children: [{ name: "a", path: "/a", children: [] }], - }, - folders: [ - { - name: "folder1", - path: "/folder1", - workspace: "a", - parentPath: "/a", - }, - { - name: "folder2", - path: "/folder2", - workspace: "a", - parentPath: "/a", - }, - ], - expected: { - name: "Root", - path: "/", - children: [ - { - name: "a", - path: "/a", - children: [ - { name: "folder1", path: "/a/folder1", children: [] }, - { name: "folder2", path: "/a/folder2", children: [] }, - ], - }, - ], - }, - }, -}; diff --git a/webapp/src/components/App/Studies/StudyTree/__test__/utils.test.tsx b/webapp/src/components/App/Studies/StudyTree/__test__/utils.test.tsx deleted file mode 100644 index 87a5419c54..0000000000 --- a/webapp/src/components/App/Studies/StudyTree/__test__/utils.test.tsx +++ /dev/null @@ -1,113 +0,0 @@ -/** - * Copyright (c) 2024, RTE (https://www.rte-france.com) - * - * See AUTHORS.txt - * - * This Source Code Form is subject to the terms of the Mozilla Public - * License, v. 2.0. If a copy of the MPL was not distributed with this - * file, You can obtain one at http://mozilla.org/MPL/2.0/. - * - * SPDX-License-Identifier: MPL-2.0 - * - * This file is part of the Antares project. - */ - -import { insertFoldersIfNotExist, insertWorkspacesIfNotExist } from "../utils"; -import { NonStudyFolderDTO, StudyTreeNode } from "../../utils"; -import { FIXTURES } from "./fixtures"; - -describe("StudyTree Utils", () => { - describe("mergeStudyTreeAndFolders", () => { - test.each(Object.values(FIXTURES))( - "$name", - ({ studyTree, folders, expected }) => { - const result = insertFoldersIfNotExist(studyTree, folders); - expect(result).toEqual(expected); - }, - ); - - test("should handle empty study tree", () => { - const emptyTree: StudyTreeNode = { - name: "Root", - path: "/", - children: [], - }; - const result = insertFoldersIfNotExist(emptyTree, []); - expect(result).toEqual(emptyTree); - }); - - test("should handle empty folders array", () => { - const tree: StudyTreeNode = { - name: "Root", - path: "/", - children: [{ name: "a", path: "/a", children: [] }], - }; - const result = insertFoldersIfNotExist(tree, []); - expect(result).toEqual(tree); - }); - - test("should handle invalid parent paths", () => { - const tree: StudyTreeNode = { - name: "Root", - path: "/", - children: [{ name: "a", path: "/a", children: [] }], - }; - const invalidFolder: NonStudyFolderDTO = { - name: "invalid", - path: "/invalid", - workspace: "nonexistent", - parentPath: "/nonexistent", - }; - const result = insertFoldersIfNotExist(tree, [invalidFolder]); - expect(result).toEqual(tree); - }); - - test("should handle empty workspaces", () => { - const tree: StudyTreeNode = { - name: "Root", - path: "/", - children: [ - { - name: "a", - path: "/a", - children: [{ name: "suba", path: "/a/suba", children: [] }], - }, - ], - }; - const workspaces: string[] = []; - const result = insertWorkspacesIfNotExist(tree, workspaces); - expect(result).toEqual(tree); - }); - - test("should merge workspaces", () => { - const tree: StudyTreeNode = { - name: "Root", - path: "/", - children: [ - { - name: "a", - path: "/a", - children: [{ name: "suba", path: "/a/suba", children: [] }], - }, - ], - }; - const expected: StudyTreeNode = { - name: "Root", - path: "/", - children: [ - { - name: "a", - path: "/a", - children: [{ name: "suba", path: "/a/suba", children: [] }], - }, - { name: "workspace1", path: "/workspace1", children: [] }, - { name: "workspace2", path: "/workspace2", children: [] }, - ], - }; - - const workspaces: string[] = ["a", "workspace1", "workspace2"]; - const result = insertWorkspacesIfNotExist(tree, workspaces); - expect(result).toEqual(expected); - }); - }); -}); diff --git a/webapp/src/components/App/Studies/StudyTree/index.tsx b/webapp/src/components/App/Studies/StudyTree/index.tsx deleted file mode 100644 index 510d70134e..0000000000 --- a/webapp/src/components/App/Studies/StudyTree/index.tsx +++ /dev/null @@ -1,154 +0,0 @@ -/** - * Copyright (c) 2024, RTE (https://www.rte-france.com) - * - * See AUTHORS.txt - * - * This Source Code Form is subject to the terms of the Mozilla Public - * License, v. 2.0. If a copy of the MPL was not distributed with this - * file, You can obtain one at http://mozilla.org/MPL/2.0/. - * - * SPDX-License-Identifier: MPL-2.0 - * - * This file is part of the Antares project. - */ - -import { StudyTreeNode } from ".././utils"; -import useAppSelector from "../../../../redux/hooks/useAppSelector"; -import { getStudiesTree, getStudyFilters } from "../../../../redux/selectors"; -import useAppDispatch from "../../../../redux/hooks/useAppDispatch"; -import { updateStudyFilters } from "../../../../redux/ducks/studies"; -import TreeItemEnhanced from "../../../common/TreeItemEnhanced"; -import { SimpleTreeView } from "@mui/x-tree-view/SimpleTreeView"; -import { getParentPaths } from "../../../../utils/pathUtils"; -import * as R from "ramda"; -import { useState } from "react"; -import useEnqueueErrorSnackbar from "@/hooks/useEnqueueErrorSnackbar"; -import useUpdateEffectOnce from "@/hooks/useUpdateEffectOnce"; -import { fetchAndInsertSubfolders, fetchAndInsertWorkspaces } from "./utils"; -import { useTranslation } from "react-i18next"; -import { toError } from "@/utils/fnUtils"; - -function StudyTree() { - const initialStudiesTree = useAppSelector(getStudiesTree); - const [studiesTree, setStudiesTree] = useState(initialStudiesTree); - const folder = useAppSelector((state) => getStudyFilters(state).folder, R.T); - const enqueueErrorSnackbar = useEnqueueErrorSnackbar(); - const dispatch = useAppDispatch(); - const [t] = useTranslation(); - - // Initialize folders once we have the tree - // we use useUpdateEffectOnce because at first render initialStudiesTree isn't initialized - useUpdateEffectOnce(() => { - updateTree("root", initialStudiesTree); - }, [initialStudiesTree]); - - /** - * This function is called at the initialization of the component and when the user clicks on a folder. - * - * The study tree is built from the studies in the database. There's a scan process that run on the server - * to update continuously the studies in the database. - * - * However this process can take a long time, and the user shouldn't wait for hours before he can see a study he knows is already uploaded. - * - * Instead of relying on the scan process to update the tree, we'll allow the user to walk into the tree and run a scan process only when he needs to. - * - * To enable this, we'll fetch the subfolders of a folder when the user clicks on it using the explorer API. - * - * @param itemId - The id of the item clicked - * @param studyTreeNode - The node of the item clicked - */ - async function updateTree(itemId: string, studyTreeNode: StudyTreeNode) { - let treeAfterWorkspacesUpdate = studiesTree; - let chidrenPaths = studyTreeNode.children.map( - (child) => `root${child.path}`, - ); - // If the user clicks on the root folder, we fetch the workspaces and insert them. - // Then we fetch the direct subfolders of the workspaces. - if (itemId === "root") { - try { - treeAfterWorkspacesUpdate = await fetchAndInsertWorkspaces(studiesTree); - chidrenPaths = treeAfterWorkspacesUpdate.children.map( - (child) => `root${child.path}`, - ); - } catch (error) { - enqueueErrorSnackbar( - "studies.tree.error.failToFetchWorkspace", - toError(error), - ); - } - } else { - // If the user clicks on a folder, we add the path of the clicked folder to the list of paths to fetch. - // as well as the path of the children of the clicked folder. - // If we don't fetch the subfolders of the children then we won't know if they're themselves folders, which we need - // to know to display the little arrow next to the subfolder. - // On the other hand, if we fetch only the subfolders of the children, then we won't fetch their "siblings" folder - // if one of them is added. - chidrenPaths = [studyTreeNode.path].concat(chidrenPaths); - } - - const [treeAfterChildrenUpdate, failedPath] = - await fetchAndInsertSubfolders(chidrenPaths, treeAfterWorkspacesUpdate); - if (failedPath.length > 0) { - enqueueErrorSnackbar( - t("studies.tree.error.failToFetchFolder", { - path: failedPath.join(" "), - interpolation: { escapeValue: false }, - }), - t("studies.tree.error.detailsInConsole"), - ); - } - setStudiesTree(treeAfterChildrenUpdate); - } - - //////////////////////////////////////////////////////////////// - // Event Handlers - //////////////////////////////////////////////////////////////// - - const handleTreeItemClick = async ( - itemId: string, - studyTreeNode: StudyTreeNode, - ) => { - dispatch(updateStudyFilters({ folder: itemId })); - updateTree(itemId, studyTreeNode); - }; - - //////////////////////////////////////////////////////////////// - // JSX - //////////////////////////////////////////////////////////////// - - const buildTree = (children: StudyTreeNode[], parentId?: string) => { - return children.map((child) => { - const id = parentId ? `${parentId}/${child.name}` : child.name; - - return ( - handleTreeItemClick(id, child)} - > - {buildTree(child.children, id)} - - ); - }); - }; - - return ( - - {buildTree([studiesTree])} - - ); -} - -export default StudyTree; diff --git a/webapp/src/components/App/Studies/StudyTree/utils.ts b/webapp/src/components/App/Studies/StudyTree/utils.ts deleted file mode 100644 index 14c4a5b008..0000000000 --- a/webapp/src/components/App/Studies/StudyTree/utils.ts +++ /dev/null @@ -1,208 +0,0 @@ -/** - * Copyright (c) 2024, RTE (https://www.rte-france.com) - * - * See AUTHORS.txt - * - * This Source Code Form is subject to the terms of the Mozilla Public - * License, v. 2.0. If a copy of the MPL was not distributed with this - * file, You can obtain one at http://mozilla.org/MPL/2.0/. - * - * SPDX-License-Identifier: MPL-2.0 - * - * This file is part of the Antares project. - */ - -import { NonStudyFolderDTO, StudyTreeNode } from "../utils"; -import * as api from "../../../../services/api/study"; - -/** - * Add a folder that was returned by the explorer into the study tree view. - * - * This function doesn't mutate the tree, it returns a new tree with the folder inserted. - * - * If the folder is already in the tree, the tree returnred will be equal to the tree given to the function. - * - * @param studiesTree study tree to insert the folder into - * @param folder folder to inert into the tree - * @returns study tree with the folder inserted if it wasn't already there. - * New branch is created if it contain the folder otherwise the branch is left unchanged. - */ -function insertFolderIfNotExist( - studiesTree: StudyTreeNode, - folder: NonStudyFolderDTO, -): StudyTreeNode { - // Early return if folder doesn't belong in this branch - if (!folder.parentPath.startsWith(studiesTree.path)) { - return studiesTree; - } - - // direct child case - if (folder.parentPath == studiesTree.path) { - const folderExists = studiesTree.children.some( - (child) => child.name === folder.name, - ); - if (folderExists) { - return studiesTree; - } - // parent path is the same, but no folder with the same name at this level - return { - ...studiesTree, - children: [ - ...studiesTree.children, - { - path: `${folder.parentPath}/${folder.name}`, - name: folder.name, - children: [], - }, - ], - }; - } - - // not a direct child, but does belong to this branch so recursively walk though the tree - return { - ...studiesTree, - children: studiesTree.children.map((child) => - insertFolderIfNotExist(child, folder), - ), - }; -} - -/** - * Insert several folders in the study tree if they don't exist already in the tree. - * - * This function doesn't mutate the tree, it returns a new tree with the folders inserted - * - * The folders are inserted in the order they are given. - * - * @param studiesTree study tree to insert the folder into - * @param folders folders to inert into the tree - * @param studiesTree study tree to insert the folder into - * @param folder folder to inert into the tree - * @returns study tree with the folder inserted if it wasn't already there. - * New branch is created if it contain the folder otherwise the branch is left unchanged. - */ -export function insertFoldersIfNotExist( - studiesTree: StudyTreeNode, - folders: NonStudyFolderDTO[], -): StudyTreeNode { - return folders.reduce( - (tree, folder) => insertFolderIfNotExist(tree, folder), - studiesTree, - ); -} - -/** - * Call the explorer api to fetch the subfolders under the given path. - * - * @param path path of the subfolder to fetch, should sart with root, e.g. root/workspace/folder1 - * @returns list of subfolders under the given path - */ -async function fetchSubfolders(path: string): Promise { - if (path === "root") { - // Under root there're workspaces not subfolders - return []; - } - // less than 2 parts means we're at the root level - const pathParts = path.split("/"); - if (pathParts.length < 2) { - return []; - } - // path parts should be ["root", workspace, "folder1", ...] - const workspace = pathParts[1]; - const subPath = pathParts.slice(2).join("/"); - return api.getFolders(workspace, subPath); -} - -/** - * Fetch and insert the subfolders under the given paths into the study tree. - * - * This function is used to fill the study tree when the user clicks on a folder. - * - * Subfolders are inserted only if they don't exist already in the tree. - * - * This function doesn't mutate the tree, it returns a new tree with the subfolders inserted - * - * @param paths list of paths to fetch the subfolders for - * @param studiesTree study tree to insert the subfolders into - * @returns a tuple with study tree with the subfolders inserted if they weren't already there and path for which - * the fetch failed. - */ -export async function fetchAndInsertSubfolders( - paths: string[], - studiesTree: StudyTreeNode, -): Promise<[StudyTreeNode, string[]]> { - const results = await Promise.allSettled( - paths.map((path) => fetchSubfolders(path)), - ); - - return results.reduce<[StudyTreeNode, string[]]>( - ([tree, failed], result, index) => { - if (result.status === "fulfilled") { - return [insertFoldersIfNotExist(tree, result.value), failed]; - } - console.error("Failed to load path:", paths[index], result.reason); - return [tree, [...failed, paths[index]]]; - }, - [studiesTree, []], - ); -} - -/** - * Insert a workspace into the study tree if it doesn't exist already. - * - * This function doesn't mutate the tree, it returns a new tree with the workspace inserted. - * - * @param workspace key of the workspace - * @param stydyTree study tree to insert the workspace into - * @returns study tree with the empty workspace inserted if it wasn't already there. - */ -function insertWorkspaceIfNotExist( - stydyTree: StudyTreeNode, - workspace: string, -) { - const emptyNode = { name: workspace, path: `/${workspace}`, children: [] }; - if (stydyTree.children.some((child) => child.name === workspace)) { - return stydyTree; - } - return { - ...stydyTree, - children: [...stydyTree.children, emptyNode], - }; -} - -/** - * Insert several workspaces into the study tree if they don't exist already in the tree. - * - * This function doesn't mutate the tree, it returns a new tree with the workspaces inserted. - * - * The workspaces are inserted in the order they are given. - * - * @param workspaces workspaces to insert into the tree - * @param stydyTree study tree to insert the workspaces into - * @returns study tree with the empty workspaces inserted if they weren't already there. - */ -export function insertWorkspacesIfNotExist( - stydyTree: StudyTreeNode, - workspaces: string[], -): StudyTreeNode { - return workspaces.reduce((acc, workspace) => { - return insertWorkspaceIfNotExist(acc, workspace); - }, stydyTree); -} - -/** - * Fetch and insert the workspaces into the study tree. - * - * Workspaces are inserted only if they don't exist already in the tree. - * - * This function doesn't mutate the tree, it returns a new tree with the workspaces inserted. - * - * @param studyTree study tree to insert the workspaces into - * @returns study tree with the workspaces inserted if they weren't already there. - */ -export async function fetchAndInsertWorkspaces( - studyTree: StudyTreeNode, -): Promise { - const workspaces = await api.getWorkspaces(); - return insertWorkspacesIfNotExist(studyTree, workspaces); -} diff --git a/webapp/src/components/App/Studies/utils.ts b/webapp/src/components/App/Studies/utils.ts index 8117450702..3f2ff61564 100644 --- a/webapp/src/components/App/Studies/utils.ts +++ b/webapp/src/components/App/Studies/utils.ts @@ -20,13 +20,6 @@ export interface StudyTreeNode { children: StudyTreeNode[]; } -export interface NonStudyFolderDTO { - name: string; - path: string; - workspace: string; - parentPath: string; -} - /** * Builds a tree structure from a list of study metadata. * diff --git a/webapp/src/redux/ducks/studies.ts b/webapp/src/redux/ducks/studies.ts index 9b4603a23f..3b3e8b04d2 100644 --- a/webapp/src/redux/ducks/studies.ts +++ b/webapp/src/redux/ducks/studies.ts @@ -94,7 +94,7 @@ const initialState = studiesAdapter.getInitialState({ filters: { inputValue: "", folder: "root", - strictFolder: true, + strictFolder: false, managed: false, archived: false, variant: false, diff --git a/webapp/src/services/api/study.ts b/webapp/src/services/api/study.ts index 349134d790..53d4a67925 100644 --- a/webapp/src/services/api/study.ts +++ b/webapp/src/services/api/study.ts @@ -34,11 +34,6 @@ import { getConfig } from "../config"; import { convertStudyDtoToMetadata } from "../utils"; import { FileDownloadTask } from "./downloads"; import { StudyMapDistrict } from "../../redux/ducks/studyMaps"; -import { NonStudyFolderDTO } from "@/components/App/Studies/utils"; - -interface Workspace { - name: string; -} const getStudiesRaw = async (): Promise> => { const res = await client.get(`/v1/studies`); @@ -53,30 +48,6 @@ export const getStudies = async (): Promise => { }); }; -export const getWorkspaces = async (): Promise => { - const res = await client.get( - `/v1/private/explorer/_list_workspaces`, - ); - return res.data.map((folder) => folder.name); -}; - -/** - * Call the explorer API to get the list of folders in a workspace - * - * @param workspace - workspace name - * @param folderPath - path starting from the workspace root (not including the workspace name) - * @returns list of folders that are not studies, under the given path - */ -export const getFolders = async ( - workspace: string, - folderPath: string, -): Promise => { - const res = await client.get( - `/v1/private/explorer/${workspace}/_list_dir?path=${encodeURIComponent(folderPath)}`, - ); - return res.data; -}; - export const getStudyVersions = async (): Promise => { const res = await client.get("/v1/studies/_versions"); return res.data; @@ -463,13 +434,8 @@ export const updateStudyMetadata = async ( return res.data; }; -export const scanFolder = async ( - folderPath: string, - recursive = false, -): Promise => { - await client.post( - `/v1/watcher/_scan?path=${encodeURIComponent(folderPath)}&recursive=${recursive}`, - ); +export const scanFolder = async (folderPath: string): Promise => { + await client.post(`/v1/watcher/_scan?path=${encodeURIComponent(folderPath)}`); }; export const getStudyLayers = async (uuid: string): Promise => { From ca0582b525fa512ed839464840f8e529ff949d65 Mon Sep 17 00:00:00 2001 From: Hatim Dinia <33469289+hdinia@users.noreply.github.com> Date: Fri, 20 Dec 2024 17:24:29 +0100 Subject: [PATCH 3/7] feat(ui-debug): add download in original file format (#2277) --- webapp/public/locales/en/main.json | 1 + webapp/public/locales/fr/main.json | 1 + .../Singlestudy/explore/Debug/Data/Json.tsx | 31 ++---- .../Singlestudy/explore/Debug/Data/Text.tsx | 17 ++- .../explore/Debug/Data/Unsupported.tsx | 30 +----- .../common/Matrix/hooks/useMatrix/index.ts | 5 +- .../Matrix/hooks/useMatrix/useMatrix.test.tsx | 4 +- .../common/buttons/DownloadMatrixButton.tsx | 18 +++- .../common/buttons/UploadFileButton.tsx | 4 +- webapp/src/services/api/studies/raw/index.ts | 100 +++++++++++++++--- webapp/src/services/api/studies/raw/types.ts | 11 +- 11 files changed, 139 insertions(+), 83 deletions(-) diff --git a/webapp/public/locales/en/main.json b/webapp/public/locales/en/main.json index 250a75f2e6..7c6a13307a 100644 --- a/webapp/public/locales/en/main.json +++ b/webapp/public/locales/en/main.json @@ -44,6 +44,7 @@ "global.date": "Date", "global.general": "General", "global.files": "Files", + "global.rawFile": "Raw file", "global.none": "None", "global.upload": "Upload", "global.key": "Key", diff --git a/webapp/public/locales/fr/main.json b/webapp/public/locales/fr/main.json index 98d6ee71b9..395ed91e7e 100644 --- a/webapp/public/locales/fr/main.json +++ b/webapp/public/locales/fr/main.json @@ -44,6 +44,7 @@ "global.date": "Date", "global.general": "Général", "global.files": "Fichiers", + "global.rawFile": "Fichier brut", "global.none": "Aucun", "global.upload": "Charger", "global.key": "Clé", diff --git a/webapp/src/components/App/Singlestudy/explore/Debug/Data/Json.tsx b/webapp/src/components/App/Singlestudy/explore/Debug/Data/Json.tsx index f014626bf5..5df699950b 100644 --- a/webapp/src/components/App/Singlestudy/explore/Debug/Data/Json.tsx +++ b/webapp/src/components/App/Singlestudy/explore/Debug/Data/Json.tsx @@ -21,16 +21,15 @@ import UsePromiseCond from "../../../../../common/utils/UsePromiseCond"; import type { DataCompProps } from "../utils"; import DownloadButton from "../../../../../common/buttons/DownloadButton"; import { downloadFile } from "../../../../../../utils/fileUtils"; -import { useEffect, useState } from "react"; import { Filename, Flex, Menubar } from "./styles"; import UploadFileButton from "../../../../../common/buttons/UploadFileButton"; +import { getRawFile } from "@/services/api/studies/raw"; function Json({ filePath, filename, studyId, canEdit }: DataCompProps) { const [t] = useTranslation(); const { enqueueSnackbar } = useSnackbar(); - const [currentJson, setCurrentJson] = useState(); - const res = usePromiseWithSnackbarError( + const jsonRes = usePromiseWithSnackbarError( () => getStudyData(studyId, filePath, -1), { errorMessage: t("studies.error.retrieveData"), @@ -38,37 +37,27 @@ function Json({ filePath, filename, studyId, canEdit }: DataCompProps) { }, ); - useEffect(() => { - setCurrentJson(res.data); - }, [res.data]); - //////////////////////////////////////////////////////////////// // Event Handlers //////////////////////////////////////////////////////////////// + const handleDownload = async () => { + const file = await getRawFile({ studyId, path: filePath }); + downloadFile(file, file.name); + }; + const handleSave: JSONEditorProps["onSave"] = (json) => { return editStudy(json, studyId, filePath); }; - const handleSaveSuccessful: JSONEditorProps["onSaveSuccessful"] = (json) => { - setCurrentJson(json); - + const handleSaveSuccessful: JSONEditorProps["onSaveSuccessful"] = () => { enqueueSnackbar(t("studies.success.saveData"), { variant: "success", }); }; - const handleDownload = () => { - if (currentJson !== undefined) { - downloadFile( - JSON.stringify(currentJson, null, 2), - filename.endsWith(".json") ? filename : `${filename}.json`, - ); - } - }; - const handleUploadSuccessful = () => { - res.reload(); + jsonRes.reload(); }; //////////////////////////////////////////////////////////////// @@ -77,7 +66,7 @@ function Json({ filePath, filename, studyId, canEdit }: DataCompProps) { return ( ( diff --git a/webapp/src/components/App/Singlestudy/explore/Debug/Data/Text.tsx b/webapp/src/components/App/Singlestudy/explore/Debug/Data/Text.tsx index 1f7a7f4909..1ee8be74c0 100644 --- a/webapp/src/components/App/Singlestudy/explore/Debug/Data/Text.tsx +++ b/webapp/src/components/App/Singlestudy/explore/Debug/Data/Text.tsx @@ -33,6 +33,7 @@ import { Filename, Flex, Menubar } from "./styles"; import UploadFileButton from "../../../../../common/buttons/UploadFileButton"; import EmptyView from "@/components/common/page/SimpleContent"; import GridOffIcon from "@mui/icons-material/GridOff"; +import { getRawFile } from "@/services/api/studies/raw"; SyntaxHighlighter.registerLanguage("xml", xml); SyntaxHighlighter.registerLanguage("plaintext", plaintext); @@ -75,7 +76,7 @@ function Text({ const { t } = useTranslation(); const theme = useTheme(); - const res = usePromiseWithSnackbarError( + const textRes = usePromiseWithSnackbarError( () => getStudyData(studyId, filePath).then((text) => parseContent(text, { filePath, fileType }), @@ -90,17 +91,13 @@ function Text({ // Event Handlers //////////////////////////////////////////////////////////////// - const handleDownload = () => { - if (res.data) { - downloadFile( - res.data, - filename.endsWith(".txt") ? filename : `${filename}.txt`, - ); - } + const handleDownload = async () => { + const file = await getRawFile({ studyId, path: filePath }); + downloadFile(file, file.name); }; const handleUploadSuccessful = () => { - res.reload(); + textRes.reload(); }; //////////////////////////////////////////////////////////////// @@ -109,7 +106,7 @@ function Text({ return ( ( diff --git a/webapp/src/components/App/Singlestudy/explore/Debug/Data/Unsupported.tsx b/webapp/src/components/App/Singlestudy/explore/Debug/Data/Unsupported.tsx index e4bcfccc5c..81307a4edf 100644 --- a/webapp/src/components/App/Singlestudy/explore/Debug/Data/Unsupported.tsx +++ b/webapp/src/components/App/Singlestudy/explore/Debug/Data/Unsupported.tsx @@ -19,33 +19,19 @@ import { Filename, Flex, Menubar } from "./styles"; import type { DataCompProps } from "../utils"; import DownloadButton from "@/components/common/buttons/DownloadButton"; import UploadFileButton from "@/components/common/buttons/UploadFileButton"; -import usePromiseWithSnackbarError from "@/hooks/usePromiseWithSnackbarError"; -import { getStudyData } from "@/services/api/study"; import { downloadFile } from "@/utils/fileUtils"; +import { getRawFile } from "@/services/api/studies/raw"; function Unsupported({ studyId, filePath, filename, canEdit }: DataCompProps) { const { t } = useTranslation(); - const res = usePromiseWithSnackbarError( - () => getStudyData(studyId, filePath), - { - errorMessage: t("studies.error.retrieveData"), - deps: [studyId, filePath], - }, - ); - //////////////////////////////////////////////////////////////// // Event Handlers //////////////////////////////////////////////////////////////// - const handleDownload = () => { - if (res.data) { - downloadFile(res.data, filename); - } - }; - - const handleUploadSuccessful = () => { - res.reload(); + const handleDownload = async () => { + const file = await getRawFile({ studyId, path: filePath }); + downloadFile(file, file.name); }; //////////////////////////////////////////////////////////////// @@ -56,13 +42,7 @@ function Unsupported({ studyId, filePath, filename, canEdit }: DataCompProps) { {filename} - {canEdit && ( - - )} + {canEdit && } diff --git a/webapp/src/components/common/Matrix/hooks/useMatrix/index.ts b/webapp/src/components/common/Matrix/hooks/useMatrix/index.ts index 6558e764e8..e89f99be12 100644 --- a/webapp/src/components/common/Matrix/hooks/useMatrix/index.ts +++ b/webapp/src/components/common/Matrix/hooks/useMatrix/index.ts @@ -39,7 +39,7 @@ import { } from "../../shared/utils"; import useUndo from "use-undo"; import { GridCellKind } from "@glideapps/glide-data-grid"; -import { importFile } from "../../../../../services/api/studies/raw"; +import { uploadFile } from "../../../../../services/api/studies/raw"; import { fetchMatrixFn } from "../../../../App/Singlestudy/explore/Modelization/Areas/Hydro/utils"; import usePrompt from "../../../../../hooks/usePrompt"; import { Aggregate, Column, Operation } from "../../shared/constants"; @@ -251,7 +251,8 @@ export function useMatrix( const handleUpload = async (file: File) => { try { - await importFile({ file, studyId, path: url }); + await uploadFile({ file, studyId, path: url }); + // TODO: update the API to return the uploaded file data and remove this await fetchMatrix(); } catch (e) { enqueueErrorSnackbar(t("matrix.error.import"), e as Error); diff --git a/webapp/src/components/common/Matrix/hooks/useMatrix/useMatrix.test.tsx b/webapp/src/components/common/Matrix/hooks/useMatrix/useMatrix.test.tsx index fc1087a927..e4b61d8587 100644 --- a/webapp/src/components/common/Matrix/hooks/useMatrix/useMatrix.test.tsx +++ b/webapp/src/components/common/Matrix/hooks/useMatrix/useMatrix.test.tsx @@ -166,7 +166,7 @@ describe("useMatrix", () => { describe("File operations", () => { test("should handle file import", async () => { const mockFile = new File([""], "test.csv", { type: "text/csv" }); - vi.mocked(rawStudy.importFile).mockResolvedValue(); + vi.mocked(rawStudy.uploadFile).mockResolvedValue(); const hook = await setupHook(); @@ -174,7 +174,7 @@ describe("useMatrix", () => { await hook.result.current.handleUpload(mockFile); }); - expect(rawStudy.importFile).toHaveBeenCalledWith({ + expect(rawStudy.uploadFile).toHaveBeenCalledWith({ file: mockFile, studyId: DATA.studyId, path: DATA.url, diff --git a/webapp/src/components/common/buttons/DownloadMatrixButton.tsx b/webapp/src/components/common/buttons/DownloadMatrixButton.tsx index 55e0d029c3..b50e3e81da 100644 --- a/webapp/src/components/common/buttons/DownloadMatrixButton.tsx +++ b/webapp/src/components/common/buttons/DownloadMatrixButton.tsx @@ -12,13 +12,15 @@ * This file is part of the Antares project. */ -import { downloadMatrix } from "../../../services/api/studies/raw"; +import { getMatrixFile, getRawFile } from "../../../services/api/studies/raw"; import { downloadFile } from "../../../utils/fileUtils"; import { StudyMetadata } from "../../../common/types"; import { useTranslation } from "react-i18next"; import DownloadButton from "./DownloadButton"; import type { TTableExportFormat } from "@/services/api/studies/raw/types"; +type ExportFormat = TTableExportFormat | "raw"; + export interface DownloadMatrixButtonProps { studyId: StudyMetadata["id"]; path: string; @@ -30,7 +32,7 @@ function DownloadMatrixButton(props: DownloadMatrixButtonProps) { const { t } = useTranslation(); const { studyId, path, disabled, label = t("global.export") } = props; - const options: Array<{ label: string; value: TTableExportFormat }> = [ + const options: Array<{ label: string; value: ExportFormat }> = [ { label: "CSV", value: "csv" }, { label: `CSV (${t("global.semicolon").toLowerCase()})`, @@ -38,20 +40,26 @@ function DownloadMatrixButton(props: DownloadMatrixButtonProps) { }, { label: "TSV", value: "tsv" }, { label: "XLSX", value: "xlsx" }, + { label: `${t("global.rawFile")}`, value: "raw" }, ]; //////////////////////////////////////////////////////////////// // Event Handlers //////////////////////////////////////////////////////////////// - const handleDownload = async (format: TTableExportFormat) => { + const handleDownload = async (format: ExportFormat) => { if (!path) { return; } + if (format === "raw") { + const file = await getRawFile({ studyId, path }); + return downloadFile(file, file.name); + } + const isXlsx = format === "xlsx"; - const res = await downloadMatrix({ + const matrixFile = await getMatrixFile({ studyId, path, format, @@ -62,7 +70,7 @@ function DownloadMatrixButton(props: DownloadMatrixButtonProps) { const extension = format === "csv (semicolon)" ? "csv" : format; return downloadFile( - res, + matrixFile, `matrix_${studyId}_${path.replace("/", "_")}.${extension}`, ); }; diff --git a/webapp/src/components/common/buttons/UploadFileButton.tsx b/webapp/src/components/common/buttons/UploadFileButton.tsx index cc16594be0..872b05b01a 100644 --- a/webapp/src/components/common/buttons/UploadFileButton.tsx +++ b/webapp/src/components/common/buttons/UploadFileButton.tsx @@ -21,7 +21,7 @@ import { toError } from "../../../utils/fnUtils"; import { Accept, useDropzone } from "react-dropzone"; import { StudyMetadata } from "../../../common/types"; import { useSnackbar } from "notistack"; -import { importFile } from "../../../services/api/studies/raw"; +import { uploadFile } from "../../../services/api/studies/raw"; type ValidateResult = boolean | null | undefined; type Validate = (file: File) => ValidateResult | Promise; @@ -89,7 +89,7 @@ function UploadFileButton(props: UploadFileButtonProps) { const filePath = typeof path === "function" ? path(fileToUpload) : path; - await importFile({ + await uploadFile({ studyId, path: filePath, file: fileToUpload, diff --git a/webapp/src/services/api/studies/raw/index.ts b/webapp/src/services/api/studies/raw/index.ts index 85524560d5..663877aafc 100644 --- a/webapp/src/services/api/studies/raw/index.ts +++ b/webapp/src/services/api/studies/raw/index.ts @@ -15,28 +15,55 @@ import client from "../../client"; import type { DeleteFileParams, - DownloadMatrixParams, - ImportFileParams, + GetMatrixFileParams, + GetRawFileParams, + UploadFileParams, } from "./types"; -export async function downloadMatrix(params: DownloadMatrixParams) { +/** + * Gets a matrix file from a study's raw files. + * + * @param params - Parameters for getting the matrix + * @param params.studyId - Unique identifier of the study + * @param params.path - Path to the matrix file + * @param params.format - Optional. Export format for the matrix + * @param params.header - Optional. Whether to include headers + * @param params.index - Optional. Whether to include indices + * @returns Promise containing the matrix data as a Blob + */ +export async function getMatrixFile(params: GetMatrixFileParams) { const { studyId, ...queryParams } = params; - const url = `/v1/studies/${studyId}/raw/download`; - - const { data } = await client.get(url, { - params: queryParams, - responseType: "blob", - }); + const { data } = await client.get( + `/v1/studies/${studyId}/raw/download`, + { + params: queryParams, + responseType: "blob", + }, + ); return data; } -export async function importFile(params: ImportFileParams) { +/** + * Uploads a file to a study's raw storage, creating or updating it based on existence. + * + * !Warning: This endpoint currently uses a non-standard REST structure (/raw) which + * may lead to confusion. It handles both create and update operations through PUT, + * while directory creation is managed through a separate flag. + * + * @param params - Parameters for the file upload + * @param params.studyId - Unique identifier of the study + * @param params.path - Destination path for the file + * @param params.file - File content to upload + * @param params.createMissing - Optional. Whether to create missing parent directories + * @param params.onUploadProgress - Optional. Callback for upload progress updates + * @returns Promise that resolves when the upload is complete + */ +export async function uploadFile(params: UploadFileParams) { const { studyId, file, onUploadProgress, ...queryParams } = params; - const url = `/v1/studies/${studyId}/raw`; const body = { file }; - await client.putForm(url, body, { + await client.putForm(`/v1/studies/${studyId}/raw`, body, { params: { ...queryParams, create_missing: queryParams.createMissing, @@ -45,9 +72,54 @@ export async function importFile(params: ImportFileParams) { }); } +/** + * Deletes a raw file from a study. + * + * @param params - Parameters for deleting the file + * @param params.studyId - Unique identifier of the study + * @param params.path - Path to the file to delete + * @returns Promise that resolves when the deletion is complete + */ export async function deleteFile(params: DeleteFileParams) { const { studyId, path } = params; - const url = `/v1/studies/${studyId}/raw`; + await client.delete(`/v1/studies/${studyId}/raw`, { params: { path } }); +} + +/** + * Gets an original raw file from a study with its metadata. + * + * @param params - Parameters for getting the raw file and name + * @param params.studyId - Unique identifier of the study + * @param params.path - Path to the file within the study + * @returns Promise containing the file data and metadata + */ +export async function getRawFile(params: GetRawFileParams) { + const { studyId, path } = params; - await client.delete(url, { params: { path } }); + const { data, headers } = await client.get( + `/v1/studies/${studyId}/raw/original-file`, + { + params: { + path, + }, + responseType: "blob", + }, + ); + + // Get the original file name from the response Headers + const contentDisposition = headers["content-disposition"]; + let filename = path.split("/").pop() || "file"; // fallback filename + + if (contentDisposition) { + const matches = /filename=([^;]+)/.exec(contentDisposition); + + if (matches?.[1]) { + filename = matches[1].replace(/"/g, "").trim(); + } + } + + return new File([data], filename, { + type: data.type, // Preserve the MIME type from the Blob + lastModified: new Date().getTime(), + }); } diff --git a/webapp/src/services/api/studies/raw/types.ts b/webapp/src/services/api/studies/raw/types.ts index 937fd84119..c74958c376 100644 --- a/webapp/src/services/api/studies/raw/types.ts +++ b/webapp/src/services/api/studies/raw/types.ts @@ -17,9 +17,10 @@ import type { StudyMetadata } from "../../../../common/types"; import { O } from "ts-toolbelt"; import { TableExportFormat } from "./constants"; +// Available export formats for matrix tables export type TTableExportFormat = O.UnionOf; -export interface DownloadMatrixParams { +export interface GetMatrixFileParams { studyId: StudyMetadata["id"]; path: string; format?: TTableExportFormat; @@ -27,10 +28,11 @@ export interface DownloadMatrixParams { index?: boolean; } -export interface ImportFileParams { +export interface UploadFileParams { studyId: StudyMetadata["id"]; path: string; file: File; + // Flag to indicate whether to create file and directories if missing createMissing?: boolean; onUploadProgress?: AxiosRequestConfig["onUploadProgress"]; } @@ -39,3 +41,8 @@ export interface DeleteFileParams { studyId: StudyMetadata["id"]; path: string; } + +export interface GetRawFileParams { + studyId: string; + path: string; +} From c094362e0b1eb907cbbc5e06046d165988e8fb51 Mon Sep 17 00:00:00 2001 From: Samir Kamal <1954121+skamril@users.noreply.github.com> Date: Mon, 30 Dec 2024 14:43:11 +0100 Subject: [PATCH 4/7] chore: add .nvmrc file containing node version number for the app --- webapp/.nvmrc | 1 + 1 file changed, 1 insertion(+) create mode 100644 webapp/.nvmrc diff --git a/webapp/.nvmrc b/webapp/.nvmrc new file mode 100644 index 0000000000..3876fd4986 --- /dev/null +++ b/webapp/.nvmrc @@ -0,0 +1 @@ +18.16.1 From c68cd50394fb1af435e98188b09bc01b9c1aee6d Mon Sep 17 00:00:00 2001 From: Samir Kamal <1954121+skamril@users.noreply.github.com> Date: Mon, 30 Dec 2024 14:44:27 +0100 Subject: [PATCH 5/7] chore(vite): update server URL --- webapp/vite.config.ts | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/webapp/vite.config.ts b/webapp/vite.config.ts index ba4fe1e0f4..71ef9f3906 100644 --- a/webapp/vite.config.ts +++ b/webapp/vite.config.ts @@ -16,7 +16,8 @@ import { defineConfig } from "vite"; import react from "@vitejs/plugin-react-swc"; import path from "path"; -const SERVER_URL = "http://localhost:8080"; +//! Keep '0.0.0.0', because 'localhost' may not working on Mac +const SERVER_URL = "http://0.0.0.0:8080"; // https://vitejs.dev/config/ export default defineConfig(({ mode }) => { From 54a719c6e6c1d48f3b3e564e5279ae144cb22bde Mon Sep 17 00:00:00 2001 From: Sylvain Leclerc Date: Fri, 3 Jan 2025 11:12:22 +0100 Subject: [PATCH 6/7] fix(ui): fix untranslated message (#2275) --- .../src/components/App/Singlestudy/Commands/Edition/index.tsx | 2 +- webapp/vite.config.ts | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/webapp/src/components/App/Singlestudy/Commands/Edition/index.tsx b/webapp/src/components/App/Singlestudy/Commands/Edition/index.tsx index 2ef264dc63..f26d44693a 100644 --- a/webapp/src/components/App/Singlestudy/Commands/Edition/index.tsx +++ b/webapp/src/components/App/Singlestudy/Commands/Edition/index.tsx @@ -522,7 +522,7 @@ function EditionView(props: Props) { ) : ( - + )} diff --git a/webapp/vite.config.ts b/webapp/vite.config.ts index 71ef9f3906..c3b5b33f2b 100644 --- a/webapp/vite.config.ts +++ b/webapp/vite.config.ts @@ -17,7 +17,7 @@ import react from "@vitejs/plugin-react-swc"; import path from "path"; //! Keep '0.0.0.0', because 'localhost' may not working on Mac -const SERVER_URL = "http://0.0.0.0:8080"; +const SERVER_URL = "http://0.0.0.0:8080"; // https://vitejs.dev/config/ export default defineConfig(({ mode }) => { From 1d83b02e55052753731ba37298acbdea251f8fcb Mon Sep 17 00:00:00 2001 From: Sylvain Leclerc Date: Fri, 3 Jan 2025 14:37:23 +0100 Subject: [PATCH 7/7] feat(mypy): enforce explicit overrides (#2270) Signed-off-by: Sylvain Leclerc Co-authored-by: belthlemar --- antarest/core/cache/business/local_chache.py | 7 +++++++ antarest/core/cache/business/redis_cache.py | 6 ++++++ antarest/core/configdata/model.py | 3 +++ antarest/core/exceptions.py | 11 +++++++++++ antarest/core/filetransfer/model.py | 2 ++ antarest/core/interfaces/eventbus.py | 9 +++++++++ antarest/core/logging/utils.py | 4 ++++ antarest/core/requests.py | 8 ++++++++ antarest/core/tasks/model.py | 5 +++++ antarest/core/tasks/service.py | 10 ++++++++++ .../core/utils/fastapi_sqlalchemy/middleware.py | 2 ++ antarest/core/utils/utils.py | 5 +++++ antarest/eventbus/business/local_eventbus.py | 7 +++++++ antarest/eventbus/business/redis_eventbus.py | 6 ++++++ antarest/eventbus/service.py | 9 +++++++++ antarest/front.py | 2 ++ .../adapters/local_launcher/local_launcher.py | 5 +++++ .../adapters/slurm_launcher/slurm_launcher.py | 4 ++++ .../extensions/adequacy_patch/extension.py | 4 ++++ antarest/launcher/model.py | 5 +++++ antarest/login/model.py | 6 ++++++ antarest/matrixstore/matrix_editor.py | 3 +++ antarest/matrixstore/matrix_garbage_collector.py | 3 +++ antarest/matrixstore/model.py | 7 +++++++ antarest/matrixstore/service.py | 9 +++++++++ antarest/study/business/enum_ignore_case.py | 3 +++ .../study/business/scenario_builder_management.py | 2 ++ antarest/study/business/table_mode_management.py | 2 ++ antarest/study/model.py | 11 +++++++++++ antarest/study/service.py | 2 ++ .../study/storage/abstract_storage_service.py | 12 ++++++++++++ antarest/study/storage/auto_archive_service.py | 3 +++ antarest/study/storage/df_download.py | 2 ++ antarest/study/storage/rawstudy/ini_reader.py | 5 +++++ antarest/study/storage/rawstudy/ini_writer.py | 4 ++++ .../rawstudy/model/filesystem/bucket_node.py | 5 +++++ .../model/filesystem/common/area_matrix_list.py | 7 +++++++ .../rawstudy/model/filesystem/common/prepro.py | 3 +++ .../rawstudy/model/filesystem/config/area.py | 3 +++ .../model/filesystem/config/exceptions.py | 4 ++++ .../model/filesystem/config/ini_properties.py | 3 +++ .../rawstudy/model/filesystem/config/model.py | 2 ++ .../rawstudy/model/filesystem/config/renewable.py | 3 +++ .../model/filesystem/config/ruleset_matrices.py | 2 ++ .../rawstudy/model/filesystem/config/thermal.py | 5 +++++ .../rawstudy/model/filesystem/folder_node.py | 10 ++++++++++ .../rawstudy/model/filesystem/ini_file_node.py | 8 ++++++++ .../rawstudy/model/filesystem/json_file_node.py | 4 ++++ .../rawstudy/model/filesystem/lazy_node.py | 6 ++++++ .../model/filesystem/matrix/date_serializer.py | 11 +++++++++++ .../model/filesystem/matrix/head_writer.py | 4 ++++ .../filesystem/matrix/input_series_matrix.py | 4 ++++ .../rawstudy/model/filesystem/matrix/matrix.py | 6 ++++++ .../filesystem/matrix/output_series_matrix.py | 7 +++++++ .../rawstudy/model/filesystem/raw_file_node.py | 8 ++++++++ .../model/filesystem/root/filestudytree.py | 3 +++ .../model/filesystem/root/input/areas/areas.py | 2 ++ .../filesystem/root/input/areas/item/item.py | 2 ++ .../model/filesystem/root/input/areas/list.py | 9 +++++++++ .../input/bindingconstraints/bindingcontraints.py | 2 ++ .../root/input/commons/prepro_series.py | 2 ++ .../root/input/hydro/allocation/allocation.py | 2 ++ .../root/input/hydro/common/capacity/capacity.py | 2 ++ .../filesystem/root/input/hydro/common/common.py | 2 ++ .../model/filesystem/root/input/hydro/hydro.py | 2 ++ .../root/input/hydro/prepro/area/area.py | 2 ++ .../filesystem/root/input/hydro/prepro/prepro.py | 2 ++ .../root/input/hydro/series/area/area.py | 3 +++ .../filesystem/root/input/hydro/series/series.py | 2 ++ .../rawstudy/model/filesystem/root/input/input.py | 2 ++ .../model/filesystem/root/input/link/area/area.py | 2 ++ .../root/input/link/area/capacities/capacities.py | 2 ++ .../model/filesystem/root/input/link/link.py | 2 ++ .../filesystem/root/input/miscgen/miscgen.py | 2 ++ .../filesystem/root/input/renewables/clusters.py | 3 +++ .../filesystem/root/input/renewables/renewable.py | 2 ++ .../filesystem/root/input/renewables/series.py | 4 ++++ .../filesystem/root/input/reserves/reserves.py | 2 ++ .../root/input/st_storage/clusters/area/area.py | 2 ++ .../root/input/st_storage/clusters/clusters.py | 2 ++ .../root/input/st_storage/series/area/area.py | 2 ++ .../series/area/st_storage/st_storage.py | 2 ++ .../root/input/st_storage/series/series.py | 2 ++ .../root/input/st_storage/st_storage.py | 3 +++ .../root/input/thermal/cluster/area/area.py | 2 ++ .../root/input/thermal/cluster/cluster.py | 2 ++ .../root/input/thermal/prepro/area/area.py | 2 ++ .../input/thermal/prepro/area/thermal/thermal.py | 2 ++ .../root/input/thermal/prepro/prepro.py | 2 ++ .../root/input/thermal/series/area/area.py | 2 ++ .../input/thermal/series/area/thermal/thermal.py | 2 ++ .../root/input/thermal/series/series.py | 2 ++ .../filesystem/root/input/thermal/thermal.py | 2 ++ .../model/filesystem/root/layers/layers.py | 2 ++ .../model/filesystem/root/output/output.py | 2 ++ .../root/output/simulation/about/about.py | 2 ++ .../root/output/simulation/mode/common/area.py | 2 ++ .../root/output/simulation/mode/common/areas.py | 2 ++ .../simulation/mode/common/binding_const.py | 2 ++ .../root/output/simulation/mode/common/link.py | 2 ++ .../root/output/simulation/mode/common/links.py | 4 ++++ .../root/output/simulation/mode/common/set.py | 2 ++ .../root/output/simulation/mode/common/utils.py | 2 ++ .../root/output/simulation/mode/economy.py | 2 ++ .../root/output/simulation/mode/mcall/grid.py | 9 +++++++++ .../root/output/simulation/mode/mcind/mcind.py | 2 ++ .../root/output/simulation/simulation.py | 2 ++ .../simulation/ts_generator/ts_generator.py | 5 +++++ .../output/simulation/ts_numbers/ts_numbers.py | 2 ++ .../simulation/ts_numbers/ts_numbers_data.py | 7 +++++++ .../output/simulation/xpansion/sensitivity.py | 2 ++ .../root/output/simulation/xpansion/xpansion.py | 2 ++ .../root/settings/resources/resources.py | 2 ++ .../filesystem/root/settings/scenariobuilder.py | 2 ++ .../model/filesystem/root/settings/settings.py | 2 ++ .../root/settings/simulations/simulations.py | 2 ++ .../filesystem/root/user/expansion/sensitivity.py | 3 +++ .../study/storage/rawstudy/raw_study_service.py | 12 ++++++++++++ antarest/study/storage/rawstudy/watcher.py | 3 +++ .../variantstudy/business/command_extractor.py | 13 +++++++++++++ .../variantstudy/model/command/create_area.py | 8 ++++++++ .../model/command/create_binding_constraint.py | 9 +++++++++ .../variantstudy/model/command/create_cluster.py | 8 ++++++++ .../variantstudy/model/command/create_district.py | 8 ++++++++ .../variantstudy/model/command/create_link.py | 13 +++++++++++++ .../model/command/create_renewables_cluster.py | 8 ++++++++ .../model/command/create_st_storage.py | 8 ++++++++ .../model/command/create_user_resource.py | 9 +++++++++ .../generate_thermal_cluster_timeseries.py | 8 ++++++++ .../variantstudy/model/command/remove_area.py | 9 +++++++++ .../model/command/remove_binding_constraint.py | 9 +++++++++ .../variantstudy/model/command/remove_cluster.py | 8 ++++++++ .../variantstudy/model/command/remove_district.py | 9 +++++++++ .../variantstudy/model/command/remove_link.py | 8 ++++++++ .../model/command/remove_renewables_cluster.py | 11 +++++++++-- .../model/command/remove_st_storage.py | 8 ++++++++ .../model/command/remove_user_resource.py | 9 +++++++++ .../variantstudy/model/command/replace_matrix.py | 8 ++++++++ .../model/command/update_binding_constraint.py | 8 ++++++++ .../variantstudy/model/command/update_comments.py | 9 +++++++++ .../variantstudy/model/command/update_config.py | 8 ++++++++ .../variantstudy/model/command/update_district.py | 9 +++++++++ .../variantstudy/model/command/update_link.py | 8 ++++++++ .../variantstudy/model/command/update_playlist.py | 9 +++++++++ .../variantstudy/model/command/update_raw_file.py | 10 ++++++++++ .../model/command/update_scenario_builder.py | 8 ++++++++ .../study/storage/variantstudy/model/dbmodel.py | 4 ++++ antarest/study/storage/variantstudy/repository.py | 2 ++ .../storage/variantstudy/variant_study_service.py | 15 +++++++++++++++ antarest/tools/lib.py | 3 +++ antarest/worker/archive_worker.py | 3 +++ antarest/worker/worker.py | 3 +++ pyproject.toml | 1 + 153 files changed, 732 insertions(+), 2 deletions(-) diff --git a/antarest/core/cache/business/local_chache.py b/antarest/core/cache/business/local_chache.py index e903ff9080..493322b9c6 100644 --- a/antarest/core/cache/business/local_chache.py +++ b/antarest/core/cache/business/local_chache.py @@ -15,6 +15,8 @@ import time from typing import Dict, List, Optional +from typing_extensions import override + from antarest.core.config import CacheConfig from antarest.core.interfaces.cache import ICache from antarest.core.model import JSON @@ -40,6 +42,7 @@ def __init__(self, config: CacheConfig = CacheConfig()): daemon=True, ) + @override def start(self) -> None: self.checker_thread.start() @@ -55,6 +58,7 @@ def checker(self) -> None: for id in to_delete: del self.cache[id] + @override def put(self, id: str, data: JSON, duration: int = 3600) -> None: # Duration in second with self.lock: logger.info(f"Adding cache key {id}") @@ -64,6 +68,7 @@ def put(self, id: str, data: JSON, duration: int = 3600) -> None: # Duration in duration=duration, ) + @override def get(self, id: str, refresh_duration: Optional[int] = None) -> Optional[JSON]: res = None with self.lock: @@ -76,12 +81,14 @@ def get(self, id: str, refresh_duration: Optional[int] = None) -> Optional[JSON] res = self.cache[id].data return res + @override def invalidate(self, id: str) -> None: with self.lock: logger.info(f"Removing cache key {id}") if id in self.cache: del self.cache[id] + @override def invalidate_all(self, ids: List[str]) -> None: with self.lock: logger.info(f"Removing cache keys {ids}") diff --git a/antarest/core/cache/business/redis_cache.py b/antarest/core/cache/business/redis_cache.py index 11eb3fcffd..ee7e30a6d1 100644 --- a/antarest/core/cache/business/redis_cache.py +++ b/antarest/core/cache/business/redis_cache.py @@ -14,6 +14,7 @@ from typing import List, Optional from redis.client import Redis +from typing_extensions import override from antarest.core.interfaces.cache import ICache from antarest.core.model import JSON @@ -31,10 +32,12 @@ class RedisCache(ICache): def __init__(self, redis_client: Redis): # type: ignore self.redis = redis_client + @override def start(self) -> None: # Assuming the Redis service is already running; no need to start it here. pass + @override def put(self, id: str, data: JSON, duration: int = 3600) -> None: redis_element = RedisCacheElement(duration=duration, data=data) redis_key = f"cache:{id}" @@ -42,6 +45,7 @@ def put(self, id: str, data: JSON, duration: int = 3600) -> None: self.redis.set(redis_key, redis_element.model_dump_json()) self.redis.expire(redis_key, duration) + @override def get(self, id: str, refresh_timeout: Optional[int] = None) -> Optional[JSON]: redis_key = f"cache:{id}" result = self.redis.get(redis_key) @@ -58,10 +62,12 @@ def get(self, id: str, refresh_timeout: Optional[int] = None) -> Optional[JSON]: logger.info(f"Cache key {id} not found") return None + @override def invalidate(self, id: str) -> None: logger.info(f"Removing cache key {id}") self.redis.delete(f"cache:{id}") + @override def invalidate_all(self, ids: List[str]) -> None: logger.info(f"Removing cache keys {ids}") self.redis.delete(*[f"cache:{id}" for id in ids]) diff --git a/antarest/core/configdata/model.py b/antarest/core/configdata/model.py index 3a4512e44c..faa6589709 100644 --- a/antarest/core/configdata/model.py +++ b/antarest/core/configdata/model.py @@ -14,6 +14,7 @@ from typing import Any, Optional from sqlalchemy import Column, Integer, String # type: ignore +from typing_extensions import override from antarest.core.persistence import Base from antarest.core.serialization import AntaresBaseModel @@ -30,11 +31,13 @@ class ConfigData(Base): # type: ignore key = Column(String(), primary_key=True) value = Column(String(), nullable=True) + @override def __eq__(self, other: Any) -> bool: if not isinstance(other, ConfigData): return False return bool(other.key == self.key and other.value == self.value and other.owner == self.owner) + @override def __repr__(self) -> str: return f"key={self.key}, value={self.value}, owner={self.owner}" diff --git a/antarest/core/exceptions.py b/antarest/core/exceptions.py index d6c392b9cf..722a385b26 100644 --- a/antarest/core/exceptions.py +++ b/antarest/core/exceptions.py @@ -15,6 +15,7 @@ from http import HTTPStatus from fastapi.exceptions import HTTPException +from typing_extensions import override class ShouldNotHappenException(Exception): @@ -81,6 +82,7 @@ def __init__(self, path: str, *area_ids: str): detail = f"{self.object_name.title()} {detail}" super().__init__(HTTPStatus.NOT_FOUND, detail) + @override def __str__(self) -> str: """Return a string representation of the exception.""" return self.detail @@ -127,6 +129,7 @@ def __init__(self, path: str, section_id: str): detail = f"{object_name.title()} '{section_id}' not found in '{path}'" super().__init__(HTTPStatus.NOT_FOUND, detail) + @override def __str__(self) -> str: """Return a string representation of the exception.""" return self.detail @@ -172,6 +175,7 @@ def __init__(self, path: str): detail = f"{self.object_name.title()} {detail}" super().__init__(HTTPStatus.NOT_FOUND, detail) + @override def __str__(self) -> str: return self.detail @@ -227,6 +231,7 @@ def __init__(self, area_id: str, *duplicates: str): detail = f"{self.object_name.title()} {detail}" super().__init__(HTTPStatus.CONFLICT, detail) + @override def __str__(self) -> str: """Return a string representation of the exception.""" return self.detail @@ -397,6 +402,7 @@ def __init__(self, object_id: str, binding_ids: t.Sequence[str], *, object_type: ) super().__init__(HTTPStatus.FORBIDDEN, message) + @override def __str__(self) -> str: """Return a string representation of the exception.""" return self.detail @@ -429,6 +435,7 @@ def __init__(self, output_id: str) -> None: message = f"Output '{output_id}' not found" super().__init__(HTTPStatus.NOT_FOUND, message) + @override def __str__(self) -> str: """Return a string representation of the exception.""" return self.detail @@ -463,6 +470,7 @@ def __init__(self, output_id: str, mc_root: str) -> None: message = f"The output '{output_id}' sub-folder '{mc_root}' does not exist" super().__init__(HTTPStatus.NOT_FOUND, message) + @override def __str__(self) -> str: """Return a string representation of the exception.""" return self.detail @@ -552,6 +560,7 @@ def __init__(self, binding_constraint_id: str, *ids: str) -> None: }[min(count, 2)] super().__init__(HTTPStatus.NOT_FOUND, message) + @override def __str__(self) -> str: """Return a string representation of the exception.""" return self.detail @@ -572,6 +581,7 @@ def __init__(self, binding_constraint_id: str, *ids: str) -> None: }[min(count, 2)] super().__init__(HTTPStatus.CONFLICT, message) + @override def __str__(self) -> str: """Return a string representation of the exception.""" return self.detail @@ -589,6 +599,7 @@ def __init__(self, binding_constraint_id: str, term_json: str) -> None: ) super().__init__(HTTPStatus.UNPROCESSABLE_ENTITY, message) + @override def __str__(self) -> str: """Return a string representation of the exception.""" return self.detail diff --git a/antarest/core/filetransfer/model.py b/antarest/core/filetransfer/model.py index 6ce194ed19..0a5958d57a 100644 --- a/antarest/core/filetransfer/model.py +++ b/antarest/core/filetransfer/model.py @@ -16,6 +16,7 @@ from typing import Optional from sqlalchemy import Boolean, Column, DateTime, Integer, String # type: ignore +from typing_extensions import override from antarest.core.persistence import Base from antarest.core.serialization import AntaresBaseModel @@ -81,6 +82,7 @@ def to_dto(self) -> FileDownloadDTO: error_message=self.error_message or "", ) + @override def __repr__(self) -> str: return ( f"(id={self.id}," diff --git a/antarest/core/interfaces/eventbus.py b/antarest/core/interfaces/eventbus.py index a60aaf481a..d9075dc49e 100644 --- a/antarest/core/interfaces/eventbus.py +++ b/antarest/core/interfaces/eventbus.py @@ -14,6 +14,8 @@ from enum import StrEnum from typing import Any, Awaitable, Callable, List, Optional +from typing_extensions import override + from antarest.core.model import PermissionInfo from antarest.core.serialization import AntaresBaseModel @@ -140,21 +142,26 @@ class DummyEventBusService(IEventBus): def __init__(self) -> None: self.events: List[Event] = [] + @override def queue(self, event: Event, queue: str) -> None: # Noop pass + @override def add_queue_consumer(self, listener: Callable[[Event], Awaitable[None]], queue: str) -> str: return "" + @override def remove_queue_consumer(self, listener_id: str) -> None: # Noop pass + @override def push(self, event: Event) -> None: # Noop self.events.append(event) + @override def add_listener( self, listener: Callable[[Event], Awaitable[None]], @@ -162,10 +169,12 @@ def add_listener( ) -> str: return "" + @override def remove_listener(self, listener_id: str) -> None: # Noop pass + @override def start(self, threaded: bool = True) -> None: # Noop pass diff --git a/antarest/core/logging/utils.py b/antarest/core/logging/utils.py index 991f21b846..2fa8c4a040 100644 --- a/antarest/core/logging/utils.py +++ b/antarest/core/logging/utils.py @@ -20,6 +20,7 @@ from starlette.middleware.base import BaseHTTPMiddleware, RequestResponseEndpoint from starlette.requests import Request from starlette.responses import Response +from typing_extensions import override from antarest.core.config import Config @@ -39,6 +40,7 @@ class CustomDefaultFormatter(logging.Formatter): fields to the log record with a value of `None`. """ + @override def format(self, record: logging.LogRecord) -> str: """ Formats the specified log record using the custom formatter, @@ -169,6 +171,7 @@ def configure_logger(config: Config, handler_cls: str = "logging.FileHandler") - class LoggingMiddleware(BaseHTTPMiddleware): + @override async def dispatch(self, request: Request, call_next: RequestResponseEndpoint) -> Response: with RequestContext(request): response = await call_next(request) @@ -176,6 +179,7 @@ async def dispatch(self, request: Request, call_next: RequestResponseEndpoint) - class ContextFilter(logging.Filter): + @override def filter(self, record: logging.LogRecord) -> bool: request: Optional[Request] = _request.get() request_id: Optional[str] = _request_id.get() diff --git a/antarest/core/requests.py b/antarest/core/requests.py index d276f79dca..055bbd8cdd 100644 --- a/antarest/core/requests.py +++ b/antarest/core/requests.py @@ -18,6 +18,7 @@ from fastapi import HTTPException from markupsafe import escape from ratelimit import Rule # type: ignore +from typing_extensions import override from antarest.core.jwt import JWTUser @@ -38,24 +39,30 @@ def __init__(self, data=None, **kwargs) -> None: # type: ignore data = {} self.update(data, **kwargs) + @override def __setitem__(self, key: str, value: t.Any) -> None: self._store[key.lower()] = (key, value) + @override def __getitem__(self, key: str) -> t.Any: return self._store[key.lower()][1] + @override def __delitem__(self, key: str) -> None: del self._store[key.lower()] + @override def __iter__(self) -> t.Any: return (casedkey for casedkey, mappedvalue in self._store.values()) + @override def __len__(self) -> int: return len(self._store) def lower_items(self) -> Generator[Tuple[Any, Any], Any, None]: return ((lowerkey, keyval[1]) for (lowerkey, keyval) in self._store.items()) + @override def __eq__(self, other: t.Any) -> bool: if isinstance(other, t.Mapping): other = CaseInsensitiveDict(other) @@ -66,6 +73,7 @@ def __eq__(self, other: t.Any) -> bool: def copy(self) -> "CaseInsensitiveDict": return CaseInsensitiveDict(self._store.values()) + @override def __repr__(self) -> str: return str(dict(self.items())) diff --git a/antarest/core/tasks/model.py b/antarest/core/tasks/model.py index 0b414b12c6..a533b73baf 100644 --- a/antarest/core/tasks/model.py +++ b/antarest/core/tasks/model.py @@ -18,6 +18,7 @@ from sqlalchemy import Boolean, Column, DateTime, ForeignKey, Integer, Sequence, String # type: ignore from sqlalchemy.engine.base import Engine # type: ignore from sqlalchemy.orm import relationship, sessionmaker # type: ignore +from typing_extensions import override from antarest.core.persistence import Base from antarest.core.serialization import AntaresBaseModel @@ -122,11 +123,13 @@ class TaskJobLog(Base): # type: ignore # If the TaskJob is deleted, all attached logs must also be deleted in cascade. job: "TaskJob" = relationship("TaskJob", back_populates="logs", uselist=False) + @override def __eq__(self, other: t.Any) -> bool: if not isinstance(other, TaskJobLog): return False return bool(other.id == self.id and other.message == self.message and other.task_id == self.task_id) + @override def __repr__(self) -> str: return f"id={self.id}, message={self.message}, task_id={self.task_id}" @@ -198,6 +201,7 @@ def to_dto(self, with_logs: bool = False) -> TaskDTO: progress=self.progress, ) + @override def __eq__(self, other: t.Any) -> bool: if not isinstance(other, TaskJob): return False @@ -213,6 +217,7 @@ def __eq__(self, other: t.Any) -> bool: and other.logs == self.logs ) + @override def __repr__(self) -> str: return ( f"id={self.id}," diff --git a/antarest/core/tasks/service.py b/antarest/core/tasks/service.py index a97a2fedf5..9b7efb8344 100644 --- a/antarest/core/tasks/service.py +++ b/antarest/core/tasks/service.py @@ -20,6 +20,7 @@ from fastapi import HTTPException from sqlalchemy.orm import Session # type: ignore +from typing_extensions import override from antarest.core.config import Config from antarest.core.interfaces.eventbus import Event, EventChannelDirectory, EventType, IEventBus @@ -108,9 +109,11 @@ def await_task(self, task_id: str, timeout_sec: int = DEFAULT_AWAIT_MAX_TIMEOUT) class NoopNotifier(ITaskNotifier): """This class is used in tasks when no notification is required.""" + @override def notify_message(self, message: str) -> None: return + @override def notify_progress(self, progress: int) -> None: return @@ -129,12 +132,14 @@ def __init__(self, task_id: str, session: Session, event_bus: IEventBus) -> None self.task_id = task_id self.event_bus = event_bus + @override def notify_message(self, message: str) -> None: task = self.session.query(TaskJob).get(self.task_id) if task: task.logs.append(TaskJobLog(message=message, task_id=self.task_id)) self.session.commit() + @override def notify_progress(self, progress: int) -> None: self.session.query(TaskJob).filter(TaskJob.id == self.task_id).update({TaskJob.progress: progress}) self.session.commit() @@ -215,6 +220,7 @@ def _send_worker_task(logger_: ITaskNotifier) -> TaskResult: def check_remote_worker_for_queue(self, task_queue: str) -> bool: return any(task_queue in rw.queues for rw in self.remote_workers) + @override def add_worker_task( self, task_type: TaskType, @@ -237,6 +243,7 @@ def add_worker_task( ) return str(task.id) + @override def add_task( self, action: Task, @@ -328,6 +335,7 @@ def _cancel_task(self, task_id: str, dispatch: bool = False) -> None: ) ) + @override def status_task( self, task_id: str, @@ -344,6 +352,7 @@ def status_task( detail=f"Failed to retrieve task {task_id} in db", ) + @override def list_tasks(self, task_filter: TaskListFilter, request_params: RequestParameters) -> t.List[TaskDTO]: return [task.to_dto() for task in self.list_db_tasks(task_filter, request_params)] @@ -353,6 +362,7 @@ def list_db_tasks(self, task_filter: TaskListFilter, request_params: RequestPara user = None if request_params.user.is_site_admin() else request_params.user.impersonator return self.repo.list(task_filter, user) + @override def await_task(self, task_id: str, timeout_sec: int = DEFAULT_AWAIT_MAX_TIMEOUT) -> None: if task_id in self.tasks: try: diff --git a/antarest/core/utils/fastapi_sqlalchemy/middleware.py b/antarest/core/utils/fastapi_sqlalchemy/middleware.py index 73a073e033..db18bc3efc 100644 --- a/antarest/core/utils/fastapi_sqlalchemy/middleware.py +++ b/antarest/core/utils/fastapi_sqlalchemy/middleware.py @@ -10,6 +10,7 @@ from starlette.requests import Request from starlette.responses import Response from starlette.types import ASGIApp +from typing_extensions import override from antarest.core.utils.fastapi_sqlalchemy.exceptions import MissingSessionError, SessionNotInitialisedError @@ -60,6 +61,7 @@ def __init__( _Session = sessionmaker(bind=engine, **session_args) + @override async def dispatch(self, request: Request, call_next: RequestResponseEndpoint) -> Response: with db(commit_on_exit=self.commit_on_exit): response = await call_next(request) diff --git a/antarest/core/utils/utils.py b/antarest/core/utils/utils.py index 2940db582a..0abe2a6fb4 100644 --- a/antarest/core/utils/utils.py +++ b/antarest/core/utils/utils.py @@ -20,6 +20,7 @@ from pathlib import Path from fastapi import HTTPException +from typing_extensions import override from antarest.core.exceptions import ShouldNotHappenException @@ -33,18 +34,22 @@ class DTO: Implement basic method for DTO objects """ + @override def __hash__(self) -> int: return hash(tuple(sorted(self.__dict__.items()))) + @override def __eq__(self, other: t.Any) -> bool: return isinstance(other, type(self)) and self.__dict__ == other.__dict__ + @override def __str__(self) -> str: return "{}({})".format( type(self).__name__, ", ".join(["{}={}".format(k, str(self.__dict__[k])) for k in sorted(self.__dict__)]), ) + @override def __repr__(self) -> str: return self.__str__() diff --git a/antarest/eventbus/business/local_eventbus.py b/antarest/eventbus/business/local_eventbus.py index bfe0e6992d..774278d5e4 100644 --- a/antarest/eventbus/business/local_eventbus.py +++ b/antarest/eventbus/business/local_eventbus.py @@ -13,6 +13,8 @@ import logging from typing import Dict, List, Optional +from typing_extensions import override + from antarest.core.interfaces.eventbus import Event from antarest.eventbus.business.interfaces import IEventBusBackend @@ -24,20 +26,25 @@ def __init__(self) -> None: self.events: List[Event] = [] self.queues: Dict[str, List[Event]] = {} + @override def push_event(self, event: Event) -> None: self.events.append(event) + @override def get_events(self) -> List[Event]: return self.events + @override def clear_events(self) -> None: self.events.clear() + @override def queue_event(self, event: Event, queue: str) -> None: if queue not in self.queues: self.queues[queue] = [] self.queues[queue].append(event) + @override def pull_queue(self, queue: str) -> Optional[Event]: if queue in self.queues and len(self.queues[queue]) > 0: return self.queues[queue].pop(0) diff --git a/antarest/eventbus/business/redis_eventbus.py b/antarest/eventbus/business/redis_eventbus.py index 8bbf6cbc38..db89c7bc68 100644 --- a/antarest/eventbus/business/redis_eventbus.py +++ b/antarest/eventbus/business/redis_eventbus.py @@ -14,6 +14,7 @@ from typing import List, Optional, cast from redis.client import Redis +from typing_extensions import override from antarest.core.interfaces.eventbus import Event from antarest.eventbus.business.interfaces import IEventBusBackend @@ -29,18 +30,22 @@ def __init__(self, redis_client: Redis) -> None: # type: ignore self.pubsub = self.redis.pubsub() self.pubsub.subscribe(REDIS_STORE_KEY) + @override def push_event(self, event: Event) -> None: self.redis.publish(REDIS_STORE_KEY, event.model_dump_json()) + @override def queue_event(self, event: Event, queue: str) -> None: self.redis.rpush(queue, event.model_dump_json()) + @override def pull_queue(self, queue: str) -> Optional[Event]: event = self.redis.lpop(queue) if event: return cast(Optional[Event], Event.parse_raw(event)) return None + @override def get_events(self) -> List[Event]: messages = [] try: @@ -60,6 +65,7 @@ def get_events(self) -> List[Event]: return events + @override def clear_events(self) -> None: # Nothing to do pass diff --git a/antarest/eventbus/service.py b/antarest/eventbus/service.py index eadf75e8c8..63b519e954 100644 --- a/antarest/eventbus/service.py +++ b/antarest/eventbus/service.py @@ -18,6 +18,8 @@ import uuid from typing import Awaitable, Callable, Dict, List, Optional +from typing_extensions import override + from antarest.core.interfaces.eventbus import Event, EventType, IEventBus from antarest.eventbus.business.interfaces import IEventBusBackend @@ -39,12 +41,15 @@ def __init__(self, backend: IEventBusBackend, autostart: bool = True) -> None: if autostart: self.start() + @override def push(self, event: Event) -> None: self.backend.push_event(event) + @override def queue(self, event: Event, queue: str) -> None: self.backend.queue_event(event, queue) + @override def add_queue_consumer(self, listener: Callable[[Event], Awaitable[None]], queue: str) -> str: with self.lock: listener_id = str(uuid.uuid4()) @@ -53,12 +58,14 @@ def add_queue_consumer(self, listener: Callable[[Event], Awaitable[None]], queue self.consumers[queue][listener_id] = listener return listener_id + @override def remove_queue_consumer(self, listener_id: str) -> None: with self.lock: for queue in self.consumers: if listener_id in self.consumers[queue]: del self.consumers[queue][listener_id] + @override def add_listener( self, listener: Callable[[Event], Awaitable[None]], @@ -71,6 +78,7 @@ def add_listener( self.listeners[listener_type][listener_id] = listener return listener_id + @override def remove_listener(self, listener_id: str) -> None: with self.lock: for listener_type in self.listeners: @@ -130,6 +138,7 @@ def _async_loop(self, new_loop: bool = True) -> None: loop = asyncio.new_event_loop() if new_loop else asyncio.get_event_loop() loop.run_until_complete(self._run_loop()) + @override def start(self, threaded: bool = True) -> None: if threaded: t = threading.Thread( diff --git a/antarest/front.py b/antarest/front.py index 16cd92544f..33314eb3f4 100644 --- a/antarest/front.py +++ b/antarest/front.py @@ -30,6 +30,7 @@ from starlette.responses import FileResponse from starlette.staticfiles import StaticFiles from starlette.types import ASGIApp +from typing_extensions import override from antarest.core.serialization import AntaresBaseModel from antarest.core.utils.string import to_camel_case @@ -76,6 +77,7 @@ def _path_matches_protected_paths(self, path: str) -> bool: return True return False + @override async def dispatch(self, request: Request, call_next: RequestResponseEndpoint) -> Any: """ Intercepts the incoming request and rewrites the URL path if necessary. diff --git a/antarest/launcher/adapters/local_launcher/local_launcher.py b/antarest/launcher/adapters/local_launcher/local_launcher.py index a3ad8fc703..30560977a7 100644 --- a/antarest/launcher/adapters/local_launcher/local_launcher.py +++ b/antarest/launcher/adapters/local_launcher/local_launcher.py @@ -23,6 +23,7 @@ from uuid import UUID from antares.study.version import SolverVersion +from typing_extensions import override from antarest.core.config import Config from antarest.core.interfaces.cache import ICache @@ -73,6 +74,7 @@ def _select_best_binary(self, version: str) -> Path: ) return antares_solver_path + @override def run_study( self, study_uuid: str, @@ -195,6 +197,7 @@ def stop_reading_output() -> bool: end = True shutil.rmtree(tmp_path) + @override def create_update_log(self, job_id: str) -> Callable[[str], None]: base_func = super().create_update_log(job_id) self.logs[job_id] = "" @@ -205,6 +208,7 @@ def append_to_log(log_line: str) -> None: return append_to_log + @override def get_log(self, job_id: str, log_type: LogType) -> Optional[str]: if job_id in self.job_id_to_study_id and job_id in self.logs: return self.logs[job_id] @@ -212,6 +216,7 @@ def get_log(self, job_id: str, log_type: LogType) -> Optional[str]: return self._get_job_final_output_path(job_id).read_text() return None + @override def kill_job(self, job_id: str) -> None: if job_id in self.job_id_to_study_id: return self.job_id_to_study_id[job_id][2].send_signal(signal.SIGTERM) diff --git a/antarest/launcher/adapters/slurm_launcher/slurm_launcher.py b/antarest/launcher/adapters/slurm_launcher/slurm_launcher.py index d4f5c30da8..0506ad5694 100644 --- a/antarest/launcher/adapters/slurm_launcher/slurm_launcher.py +++ b/antarest/launcher/adapters/slurm_launcher/slurm_launcher.py @@ -28,6 +28,7 @@ from antareslauncher.main_option_parser import MainOptionParser, ParserParameters from antareslauncher.study_dto import StudyDTO from filelock import FileLock +from typing_extensions import override from antarest.core.config import Config, NbCoresConfig, SlurmConfig, TimeLimitConfig from antarest.core.interfaces.cache import ICache @@ -589,6 +590,7 @@ def _apply_params(self, launcher_params: LauncherParametersDTO) -> argparse.Name return self.launcher_args + @override def run_study( self, study_uuid: str, @@ -604,6 +606,7 @@ def run_study( ) thread.start() + @override def get_log(self, job_id: str, log_type: LogType) -> t.Optional[str]: log_path: t.Optional[Path] = None for study in self.data_repo_tinydb.get_list_of_studies(): @@ -621,6 +624,7 @@ async def _listen_to_kill_job(event: Event) -> None: return _listen_to_kill_job + @override def kill_job(self, job_id: str, dispatch: bool = True) -> None: launcher_args = LauncherArgs(self.launcher_args) for study in self.data_repo_tinydb.get_list_of_studies(): diff --git a/antarest/launcher/extensions/adequacy_patch/extension.py b/antarest/launcher/extensions/adequacy_patch/extension.py index bb4d19f323..10f2f28f52 100644 --- a/antarest/launcher/extensions/adequacy_patch/extension.py +++ b/antarest/launcher/extensions/adequacy_patch/extension.py @@ -16,6 +16,7 @@ from typing import Any, Dict, List, cast import yaml +from typing_extensions import override from antarest.core.config import Config from antarest.core.model import JSON @@ -95,9 +96,11 @@ class AdequacyPatchExtension(ILauncherExtension): def __init__(self, study_service: StudyService, config: Config): self.study_service = study_service + @override def get_name(self) -> str: return AdequacyPatchExtension.EXTENSION_NAME + @override def after_export_flat_hook( self, job_id: str, @@ -129,6 +132,7 @@ def after_export_flat_hook( post_processing_file = Path(__file__).parent / "resources" / "post-processing.R" shutil.copy(post_processing_file, study_export_path / "post-processing.R") + @override def before_import_hook( self, job_id: str, diff --git a/antarest/launcher/model.py b/antarest/launcher/model.py index dbd90e3ef4..934b2b5753 100644 --- a/antarest/launcher/model.py +++ b/antarest/launcher/model.py @@ -17,6 +17,7 @@ from pydantic import Field from sqlalchemy import Column, DateTime, Enum, ForeignKey, Integer, Sequence, String # type: ignore from sqlalchemy.orm import relationship # type: ignore +from typing_extensions import override from antarest.core.persistence import Base from antarest.core.serialization import AntaresBaseModel, from_json @@ -157,9 +158,11 @@ class JobLog(Base): # type: ignore # that the comparison is based on the database identity of the objects. # So, implementing `__eq__` and `__ne__` is not necessary. + @override def __str__(self) -> str: return f"Job log #{self.id} {self.log_type}: '{self.message}'" + @override def __repr__(self) -> str: return ( f" JobResultDTO: # that the comparison is based on the database identity of the objects. # So, implementing `__eq__` and `__ne__` is not necessary. + @override def __str__(self) -> str: return f"Job result #{self.id} (study '{self.study_id}'): {self.job_status}" + @override def __repr__(self) -> str: return ( f" str: def check(self, pwd: str) -> bool: return bcrypt.checkpw(pwd.encode(), self._pwd) + @override def __str__(self) -> str: return "*****" + @override def __repr__(self) -> str: return self.__str__() @@ -245,6 +248,7 @@ class Bot(Identity): owner = Column(Integer, ForeignKey("identities.id", name="bots_owner_fkey")) is_author = Column(Boolean(), default=True) + @override def get_impersonator(self) -> int: return int(self.id if self.is_author else self.owner) @@ -253,6 +257,7 @@ def get_impersonator(self) -> int: "inherit_condition": id == Identity.id, } + @override def to_dto(self) -> BotDTO: return BotDTO( id=self.id, @@ -286,6 +291,7 @@ def to_dto(self) -> GroupDTO: # Implementing a `__eq__` method is superfluous, since the default implementation # is to compare the identity of the objects using the primary key. + @override def __repr__(self) -> str: return f"Group(id={self.id}, name={self.name})" diff --git a/antarest/matrixstore/matrix_editor.py b/antarest/matrixstore/matrix_editor.py index 89bb866336..33bf780cab 100644 --- a/antarest/matrixstore/matrix_editor.py +++ b/antarest/matrixstore/matrix_editor.py @@ -15,6 +15,7 @@ from typing import Any, Dict, List, Optional, Tuple from pydantic import Field, field_validator, model_validator +from typing_extensions import override from antarest.core.serialization import AntaresBaseModel @@ -128,6 +129,7 @@ def compute(self, x: Any, use_coords: bool = False) -> Any: return x return OPERATIONS[self.operation](x, self.value) # type: ignore + @override def __str__(self) -> str: """Returns a string representation used in error messages.""" return f"['{self.operation}' {self.value}]" @@ -215,6 +217,7 @@ def validate_coordinates(cls, coordinates: Optional[List[Tuple[int, int]]]) -> O raise ValueError(f"Invalid coordinate {coordinate}: values must be greater than or equal to 0.") return coordinates + @override def __str__(self) -> str: """Returns a string representation used in error messages.""" diff --git a/antarest/matrixstore/matrix_garbage_collector.py b/antarest/matrixstore/matrix_garbage_collector.py index eecbf19f96..3d7e513b94 100644 --- a/antarest/matrixstore/matrix_garbage_collector.py +++ b/antarest/matrixstore/matrix_garbage_collector.py @@ -16,6 +16,8 @@ from pathlib import Path from typing import List, Set +from typing_extensions import override + from antarest.core.config import Config from antarest.core.interfaces.service import IService from antarest.core.utils.fastapi_sqlalchemy import db @@ -119,6 +121,7 @@ def _clean_matrices(self) -> None: self._delete_unused_saved_matrices(unused_matrices=unused_matrices) stopwatch.log_elapsed(lambda x: logger.info(f"Finished cleaning matrices in {x}s")) + @override def _loop(self) -> None: while True: try: diff --git a/antarest/matrixstore/model.py b/antarest/matrixstore/model.py index 2dccd350ab..a46dad9455 100644 --- a/antarest/matrixstore/model.py +++ b/antarest/matrixstore/model.py @@ -16,6 +16,7 @@ from sqlalchemy import Boolean, Column, DateTime, ForeignKey, Integer, String, Table # type: ignore from sqlalchemy.orm import relationship # type: ignore +from typing_extensions import override from antarest.core.persistence import Base from antarest.core.serialization import AntaresBaseModel @@ -41,10 +42,12 @@ class Matrix(Base): # type: ignore height: int = Column(Integer) created_at: datetime.datetime = Column(DateTime) + @override def __repr__(self) -> str: # pragma: no cover """Returns a string representation of the matrix.""" return f"Matrix(id={self.id}, shape={(self.height, self.width)}, created_at={self.created_at})" + @override def __eq__(self, other: t.Any) -> bool: if not isinstance(other, Matrix): return False @@ -101,10 +104,12 @@ class MatrixDataSetRelation(Base): # type: ignore name: str = Column(String, primary_key=True) matrix: Matrix = relationship(Matrix) + @override def __repr__(self) -> str: # pragma: no cover """Returns a string representation of the matrix.""" return f"MatrixDataSetRelation(dataset_id={self.dataset_id}, matrix_id={self.matrix_id}, name={self.name})" + @override def __eq__(self, other: t.Any) -> bool: if not isinstance(other, MatrixDataSetRelation): return False @@ -172,6 +177,7 @@ def to_dto(self) -> MatrixDataSetDTO: updated_at=str(self.updated_at), ) + @override def __repr__(self) -> str: # pragma: no cover """Returns a string representation of the matrix.""" return ( @@ -183,6 +189,7 @@ def __repr__(self) -> str: # pragma: no cover f" updated_at={self.updated_at})" ) + @override def __eq__(self, other: t.Any) -> bool: if not isinstance(other, MatrixDataSet): return False diff --git a/antarest/matrixstore/service.py b/antarest/matrixstore/service.py index c2a8ca5c5d..055fbca413 100644 --- a/antarest/matrixstore/service.py +++ b/antarest/matrixstore/service.py @@ -24,6 +24,7 @@ import py7zr from fastapi import UploadFile from numpy import typing as npt +from typing_extensions import override from antarest.core.config import Config from antarest.core.filetransfer.model import FileDownloadTaskDTO @@ -112,9 +113,11 @@ class SimpleMatrixService(ISimpleMatrixService): def __init__(self, matrix_content_repository: MatrixContentRepository): super().__init__(matrix_content_repository=matrix_content_repository) + @override def create(self, data: t.Union[t.List[t.List[MatrixData]], npt.NDArray[np.float64]]) -> str: return self.matrix_content_repository.save(data) + @override def get(self, matrix_id: str) -> MatrixDTO: data = self.matrix_content_repository.get(matrix_id) return MatrixDTO.construct( @@ -126,9 +129,11 @@ def get(self, matrix_id: str) -> MatrixDTO: data=data.data, ) + @override def exists(self, matrix_id: str) -> bool: return self.matrix_content_repository.exists(matrix_id) + @override def delete(self, matrix_id: str) -> None: self.matrix_content_repository.delete(matrix_id) @@ -165,6 +170,7 @@ def _from_dto(dto: MatrixDTO) -> t.Tuple[Matrix, MatrixContent]: return matrix, content + @override def create(self, data: t.Union[t.List[t.List[MatrixData]], npt.NDArray[np.float64]]) -> str: """ Creates a new matrix object with the specified data. @@ -372,6 +378,7 @@ def delete_dataset(self, id: str, params: RequestParameters) -> str: self.repo_dataset.delete(id) return id + @override def get(self, matrix_id: str) -> t.Optional[MatrixDTO]: """ Get a matrix object from the database and the matrix content repository. @@ -397,6 +404,7 @@ def get(self, matrix_id: str) -> t.Optional[MatrixDTO]: data=content.data, ) + @override def exists(self, matrix_id: str) -> bool: """ Check if a matrix object exists in both the matrix content repository and the database. @@ -409,6 +417,7 @@ def exists(self, matrix_id: str) -> bool: """ return self.matrix_content_repository.exists(matrix_id) and self.repo.exists(matrix_id) + @override def delete(self, matrix_id: str) -> None: """ Delete a matrix object from the matrix content repository and the database. diff --git a/antarest/study/business/enum_ignore_case.py b/antarest/study/business/enum_ignore_case.py index 9d0bcf3396..56a06fabc1 100644 --- a/antarest/study/business/enum_ignore_case.py +++ b/antarest/study/business/enum_ignore_case.py @@ -13,6 +13,8 @@ import enum import typing +from typing_extensions import override + class EnumIgnoreCase(enum.StrEnum): """ @@ -35,6 +37,7 @@ class EnumIgnoreCase(enum.StrEnum): """ @classmethod + @override def _missing_(cls, value: object) -> typing.Optional["EnumIgnoreCase"]: if isinstance(value, str): for member in cls: diff --git a/antarest/study/business/scenario_builder_management.py b/antarest/study/business/scenario_builder_management.py index ce8834fbeb..71a7833cee 100644 --- a/antarest/study/business/scenario_builder_management.py +++ b/antarest/study/business/scenario_builder_management.py @@ -14,6 +14,7 @@ import typing as t import typing_extensions as te +from typing_extensions import override from antarest.study.business.utils import execute_or_add_commands from antarest.study.model import Study @@ -66,6 +67,7 @@ class ScenarioType(enum.StrEnum): HYDRO_FINAL_LEVEL = "hydroFinalLevels" HYDRO_GENERATION_POWER = "hydroGenerationPower" + @override def __str__(self) -> str: """Return the string representation of the enum value.""" return self.value diff --git a/antarest/study/business/table_mode_management.py b/antarest/study/business/table_mode_management.py index e7a12b0289..f5824d0296 100644 --- a/antarest/study/business/table_mode_management.py +++ b/antarest/study/business/table_mode_management.py @@ -16,6 +16,7 @@ import numpy as np import pandas as pd from antares.study.version import StudyVersion +from typing_extensions import override from antarest.core.exceptions import ChildNotFoundError from antarest.core.model import JSON @@ -61,6 +62,7 @@ class TableModeType(EnumIgnoreCase): BINDING_CONSTRAINT = "binding-constraints" @classmethod + @override def _missing_(cls, value: object) -> t.Optional["EnumIgnoreCase"]: if isinstance(value, str): # handle aliases of old table types diff --git a/antarest/study/model.py b/antarest/study/model.py index b8378aa356..9aed7c852a 100644 --- a/antarest/study/model.py +++ b/antarest/study/model.py @@ -31,6 +31,7 @@ String, ) from sqlalchemy.orm import relationship # type: ignore +from typing_extensions import override from antarest.core.exceptions import ShouldNotHappenException from antarest.core.model import PublicMode @@ -103,10 +104,12 @@ class StudyGroup(Base): # type:ignore group_id: str = Column(String(36), ForeignKey("groups.id", ondelete="CASCADE"), index=True, nullable=False) study_id: str = Column(String(36), ForeignKey("study.id", ondelete="CASCADE"), index=True, nullable=False) + @override def __str__(self) -> str: # pragma: no cover cls_name = self.__class__.__name__ return f"[{cls_name}] study_id={self.study_id}, group={self.group_id}" + @override def __repr__(self) -> str: # pragma: no cover cls_name = self.__class__.__name__ study_id = self.study_id @@ -129,10 +132,12 @@ class StudyTag(Base): # type:ignore study_id: str = Column(String(36), ForeignKey("study.id", ondelete="CASCADE"), index=True, nullable=False) tag_label: str = Column(String(40), ForeignKey("tag.label", ondelete="CASCADE"), index=True, nullable=False) + @override def __str__(self) -> str: # pragma: no cover cls_name = self.__class__.__name__ return f"[{cls_name}] study_id={self.study_id}, tag={self.tag}" + @override def __repr__(self) -> str: # pragma: no cover cls_name = self.__class__.__name__ study_id = self.study_id @@ -158,9 +163,11 @@ class Tag(Base): # type:ignore studies: t.List["Study"] = relationship("Study", secondary=StudyTag.__table__, back_populates="tags") + @override def __str__(self) -> str: # pragma: no cover return t.cast(str, self.label) + @override def __repr__(self) -> str: # pragma: no cover cls_name = self.__class__.__name__ label = self.label @@ -194,6 +201,7 @@ class StudyAdditionalData(Base): # type:ignore horizon = Column(String) patch = Column(String(), index=True, nullable=True) + @override def __eq__(self, other: t.Any) -> bool: if not super().__eq__(other): return False @@ -244,6 +252,7 @@ class Study(Base): # type: ignore __mapper_args__ = {"polymorphic_identity": "study", "polymorphic_on": type} + @override def __str__(self) -> str: cls = self.__class__.__name__ return ( @@ -258,6 +267,7 @@ def __str__(self) -> str: f" groups={[str(u) + ',' for u in self.groups]}" ) + @override def __eq__(self, other: t.Any) -> bool: if not isinstance(other, Study): return False @@ -299,6 +309,7 @@ class RawStudy(Study): "polymorphic_identity": "rawstudy", } + @override def __eq__(self, other: t.Any) -> bool: if not super().__eq__(other): return False diff --git a/antarest/study/service.py b/antarest/study/service.py index 3e198addb3..d394056f95 100644 --- a/antarest/study/service.py +++ b/antarest/study/service.py @@ -30,6 +30,7 @@ from fastapi import HTTPException, UploadFile from markupsafe import escape from starlette.responses import FileResponse, Response +from typing_extensions import override from antarest.core.config import Config from antarest.core.exceptions import ( @@ -212,6 +213,7 @@ class TaskProgressRecorder(ICommandListener): def __init__(self, notifier: ITaskNotifier) -> None: self.notifier = notifier + @override def notify_progress(self, progress: int) -> None: return self.notifier.notify_progress(progress) diff --git a/antarest/study/storage/abstract_storage_service.py b/antarest/study/storage/abstract_storage_service.py index 3b4c002597..f845eab757 100644 --- a/antarest/study/storage/abstract_storage_service.py +++ b/antarest/study/storage/abstract_storage_service.py @@ -18,6 +18,8 @@ from pathlib import Path from uuid import uuid4 +from typing_extensions import override + from antarest.core.config import Config from antarest.core.exceptions import BadOutputError, StudyOutputNotFoundError from antarest.core.interfaces.cache import CacheConstants, ICache @@ -63,6 +65,7 @@ def __init__( self.patch_service = patch_service self.cache = cache + @override def patch_update_study_metadata( self, study: T, @@ -82,6 +85,7 @@ def patch_update_study_metadata( remove_from_cache(self.cache, study.id) return self.get_study_information(study) + @override def get_study_information( self, study: T, @@ -129,6 +133,7 @@ def get_study_information( tags=[tag.label for tag in study.tags], ) + @override def get( self, metadata: T, @@ -170,6 +175,7 @@ def get( del study return data + @override def get_file( self, metadata: T, @@ -194,6 +200,7 @@ def get_file( return file_node.get_file_content() + @override def get_study_sim_result( self, study: T, @@ -243,6 +250,7 @@ def get_study_sim_result( ) return results + @override def import_output( self, metadata: T, @@ -302,6 +310,7 @@ def import_output( return output_full_name + @override def export_study(self, metadata: T, target: Path, outputs: bool = True) -> Path: """ Export and compress the study inside a 7zip file. @@ -326,6 +335,7 @@ def export_study(self, metadata: T, target: Path, outputs: bool = True) -> Path: ) return target + @override def export_output(self, metadata: T, output_id: str, target: Path) -> None: """ Export and compresses study inside zip @@ -358,6 +368,7 @@ def _read_additional_data_from_files(self, file_study: FileStudy) -> StudyAdditi study_additional_data = StudyAdditionalData(horizon=horizon, author=author, patch=patch.model_dump_json()) return study_additional_data + @override def archive_study_output(self, study: T, output_id: str) -> bool: try: archive_dir( @@ -375,6 +386,7 @@ def archive_study_output(self, study: T, output_id: str) -> bool: ) return False + @override def unarchive_study_output(self, study: T, output_id: str, keep_src_zip: bool) -> bool: if not (Path(study.path) / "output" / f"{output_id}{ArchiveFormat.ZIP}").exists(): logger.warning( diff --git a/antarest/study/storage/auto_archive_service.py b/antarest/study/storage/auto_archive_service.py index 8dbf6c3de0..5efd2dda1c 100644 --- a/antarest/study/storage/auto_archive_service.py +++ b/antarest/study/storage/auto_archive_service.py @@ -15,6 +15,8 @@ import time import typing as t +from typing_extensions import override + from antarest.core.config import Config from antarest.core.exceptions import TaskAlreadyRunning from antarest.core.interfaces.service import IService @@ -90,6 +92,7 @@ def _try_archive_studies(self) -> None: params=RequestParameters(DEFAULT_ADMIN_USER), ) + @override def _loop(self) -> None: while True: try: diff --git a/antarest/study/storage/df_download.py b/antarest/study/storage/df_download.py index 97882e2d2e..f1c4b80a3f 100644 --- a/antarest/study/storage/df_download.py +++ b/antarest/study/storage/df_download.py @@ -17,6 +17,7 @@ import pandas as pd from fastapi import HTTPException from starlette.responses import FileResponse +from typing_extensions import override from antarest.core.filetransfer.model import FileDownloadNotFound from antarest.core.filetransfer.service import FileTransferManager @@ -33,6 +34,7 @@ class TableExportFormat(EnumIgnoreCase): CSV = "csv" CSV_SEMICOLON = "csv (semicolon)" + @override def __str__(self) -> str: """Return the format as a string for display.""" return self.value.title() diff --git a/antarest/study/storage/rawstudy/ini_reader.py b/antarest/study/storage/rawstudy/ini_reader.py index fd3bac2dea..e51d9c9672 100644 --- a/antarest/study/storage/rawstudy/ini_reader.py +++ b/antarest/study/storage/rawstudy/ini_reader.py @@ -16,6 +16,8 @@ from abc import ABC, abstractmethod from pathlib import Path +from typing_extensions import override + from antarest.core.model import JSON @@ -168,6 +170,7 @@ def __init__(self, special_keys: t.Sequence[str] = (), section_name: str = "sett # Current option name used during paring self._curr_option = "" + @override def __repr__(self) -> str: # pragma: no cover """Return a string representation of the object.""" cls = self.__class__.__name__ @@ -176,6 +179,7 @@ def __repr__(self) -> str: # pragma: no cover section_name = getattr(self, "_section_name", "settings") return f"{cls}(special_keys={special_keys!r}, section_name={section_name!r})" + @override def read(self, path: t.Any, **kwargs: t.Any) -> JSON: if isinstance(path, (Path, str)): try: @@ -321,6 +325,7 @@ class SimpleKeyValueReader(IniReader): Simple INI reader for "settings.ini" file which has no section. """ + @override def read(self, path: t.Any, **kwargs: t.Any) -> JSON: """ Parse `.ini` file which has no section to JSON object. diff --git a/antarest/study/storage/rawstudy/ini_writer.py b/antarest/study/storage/rawstudy/ini_writer.py index f72f9b2197..d4ba73cf5d 100644 --- a/antarest/study/storage/rawstudy/ini_writer.py +++ b/antarest/study/storage/rawstudy/ini_writer.py @@ -15,6 +15,8 @@ import typing as t from pathlib import Path +from typing_extensions import override + from antarest.core.model import JSON @@ -24,6 +26,7 @@ def __init__(self, special_keys: t.Optional[t.List[str]] = None) -> None: self.special_keys = special_keys # noinspection SpellCheckingInspection + @override def optionxform(self, optionstr: str) -> str: return optionstr @@ -89,6 +92,7 @@ class SimpleKeyValueWriter(IniWriter): Simple key/value INI writer. """ + @override def write(self, data: JSON, path: Path) -> None: """ Write `.ini` file from JSON content diff --git a/antarest/study/storage/rawstudy/model/filesystem/bucket_node.py b/antarest/study/storage/rawstudy/model/filesystem/bucket_node.py index ad10455b2e..3def015763 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/bucket_node.py +++ b/antarest/study/storage/rawstudy/model/filesystem/bucket_node.py @@ -12,6 +12,8 @@ import typing as t +from typing_extensions import override + from antarest.core.model import JSON, SUB_JSON from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer @@ -54,6 +56,7 @@ def _get_registered_file_by_key(self, key: str) -> t.Optional[RegisteredFile]: def _get_registered_file_by_filename(self, filename: str) -> t.Optional[RegisteredFile]: return next((rf for rf in self.registered_files if rf.filename == filename), None) + @override def save( self, data: SUB_JSON, @@ -93,6 +96,7 @@ def _save(self, data: SUB_JSON, key: str) -> None: elif isinstance(data, dict): BucketNode(self.context, self.config.next_file(key)).save(data) + @override def build(self) -> TREE: if not self.config.path.is_dir(): return {} @@ -110,6 +114,7 @@ def build(self) -> TREE: return children + @override def check_errors( self, data: JSON, diff --git a/antarest/study/storage/rawstudy/model/filesystem/common/area_matrix_list.py b/antarest/study/storage/rawstudy/model/filesystem/common/area_matrix_list.py index 0cff5af838..e4022a3600 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/common/area_matrix_list.py +++ b/antarest/study/storage/rawstudy/model/filesystem/common/area_matrix_list.py @@ -12,6 +12,8 @@ import typing as t +from typing_extensions import override + from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode @@ -66,6 +68,7 @@ def __init__( self.matrix_class = matrix_class self.additional_matrix_params = additional_matrix_params or {} + @override def build(self) -> TREE: """ Builds the folder structure and creates child nodes representing each matrix file. @@ -100,6 +103,7 @@ def __init__( self.area = area self.matrix_class = matrix_class + @override def build(self) -> TREE: children: TREE = { "ror": self.matrix_class(self.context, self.config.next_file("ror.txt")), @@ -118,6 +122,7 @@ def __init__( super().__init__(context, config) self.matrix_class = matrix_class + @override def build(self) -> TREE: """Builds the folder structure and creates child nodes representing each matrix file.""" return { @@ -138,6 +143,7 @@ def __init__( self.area = area self.matrix_class = matrix_class + @override def build(self) -> TREE: # Note that cluster IDs are case-insensitive, but series IDs are in lower case. # For instance, if your cluster ID is "Base", then the series ID will be "base". @@ -185,6 +191,7 @@ def __init__( self.klass = klass self.matrix_class = matrix_class + @override def build(self) -> TREE: folders = [d.name for d in self.config.path.iterdir() if d.is_dir()] children: TREE = { diff --git a/antarest/study/storage/rawstudy/model/filesystem/common/prepro.py b/antarest/study/storage/rawstudy/model/filesystem/common/prepro.py index 1813760f13..b18d53b81d 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/common/prepro.py +++ b/antarest/study/storage/rawstudy/model/filesystem/common/prepro.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer @@ -44,6 +45,7 @@ def __init__(self, context: ContextServer, config: FileStudyTreeConfig): class PreproArea(FolderNode): + @override def build(self) -> TREE: children: TREE = { "conversion": InputSeriesMatrix(self.context, self.config.next_file("conversion.txt")), @@ -56,6 +58,7 @@ def build(self) -> TREE: class InputPrepro(FolderNode): + @override def build(self) -> TREE: children: TREE = {a: PreproArea(self.context, self.config.next_file(a)) for a in self.config.area_names()} children["correlation"] = PreproCorrelation(self.context, self.config.next_file("correlation.ini")) diff --git a/antarest/study/storage/rawstudy/model/filesystem/config/area.py b/antarest/study/storage/rawstudy/model/filesystem/config/area.py index 74d15e5b55..db83868816 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/config/area.py +++ b/antarest/study/storage/rawstudy/model/filesystem/config/area.py @@ -17,6 +17,7 @@ import typing as t from pydantic import Field, field_validator, model_validator +from typing_extensions import override from antarest.study.business.enum_ignore_case import EnumIgnoreCase from antarest.study.storage.rawstudy.model.filesystem.config.field_validators import ( @@ -185,6 +186,7 @@ def _validate_color_rgb(cls, v: t.Any) -> str: def _validate_colors(cls, values: t.MutableMapping[str, t.Any]) -> t.Mapping[str, t.Any]: return validate_colors(values) + @override def to_config(self) -> t.Dict[str, t.Any]: """ Convert the object to a dictionary for writing to a configuration file: @@ -342,6 +344,7 @@ def _validate_layers(cls, values: t.MutableMapping[str, t.Any]) -> t.Mapping[str return values + @override def to_config(self) -> t.Dict[str, t.Dict[str, t.Any]]: """ Convert the object to a dictionary for writing to a configuration file: diff --git a/antarest/study/storage/rawstudy/model/filesystem/config/exceptions.py b/antarest/study/storage/rawstudy/model/filesystem/config/exceptions.py index 936f7f76e2..3a903d9cf4 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/config/exceptions.py +++ b/antarest/study/storage/rawstudy/model/filesystem/config/exceptions.py @@ -13,6 +13,8 @@ from pathlib import Path from typing import cast +from typing_extensions import override + class BaseConfigError(Exception): """Base class of the configuration errors.""" @@ -30,6 +32,7 @@ def output_path(self) -> Path: def reason(self) -> str: return cast(str, self.args[1]) + @override def __str__(self) -> str: output_path = self.output_path reason = self.reason @@ -48,6 +51,7 @@ def xpansion_json(self) -> Path: def reason(self) -> str: return cast(str, self.args[1]) + @override def __str__(self) -> str: xpansion_json = self.xpansion_json reason = self.reason diff --git a/antarest/study/storage/rawstudy/model/filesystem/config/ini_properties.py b/antarest/study/storage/rawstudy/model/filesystem/config/ini_properties.py index c5c3c83950..abbf6fb77b 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/config/ini_properties.py +++ b/antarest/study/storage/rawstudy/model/filesystem/config/ini_properties.py @@ -12,6 +12,8 @@ import typing as t +from typing_extensions import override + from antarest.core.serialization import AntaresBaseModel, from_json, to_json @@ -52,6 +54,7 @@ def to_config(self) -> t.Dict[str, t.Any]: return config @classmethod + @override def construct(cls, _fields_set: t.Optional[t.Set[str]] = None, **values: t.Any) -> "IniProperties": """ Construct a new model instance from a dict of values, replacing aliases with real field names. diff --git a/antarest/study/storage/rawstudy/model/filesystem/config/model.py b/antarest/study/storage/rawstudy/model/filesystem/config/model.py index 938cddaba3..bd35498e38 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/config/model.py +++ b/antarest/study/storage/rawstudy/model/filesystem/config/model.py @@ -16,6 +16,7 @@ from antares.study.version import StudyVersion from pydantic import Field, model_validator +from typing_extensions import override from antarest.core.serialization import AntaresBaseModel from antarest.core.utils.utils import DTO @@ -47,6 +48,7 @@ class EnrModelling(EnumIgnoreCase): AGGREGATED = "aggregated" CLUSTERS = "clusters" + @override def __str__(self) -> str: """Return the string representation of the enum value.""" return self.value diff --git a/antarest/study/storage/rawstudy/model/filesystem/config/renewable.py b/antarest/study/storage/rawstudy/model/filesystem/config/renewable.py index dcd7e6f16c..d9f4f9897e 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/config/renewable.py +++ b/antarest/study/storage/rawstudy/model/filesystem/config/renewable.py @@ -14,6 +14,7 @@ from antares.study.version import StudyVersion from pydantic import Field +from typing_extensions import override from antarest.study.business.enum_ignore_case import EnumIgnoreCase from antarest.study.model import STUDY_VERSION_8_1 @@ -54,10 +55,12 @@ class RenewableClusterGroup(EnumIgnoreCase): OTHER3 = "other res 3" OTHER4 = "other res 4" + @override def __repr__(self) -> str: return f"{self.__class__.__name__}.{self.name}" @classmethod + @override def _missing_(cls, value: object) -> t.Optional["RenewableClusterGroup"]: """ Retrieves the default group or the matched group when an unknown value is encountered. diff --git a/antarest/study/storage/rawstudy/model/filesystem/config/ruleset_matrices.py b/antarest/study/storage/rawstudy/model/filesystem/config/ruleset_matrices.py index 6ed47a3e17..ca9ff2e724 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/config/ruleset_matrices.py +++ b/antarest/study/storage/rawstudy/model/filesystem/config/ruleset_matrices.py @@ -15,6 +15,7 @@ import numpy as np import pandas as pd import typing_extensions as te +from typing_extensions import override SCENARIO_TYPES = { "l": "load", @@ -107,6 +108,7 @@ def __init__( self.scenarios: _ScenarioMapping = {} self._setup() + @override def __str__(self) -> str: lines = [] for symbol, scenario_type in self.scenario_types.items(): diff --git a/antarest/study/storage/rawstudy/model/filesystem/config/thermal.py b/antarest/study/storage/rawstudy/model/filesystem/config/thermal.py index 5093c9c289..78794052f9 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/config/thermal.py +++ b/antarest/study/storage/rawstudy/model/filesystem/config/thermal.py @@ -14,6 +14,7 @@ from antares.study.version import StudyVersion from pydantic import Field, ValidationError +from typing_extensions import override from antarest.study.business.enum_ignore_case import EnumIgnoreCase from antarest.study.storage.rawstudy.model.filesystem.config.cluster import ClusterProperties @@ -35,6 +36,7 @@ class LocalTSGenerationBehavior(EnumIgnoreCase): FORCE_NO_GENERATION = "force no generation" FORCE_GENERATION = "force generation" + @override def __repr__(self) -> str: # pragma: no cover return f"{self.__class__.__name__}.{self.name}" @@ -48,6 +50,7 @@ class LawOption(EnumIgnoreCase): UNIFORM = "uniform" GEOMETRIC = "geometric" + @override def __repr__(self) -> str: # pragma: no cover return f"{self.__class__.__name__}.{self.name}" @@ -69,10 +72,12 @@ class ThermalClusterGroup(EnumIgnoreCase): OTHER3 = "other 3" OTHER4 = "other 4" + @override def __repr__(self) -> str: # pragma: no cover return f"{self.__class__.__name__}.{self.name}" @classmethod + @override def _missing_(cls, value: object) -> t.Optional["ThermalClusterGroup"]: """ Retrieves the default group or the matched group when an unknown value is encountered. diff --git a/antarest/study/storage/rawstudy/model/filesystem/folder_node.py b/antarest/study/storage/rawstudy/model/filesystem/folder_node.py index 974c2acdfa..c8b3604e88 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/folder_node.py +++ b/antarest/study/storage/rawstudy/model/filesystem/folder_node.py @@ -14,6 +14,8 @@ import typing as t from abc import ABC, abstractmethod +from typing_extensions import override + from antarest.core.exceptions import ChildNotFoundError, PathIsAFolderError from antarest.core.model import JSON, SUB_JSON from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig @@ -114,6 +116,7 @@ def _get( else: return self._expand_get(depth, formatted, get_node) + @override def get( self, url: t.Optional[t.List[str]] = None, @@ -125,6 +128,7 @@ def get( assert not isinstance(output, INode) return output + @override def get_node( self, url: t.Optional[t.List[str]] = None, @@ -133,6 +137,7 @@ def get_node( assert isinstance(output, INode) return output + @override def save( self, data: SUB_JSON, @@ -151,6 +156,7 @@ def save( for key in data: children[key].save(data[key]) + @override def delete(self, url: t.Optional[t.List[str]] = None) -> None: if url and url != [""]: children = self.build() @@ -160,6 +166,7 @@ def delete(self, url: t.Optional[t.List[str]] = None) -> None: elif self.config.path.exists(): shutil.rmtree(self.config.path) + @override def check_errors( self, data: JSON, @@ -183,10 +190,12 @@ def check_errors( errors += children[key].check_errors(data[key], raising=raising) return errors + @override def normalize(self) -> None: for child in self.build().values(): child.normalize() + @override def denormalize(self) -> None: for child in self.build().values(): child.denormalize() @@ -217,6 +226,7 @@ def extract_child(self, children: TREE, url: t.List[str]) -> t.Tuple[t.List[str] raise FilterError("Filter selection has different classes") return names, sub_url + @override def get_file_content(self) -> OriginalFile: relative_path = self.config.path.relative_to(self.config.study_path).as_posix() raise PathIsAFolderError(f"Node at {relative_path} is a folder node.") diff --git a/antarest/study/storage/rawstudy/model/filesystem/ini_file_node.py b/antarest/study/storage/rawstudy/model/filesystem/ini_file_node.py index 55a8645ab6..dda7a2a6c0 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/ini_file_node.py +++ b/antarest/study/storage/rawstudy/model/filesystem/ini_file_node.py @@ -23,6 +23,7 @@ import py7zr import pydantic_core from filelock import FileLock +from typing_extensions import override from antarest.core.exceptions import ShouldNotHappenException from antarest.core.model import JSON, SUB_JSON @@ -168,6 +169,7 @@ def _get_filtering_kwargs(self, url: t.List[str]) -> t.Dict[str, str]: else: return {} + @override def get( self, url: t.Optional[t.List[str]] = None, @@ -179,6 +181,7 @@ def get( assert not isinstance(output, INode) return output + @override def get_node( self, url: t.Optional[t.List[str]] = None, @@ -187,6 +190,7 @@ def get_node( assert isinstance(output, INode) return output + @override def save(self, data: SUB_JSON, url: t.Optional[t.List[str]] = None) -> None: self._assert_not_in_zipped_file() url = url or [] @@ -212,6 +216,7 @@ def save(self, data: SUB_JSON, url: t.Optional[t.List[str]] = None) -> None: self.writer.write(info, self.path) @log_warning + @override def delete(self, url: t.Optional[t.List[str]] = None) -> None: """ Deletes the specified section or key from the INI file, @@ -274,6 +279,7 @@ def delete(self, url: t.Optional[t.List[str]] = None) -> None: self.writer.write(data, self.path) + @override def check_errors( self, data: JSON, @@ -292,9 +298,11 @@ def check_errors( return errors + @override def normalize(self) -> None: pass # no external store in this node + @override def denormalize(self) -> None: pass # no external store in this node diff --git a/antarest/study/storage/rawstudy/model/filesystem/json_file_node.py b/antarest/study/storage/rawstudy/model/filesystem/json_file_node.py index 54333e2c4b..ca877e9fd6 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/json_file_node.py +++ b/antarest/study/storage/rawstudy/model/filesystem/json_file_node.py @@ -14,6 +14,8 @@ import typing as t from pathlib import Path +from typing_extensions import override + from antarest.core.model import JSON from antarest.core.serialization import from_json, to_json from antarest.study.storage.rawstudy.ini_reader import IReader @@ -28,6 +30,7 @@ class JsonReader(IReader): JSON file reader. """ + @override def read(self, path: t.Any, **kwargs: t.Any) -> JSON: content: t.Union[str, bytes] @@ -59,6 +62,7 @@ class JsonWriter(IniWriter): JSON file writer. """ + @override def write(self, data: JSON, path: Path) -> None: with open(path, "wb") as fh: fh.write(to_json(data)) diff --git a/antarest/study/storage/rawstudy/model/filesystem/lazy_node.py b/antarest/study/storage/rawstudy/model/filesystem/lazy_node.py index 296f3efc13..d442d9d732 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/lazy_node.py +++ b/antarest/study/storage/rawstudy/model/filesystem/lazy_node.py @@ -16,6 +16,8 @@ from pathlib import Path from zipfile import ZipFile +from typing_extensions import override + from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer from antarest.study.storage.rawstudy.model.filesystem.inode import G, INode, S, V @@ -92,6 +94,7 @@ def _get( else: return self.load(url, depth, expanded, formatted) + @override def get( self, url: t.Optional[t.List[str]] = None, @@ -103,6 +106,7 @@ def get( assert not isinstance(output, INode) return output + @override def get_node( self, url: t.Optional[t.List[str]] = None, @@ -111,6 +115,7 @@ def get_node( assert isinstance(output, INode) return output + @override def delete(self, url: t.Optional[t.List[str]] = None) -> None: self._assert_url_end(url) if self.get_link_path().exists(): @@ -122,6 +127,7 @@ def get_link_path(self) -> Path: path = self.config.path.parent / (self.config.path.name + ".link") return path + @override def save(self, data: t.Union[str, bytes, S], url: t.Optional[t.List[str]] = None) -> None: self._assert_not_in_zipped_file() self._assert_url_end(url) diff --git a/antarest/study/storage/rawstudy/model/filesystem/matrix/date_serializer.py b/antarest/study/storage/rawstudy/model/filesystem/matrix/date_serializer.py index 2b8c431085..ef856c5f91 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/matrix/date_serializer.py +++ b/antarest/study/storage/rawstudy/model/filesystem/matrix/date_serializer.py @@ -17,6 +17,7 @@ from typing import Hashable, List, Sequence, Tuple, cast import pandas as pd +from typing_extensions import override class IDateMatrixSerializer(ABC): @@ -80,6 +81,7 @@ class HourlyMatrixSerializer(IDateMatrixSerializer): Class implementation for hourly index """ + @override def build_date(self, index: pd.Index[str]) -> pd.DataFrame: def _map(row: str) -> Tuple[str, int, str, str, str]: m, d, h = re.split(r"[\s/]", row) @@ -99,6 +101,7 @@ def _map(row: str) -> Tuple[str, int, str, str, str]: return pd.concat([headers, matrix], axis=0) + @override def extract_date(self, df: pd.DataFrame) -> Tuple[pd.Index[str], pd.DataFrame]: # Extract left part with date df_date = df.iloc[:, 2:5] @@ -118,6 +121,7 @@ class DailyMatrixSerializer(IDateMatrixSerializer): Class implementation for daily index """ + @override def build_date(self, index: pd.Index[str]) -> pd.DataFrame: def _map(row: str) -> Tuple[str, int, str, str]: m, d = row.split("/") @@ -137,6 +141,7 @@ def _map(row: str) -> Tuple[str, int, str, str]: return pd.concat([headers, matrix], axis=0) + @override def extract_date(self, df: pd.DataFrame) -> Tuple[pd.Index[str], pd.DataFrame]: # Extract left part with date df_date = df.iloc[:, 2:4] @@ -156,6 +161,7 @@ class WeeklyMatrixSerializer(IDateMatrixSerializer): Class implementation for weekly index """ + @override def build_date(self, index: pd.Index[str]) -> pd.DataFrame: matrix = pd.DataFrame({0: [""] * index.size, 1: index.values}) @@ -169,6 +175,7 @@ def build_date(self, index: pd.Index[str]) -> pd.DataFrame: return pd.concat([headers, matrix], axis=0) + @override def extract_date(self, df: pd.DataFrame) -> Tuple[pd.Index[str], pd.DataFrame]: # Extract left part with date df_date = df.iloc[:, 1:2] @@ -186,6 +193,7 @@ class MonthlyMatrixSerializer(IDateMatrixSerializer): Class implementation for monthly index """ + @override def build_date(self, index: pd.Index[str]) -> pd.DataFrame: matrix = pd.DataFrame( { @@ -205,6 +213,7 @@ def build_date(self, index: pd.Index[str]) -> pd.DataFrame: return pd.concat([headers, matrix], axis=0) + @override def extract_date(self, df: pd.DataFrame) -> Tuple[pd.Index[str], pd.DataFrame]: # Extract left part with date df_date = df.iloc[:, 2:3] @@ -224,6 +233,7 @@ class AnnualMatrixSerializer(IDateMatrixSerializer): Class implementation for annual index """ + @override def build_date(self, index: pd.Index[str]) -> pd.DataFrame: return pd.DataFrame( [ @@ -234,6 +244,7 @@ def build_date(self, index: pd.Index[str]) -> pd.DataFrame: ] ) + @override def extract_date(self, df: pd.DataFrame) -> Tuple[pd.Index[str], pd.DataFrame]: # Extract left part with date df_date = df.iloc[:, 1:2] diff --git a/antarest/study/storage/rawstudy/model/filesystem/matrix/head_writer.py b/antarest/study/storage/rawstudy/model/filesystem/matrix/head_writer.py index 5909f69ba7..7c8d2fd588 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/matrix/head_writer.py +++ b/antarest/study/storage/rawstudy/model/filesystem/matrix/head_writer.py @@ -12,6 +12,8 @@ from abc import ABC, abstractmethod +from typing_extensions import override + class HeadWriter(ABC): """ @@ -43,6 +45,7 @@ def __init__(self, area: str, data_type: str, freq: str): \tVARIABLES\tBEGIN\tEND """ + @override def build(self, var: int, end: int, start: int = 1) -> str: return self.head + f"\t{var}\t{start}\t{end}\n\n" @@ -57,5 +60,6 @@ def __init__(self, src: str, dest: str, freq: str): {dest.upper()}\tVARIABLES\tBEGIN\tEND """ + @override def build(self, var: int, end: int, start: int = 1) -> str: return self.head + f"\t{var}\t{start}\t{end}\n\n" diff --git a/antarest/study/storage/rawstudy/model/filesystem/matrix/input_series_matrix.py b/antarest/study/storage/rawstudy/model/filesystem/matrix/input_series_matrix.py index ec9d04a2a9..d2ac83c19c 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/matrix/input_series_matrix.py +++ b/antarest/study/storage/rawstudy/model/filesystem/matrix/input_series_matrix.py @@ -19,6 +19,7 @@ import pandas as pd from numpy import typing as npt from pandas.errors import EmptyDataError +from typing_extensions import override from antarest.core.exceptions import ChildNotFoundError from antarest.core.model import JSON @@ -89,6 +90,7 @@ def parse_as_dataframe(self, file_path: t.Optional[Path] = None) -> pd.DataFrame final_matrix = pd.DataFrame(self.default_empty) return final_matrix + @override def parse_as_json(self, file_path: t.Optional[Path] = None) -> JSON: df = self.parse_as_dataframe(file_path) stopwatch = StopWatch() @@ -96,6 +98,7 @@ def parse_as_json(self, file_path: t.Optional[Path] = None) -> JSON: stopwatch.log_elapsed(lambda x: logger.info(f"Matrix to dict in {x}s")) return data + @override def check_errors( self, data: JSON, @@ -128,6 +131,7 @@ def copy_file(self, target: str) -> None: target_path.unlink(missing_ok=True) shutil.copy(self._infer_path(), target_path) + @override def get_file_content(self) -> OriginalFile: suffix = self.config.path.suffix filename = self.config.path.name diff --git a/antarest/study/storage/rawstudy/model/filesystem/matrix/matrix.py b/antarest/study/storage/rawstudy/model/filesystem/matrix/matrix.py index 6af089a8a5..7897715f92 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/matrix/matrix.py +++ b/antarest/study/storage/rawstudy/model/filesystem/matrix/matrix.py @@ -17,6 +17,7 @@ from typing import List, Optional, Union, cast import pandas as pd +from typing_extensions import override from antarest.core.model import JSON from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig @@ -64,6 +65,7 @@ def __init__( LazyNode.__init__(self, context, config) self.freq = freq + @override def get_lazy_content( self, url: Optional[List[str]] = None, @@ -72,6 +74,7 @@ def get_lazy_content( ) -> str: return f"matrixfile://{self.config.path.name}" + @override def normalize(self) -> None: # noinspection SpellCheckingInspection """ @@ -95,6 +98,7 @@ def normalize(self) -> None: self.get_link_path().write_text(self.context.resolver.build_matrix_uri(uuid)) self.config.path.unlink() + @override def denormalize(self) -> None: """ Read the matrix ID from the matrix link, retrieve the original matrix @@ -114,6 +118,7 @@ def denormalize(self) -> None: self.dump(matrix) self.get_link_path().unlink() + @override def load( self, url: Optional[List[str]] = None, @@ -140,6 +145,7 @@ def parse_as_json(self, file_path: Optional[Path] = None) -> JSON: """ raise NotImplementedError() + @override def dump( self, data: Union[bytes, JSON], diff --git a/antarest/study/storage/rawstudy/model/filesystem/matrix/output_series_matrix.py b/antarest/study/storage/rawstudy/model/filesystem/matrix/output_series_matrix.py index 52c9867b2f..c3b66d1245 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/matrix/output_series_matrix.py +++ b/antarest/study/storage/rawstudy/model/filesystem/matrix/output_series_matrix.py @@ -16,6 +16,7 @@ import pandas as pd from pandas import DataFrame +from typing_extensions import override from antarest.core.exceptions import ChildNotFoundError, MustNotModifyOutputException from antarest.core.model import JSON @@ -56,6 +57,7 @@ def __init__( self.head_writer = head_writer self.freq = freq + @override def get_lazy_content( self, url: Optional[List[str]] = None, @@ -106,6 +108,7 @@ def parse( matrix = self.parse_dataframe(file_path, tmp_dir) return cast(JSON, matrix.to_dict(orient="split")) + @override def check_errors( self, data: JSON, @@ -119,6 +122,7 @@ def check_errors( errors.append(f"Output Series Matrix f{self.config.path} not exists") return errors + @override def load( self, url: Optional[List[str]] = None, @@ -148,12 +152,15 @@ def load( f"Output file '{self.config.path.name}' not found in study {self.config.study_id}" ) from e + @override def dump(self, data: Union[bytes, JSON], url: Optional[List[str]] = None) -> None: raise MustNotModifyOutputException(self.config.path.name) + @override def normalize(self) -> None: pass # no external store in this node + @override def denormalize(self) -> None: pass # no external store in this node diff --git a/antarest/study/storage/rawstudy/model/filesystem/raw_file_node.py b/antarest/study/storage/rawstudy/model/filesystem/raw_file_node.py index c92f3d9152..7752e93e0c 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/raw_file_node.py +++ b/antarest/study/storage/rawstudy/model/filesystem/raw_file_node.py @@ -13,6 +13,8 @@ import logging from typing import List, Optional +from typing_extensions import override + from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer from antarest.study.storage.rawstudy.model.filesystem.lazy_node import LazyNode @@ -28,6 +30,7 @@ class RawFileNode(LazyNode[bytes, bytes, str]): def __init__(self, context: ContextServer, config: FileStudyTreeConfig): LazyNode.__init__(self, config=config, context=context) + @override def get_lazy_content( self, url: Optional[List[str]] = None, @@ -36,6 +39,7 @@ def get_lazy_content( ) -> str: return f"file://{self.config.path.name}" + @override def load( self, url: Optional[List[str]] = None, @@ -56,10 +60,12 @@ def load( return bytes + @override def dump(self, data: bytes, url: Optional[List[str]] = None) -> None: self.config.path.parent.mkdir(exist_ok=True, parents=True) self.config.path.write_bytes(data) + @override def check_errors(self, data: str, url: Optional[List[str]] = None, raising: bool = False) -> List[str]: if not self.config.path.exists(): msg = f"{self.config.path} not exist" @@ -68,8 +74,10 @@ def check_errors(self, data: str, url: Optional[List[str]] = None, raising: bool return [msg] return [] + @override def normalize(self) -> None: pass # no external store in this node + @override def denormalize(self) -> None: pass # no external store in this node diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/filestudytree.py b/antarest/study/storage/rawstudy/model/filesystem/root/filestudytree.py index bfef345ccc..409c8aac7f 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/filestudytree.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/filestudytree.py @@ -12,6 +12,8 @@ import logging +from typing_extensions import override + from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE from antarest.study.storage.rawstudy.model.filesystem.root.desktop import Desktop @@ -31,6 +33,7 @@ class FileStudyTree(FolderNode): Top level node of antares tree structure """ + @override def build(self) -> TREE: children: TREE = { "Desktop": Desktop(self.context, self.config.next_file("Desktop.ini")), diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/areas/areas.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/areas/areas.py index 7dadadcc87..c9dd2b4e41 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/areas/areas.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/areas/areas.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer @@ -23,6 +24,7 @@ class InputAreas(FolderNode): def __init__(self, context: ContextServer, config: FileStudyTreeConfig): super().__init__(context, config, ["list", "sets"]) + @override def build(self) -> TREE: children: TREE = {a: InputAreasItem(self.context, self.config.next_file(a)) for a in self.config.area_names()} children["list"] = InputAreasList(self.context, self.config.next_file("list.txt")) diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/areas/item/item.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/areas/item/item.py index 5ea6ad808a..822e3f80a3 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/areas/item/item.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/areas/item/item.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.model import STUDY_VERSION_8_3 from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode @@ -21,6 +22,7 @@ class InputAreasItem(FolderNode): + @override def build(self) -> TREE: children: TREE = { "ui": InputAreasUi(self.context, self.config.next_file("ui.ini")), diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/areas/list.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/areas/list.py index 04564300c6..cdc26f097e 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/areas/list.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/areas/list.py @@ -12,6 +12,8 @@ import typing as t +from typing_extensions import override + from antarest.core.utils.archives import extract_lines_from_archive from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer @@ -21,9 +23,11 @@ class InputAreasList(INode[t.List[str], t.List[str], t.List[str]]): + @override def normalize(self) -> None: pass # no external store in this node + @override def denormalize(self) -> None: pass # no external store in this node @@ -31,6 +35,7 @@ def __init__(self, context: ContextServer, config: FileStudyTreeConfig): super().__init__(config) self.context = context + @override def get_node( self, url: t.Optional[t.List[str]] = None, @@ -40,6 +45,7 @@ def get_node( ) -> INode[t.List[str], t.List[str], t.List[str]]: return self + @override def get( self, url: t.Optional[t.List[str]] = None, @@ -53,14 +59,17 @@ def get( lines = self.config.path.read_text().split("\n") return [l.strip() for l in lines if l.strip()] + @override def save(self, data: t.List[str], url: t.Optional[t.List[str]] = None) -> None: self._assert_not_in_zipped_file() self.config.path.write_text("\n".join(data)) + @override def delete(self, url: t.Optional[t.List[str]] = None) -> None: if self.config.path.exists(): self.config.path.unlink() + @override def check_errors( self, data: t.List[str], diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/bindingconstraints/bindingcontraints.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/bindingconstraints/bindingcontraints.py index 28dba53eb5..fedaa47d54 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/bindingconstraints/bindingcontraints.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/bindingconstraints/bindingcontraints.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.config.binding_constraint import ( OPERATOR_MATRICES_MAP, @@ -41,6 +42,7 @@ class BindingConstraints(FolderNode): configuration and matrices. """ + @override def build(self) -> TREE: cfg = self.config frequency_mapping = { diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/commons/prepro_series.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/commons/prepro_series.py index 277e2311be..74c99bf04b 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/commons/prepro_series.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/commons/prepro_series.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.common.area_matrix_list import AreaMatrixList from antarest.study.storage.rawstudy.model.filesystem.common.prepro import InputPrepro @@ -51,6 +52,7 @@ def __init__(self, context: ContextServer, config: FileStudyTreeConfig, prefix: super().__init__(context, config) self.prefix = prefix + @override def build(self) -> TREE: children: TREE = { "prepro": InputPrepro(self.context, self.config.next_file("prepro")), diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/allocation/allocation.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/allocation/allocation.py index 0714f710a5..afa09c4c0e 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/allocation/allocation.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/allocation/allocation.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -16,6 +17,7 @@ class InputHydroAllocation(FolderNode): + @override def build(self) -> TREE: children: TREE = { a: InputHydroAllocationArea(self.context, self.config.next_file(f"{a}.ini"), area=a) diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/common/capacity/capacity.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/common/capacity/capacity.py index 56eeceadd2..33a6e999e7 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/common/capacity/capacity.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/common/capacity/capacity.py @@ -13,6 +13,7 @@ from typing import List, TypedDict from antares.study.version import StudyVersion +from typing_extensions import override from antarest.study.model import STUDY_VERSION_6_5 from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode @@ -59,6 +60,7 @@ class MatrixInfo(TypedDict, total=False): class InputHydroCommonCapacity(FolderNode): + @override def build(self) -> TREE: children: TREE = {} for info in MATRICES_INFO: diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/common/common.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/common/common.py index 46715ff6b7..d89d80014a 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/common/common.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/common/common.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -18,6 +19,7 @@ class InputHydroCommon(FolderNode): + @override def build(self) -> TREE: children: TREE = {"capacity": InputHydroCommonCapacity(self.context, self.config.next_file("capacity"))} return children diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/hydro.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/hydro.py index c1c3d2984a..f5e021ce61 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/hydro.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/hydro.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -20,6 +21,7 @@ class InputHydro(FolderNode): + @override def build(self) -> TREE: children: TREE = { "allocation": InputHydroAllocation(self.context, self.config.next_file("allocation")), diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/prepro/area/area.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/prepro/area/area.py index 13d504d1c0..bdd8ac80db 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/prepro/area/area.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/prepro/area/area.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -19,6 +20,7 @@ class InputHydroPreproArea(FolderNode): + @override def build(self) -> TREE: children: TREE = { "energy": InputSeriesMatrix(self.context, self.config.next_file("energy.txt")), diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/prepro/prepro.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/prepro/prepro.py index 93104392f3..ba884200d5 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/prepro/prepro.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/prepro/prepro.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.common.prepro import PreproCorrelation from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode @@ -17,6 +18,7 @@ class InputHydroPrepro(FolderNode): + @override def build(self) -> TREE: children: TREE = { a: InputHydroPreproArea(self.context, self.config.next_file(a)) for a in self.config.area_names() diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/series/area/area.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/series/area/area.py index 792fb9335f..4c6ada156a 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/series/area/area.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/series/area/area.py @@ -12,6 +12,8 @@ from typing import Any, Dict +from typing_extensions import override + from antarest.study.model import STUDY_VERSION_6_5, STUDY_VERSION_8_6 from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE, INode @@ -25,6 +27,7 @@ class InputHydroSeriesArea(FolderNode): + @override def build(self) -> TREE: study_version = self.config.version freq = MatrixFrequency.DAILY if study_version >= STUDY_VERSION_6_5 else MatrixFrequency.MONTHLY diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/series/series.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/series/series.py index 0a92743b2b..bb22eab408 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/series/series.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/hydro/series/series.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -16,6 +17,7 @@ class InputHydroSeries(FolderNode): + @override def build(self) -> TREE: children: TREE = { a: InputHydroSeriesArea(self.context, self.config.next_file(a)) for a in self.config.area_names() diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/input.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/input.py index 1c19358baf..429831b818 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/input.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/input.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.model import STUDY_VERSION_8_1, STUDY_VERSION_8_6 from antarest.study.storage.rawstudy.model.filesystem.config.model import EnrModelling @@ -33,6 +34,7 @@ class Input(FolderNode): Handle the input folder which contains all the input data of the study. """ + @override def build(self) -> TREE: config = self.config diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/link/area/area.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/link/area/area.py index a67fbf6dfc..1eefe1d53c 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/link/area/area.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/link/area/area.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.model import STUDY_VERSION_8_2 from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig @@ -32,6 +33,7 @@ def __init__( super().__init__(context, config) self.area = area + @override def build(self) -> TREE: children: TREE ctx = self.context diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/link/area/capacities/capacities.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/link/area/capacities/capacities.py index cb86c24c89..aabc88f5f4 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/link/area/capacities/capacities.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/link/area/capacities/capacities.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer @@ -27,6 +28,7 @@ def __init__( super().__init__(context, config) self.area = area + @override def build(self) -> TREE: children: TREE = {} for area_to in self.config.get_links(self.area): diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/link/link.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/link/link.py index e12f804f9a..d94faabe71 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/link/link.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/link/link.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -16,6 +17,7 @@ class InputLink(FolderNode): + @override def build(self) -> TREE: children: TREE = { a: InputLinkArea(self.context, self.config.next_file(a), area=a) for a in self.config.area_names() diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/miscgen/miscgen.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/miscgen/miscgen.py index 0590793e5c..213c9b9632 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/miscgen/miscgen.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/miscgen/miscgen.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -17,6 +18,7 @@ class InputMiscGen(FolderNode): + @override def build(self) -> TREE: children: TREE = { f"miscgen-{a}": InputSeriesMatrix( diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/renewables/clusters.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/renewables/clusters.py index d1be73eff4..5f82229251 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/renewables/clusters.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/renewables/clusters.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer @@ -46,11 +47,13 @@ def __init__( super().__init__(context, config) self.area = area + @override def build(self) -> TREE: return {"list": ClusteredRenewableClusterConfig(self.context, self.config.next_file("list.ini"), self.area)} class ClusteredRenewableAreaCluster(FolderNode): + @override def build(self) -> TREE: return { area: ClusteredRenewableCluster(self.context, self.config.next_file(area), area) diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/renewables/renewable.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/renewables/renewable.py index e9b7181a19..aa6e9bd94b 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/renewables/renewable.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/renewables/renewable.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -19,6 +20,7 @@ class ClusteredRenewables(FolderNode): + @override def build(self) -> TREE: children: TREE = { "clusters": ClusteredRenewableAreaCluster(self.context, self.config.next_file("clusters")), diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/renewables/series.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/renewables/series.py index cf323c311d..91aa09642d 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/renewables/series.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/renewables/series.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer @@ -19,6 +20,7 @@ class ClusteredRenewableSeries(FolderNode): + @override def build(self) -> TREE: series_config = self.config.next_file("series.txt") children: TREE = { @@ -41,6 +43,7 @@ def __init__( super().__init__(context, config) self.area = area + @override def build(self) -> TREE: # Note that cluster IDs may not be in lower case, but series IDs are. series_ids = map(str.lower, self.config.get_renewable_ids(self.area)) @@ -52,6 +55,7 @@ def build(self) -> TREE: class ClusteredRenewableAreaSeries(FolderNode): + @override def build(self) -> TREE: return { area: ClusteredRenewableClusterSeries(self.context, self.config.next_file(area), area) diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/reserves/reserves.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/reserves/reserves.py index 6c558e115c..7b9b89c982 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/reserves/reserves.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/reserves/reserves.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -17,6 +18,7 @@ class InputReserves(FolderNode): + @override def build(self) -> TREE: children: TREE = { a: InputSeriesMatrix( diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/clusters/area/area.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/clusters/area/area.py index e77b3e16a2..375a8d64fa 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/clusters/area/area.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/clusters/area/area.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer @@ -29,6 +30,7 @@ def __init__( super().__init__(context, config) self.area = area + @override def build(self) -> TREE: # Each area has a folder containing a file named "list.ini" # If the area does not have any short term storage cluster, the file is empty. diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/clusters/clusters.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/clusters/clusters.py index ea2c055bed..3723410414 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/clusters/clusters.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/clusters/clusters.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -17,6 +18,7 @@ class InputSTStorageClusters(FolderNode): # Each area has it own folder named after the area id. + @override def build(self) -> TREE: children: TREE = { a: InputSTStorageArea(self.context, self.config.next_file(a), area=a) for a in self.config.area_names() diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/series/area/area.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/series/area/area.py index 3a24f37dc6..3acba900a1 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/series/area/area.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/series/area/area.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer @@ -29,6 +30,7 @@ def __init__( super().__init__(context, config) self.area = area + @override def build(self) -> TREE: children: TREE = { st_storage_id: InputSTStorageAreaStorage(self.context, self.config.next_file(st_storage_id)) diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/series/area/st_storage/st_storage.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/series/area/st_storage/st_storage.py index a531bd4bee..c2a2a4a19a 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/series/area/st_storage/st_storage.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/series/area/st_storage/st_storage.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -17,6 +18,7 @@ class InputSTStorageAreaStorage(FolderNode): + @override def build(self) -> TREE: children: TREE = { "pmax_injection": InputSeriesMatrix( diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/series/series.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/series/series.py index 32aebb3351..3c071bfcd8 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/series/series.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/series/series.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -20,6 +21,7 @@ class InputSTStorageSeries(FolderNode): # For each short-term storage, a time-series matrix is created after the name of the cluster. # This matrix is created inside the folder's area corresponding to the cluster. + @override def build(self) -> TREE: children: TREE = { a: InputSTStorageSeriesArea(self.context, self.config.next_file(a), area=a) diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/st_storage.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/st_storage.py index b70e60381f..4ce2fcbe9c 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/st_storage.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/st_storage/st_storage.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -21,6 +22,8 @@ class InputSTStorage(FolderNode): # Short-term storage objects are introduced in the v8.6 of AntaresSimulator. # This new object simplifies the previously complex modeling of short-term storage such as batteries or STEPs. + + @override def build(self) -> TREE: children: TREE = { "clusters": InputSTStorageClusters(self.context, self.config.next_file("clusters")), diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/cluster/area/area.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/cluster/area/area.py index 6fefc8ca31..e97c5d07ae 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/cluster/area/area.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/cluster/area/area.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer @@ -29,6 +30,7 @@ def __init__( super().__init__(context, config) self.area = area + @override def build(self) -> TREE: children: TREE = { "list": InputThermalClustersAreaList(self.context, self.config.next_file("list.ini"), self.area) diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/cluster/cluster.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/cluster/cluster.py index d30376d3b3..77ec2e705f 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/cluster/cluster.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/cluster/cluster.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -18,6 +19,7 @@ class InputThermalClusters(FolderNode): + @override def build(self) -> TREE: children: TREE = { a: InputThermalClustersArea(self.context, self.config.next_file(a), area=a) diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/prepro/area/area.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/prepro/area/area.py index a804f5dbae..8d4fb248b8 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/prepro/area/area.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/prepro/area/area.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer @@ -29,6 +30,7 @@ def __init__( super().__init__(context, config) self.area = area + @override def build(self) -> TREE: children: TREE = { series_id: InputThermalPreproAreaThermal(self.context, self.config.next_file(series_id)) diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/prepro/area/thermal/thermal.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/prepro/area/thermal/thermal.py index 1beeb19797..dc57ee3c4a 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/prepro/area/thermal/thermal.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/prepro/area/thermal/thermal.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -26,6 +27,7 @@ class InputThermalPreproAreaThermal(FolderNode): - `modulation.txt` (matrix): Modulation matrix (hourly) """ + @override def build(self) -> TREE: children: TREE = { "data": InputSeriesMatrix(self.context, self.config.next_file("data.txt"), freq=MatrixFrequency.DAILY), diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/prepro/prepro.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/prepro/prepro.py index a9e3c19db6..bf75f6ad07 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/prepro/prepro.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/prepro/prepro.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -16,6 +17,7 @@ class InputThermalPrepro(FolderNode): + @override def build(self) -> TREE: children: TREE = { a: InputThermalPreproArea(self.context, self.config.next_file(a), area=a) for a in self.config.area_names() diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/series/area/area.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/series/area/area.py index 33d3bf15b1..c87e73a108 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/series/area/area.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/series/area/area.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer @@ -29,6 +30,7 @@ def __init__( super().__init__(context, config) self.area = area + @override def build(self) -> TREE: # Note that cluster IDs are case-insensitive, but series IDs are in lower case. # For instance, if your cluster ID is "Base", then the series ID will be "base". diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/series/area/thermal/thermal.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/series/area/thermal/thermal.py index 8d765a2c16..b18dde23a7 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/series/area/thermal/thermal.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/series/area/thermal/thermal.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -18,6 +19,7 @@ class InputThermalSeriesAreaThermal(FolderNode): + @override def build(self) -> TREE: children: TREE = { "series": InputSeriesMatrix( diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/series/series.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/series/series.py index 44bb37d4b4..fc639c99d9 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/series/series.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/series/series.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -16,6 +17,7 @@ class InputThermalSeries(FolderNode): + @override def build(self) -> TREE: children: TREE = { a: InputThermalSeriesArea(self.context, self.config.next_file(a), area=a) for a in self.config.area_names() diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/thermal.py b/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/thermal.py index d5ee6cc7b1..dcdcd5dc70 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/thermal.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/input/thermal/thermal.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -19,6 +20,7 @@ class InputThermal(FolderNode): + @override def build(self) -> TREE: children: TREE = { "clusters": InputThermalClusters(self.context, self.config.next_file("clusters")), diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/layers/layers.py b/antarest/study/storage/rawstudy/model/filesystem/root/layers/layers.py index 086475e021..9f40e3fc6d 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/layers/layers.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/layers/layers.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -16,6 +17,7 @@ class Layers(FolderNode): + @override def build(self) -> TREE: children: TREE = {"layers": LayersIni(self.context, self.config.next_file("layers.ini"))} return children diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/output/output.py b/antarest/study/storage/rawstudy/model/filesystem/root/output/output.py index b45cc32c22..f33ed11bb8 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/output/output.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/output/output.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.bucket_node import BucketNode from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode @@ -17,6 +18,7 @@ class Output(FolderNode): + @override def build(self) -> TREE: children: TREE = { str(s.get_file()): OutputSimulation( diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/about/about.py b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/about/about.py index bf2ce89c37..35cffab3db 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/about/about.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/about/about.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -20,6 +21,7 @@ class OutputSimulationAbout(FolderNode): + @override def build(self) -> TREE: children: TREE = { "areas": RawFileNode(self.context, self.config.next_file("areas.txt")), diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/area.py b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/area.py index 0b97fa5735..398ec9bf4e 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/area.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/area.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer @@ -28,6 +29,7 @@ def __init__( super().__init__(context, config) self.area = area + @override def build(self) -> TREE: children: TREE = {} freq: MatrixFrequency diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/areas.py b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/areas.py index 5dcac4d82e..4c1a14cdff 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/areas.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/areas.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer @@ -30,6 +31,7 @@ def __init__( ) -> None: super().__init__(context, config) + @override def build(self) -> TREE: areas = set() sets = set() diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/binding_const.py b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/binding_const.py index 662cef5af6..e956da727b 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/binding_const.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/binding_const.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -19,6 +20,7 @@ class OutputSimulationBindingConstraintItem(FolderNode): + @override def build(self) -> TREE: existing_files = [d.stem.replace("binding-constraints-", "") for d in self.config.path.iterdir()] children: TREE = { diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/link.py b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/link.py index b5f0b15dcb..81922bc9c9 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/link.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/link.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer @@ -30,6 +31,7 @@ def __init__( self.area = area self.link = link + @override def build(self) -> TREE: children: TREE = {} freq: MatrixFrequency diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/links.py b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/links.py index 9d1109ee5f..b7d4ad1d70 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/links.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/links.py @@ -12,6 +12,8 @@ import typing as t +from typing_extensions import override + from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode @@ -33,6 +35,7 @@ def __init__( self.area_from = area_from self.link_names = link_names + @override def build(self) -> TREE: children: TREE = {} for link_name in self.link_names: @@ -51,6 +54,7 @@ def __init__( ): super().__init__(context, config) + @override def build(self) -> TREE: children: TREE = {} links = [d.stem for d in self.config.path.iterdir()] diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/set.py b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/set.py index 4c8a624a5c..11076e977c 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/set.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/set.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer @@ -28,6 +29,7 @@ def __init__( super().__init__(context, config) self.set = set + @override def build(self) -> TREE: children: TREE = {} freq: MatrixFrequency diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/utils.py b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/utils.py index f8f940fd28..d5d6b6495c 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/utils.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/common/utils.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -34,6 +35,7 @@ class OutputSimulationModeCommon(FolderNode): + @override def build(self) -> TREE: if not self.config.output_path: return {} diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/economy.py b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/economy.py index a7da7fa642..c39642589e 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/economy.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/economy.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig, Simulation from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer @@ -32,6 +33,7 @@ def __init__( super().__init__(context, config) self.simulation = simulation + @override def build(self) -> TREE: children: TREE = {} if self.simulation.by_year: diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/mcall/grid.py b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/mcall/grid.py index 943de12ef2..374b4fc6a1 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/mcall/grid.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/mcall/grid.py @@ -13,6 +13,7 @@ import typing as t import pandas as pd +from typing_extensions import override from antarest.core.exceptions import MustNotModifyOutputException from antarest.core.model import JSON @@ -24,6 +25,7 @@ class OutputSimulationModeMcAllGrid(FolderNode): + @override def build(self) -> TREE: files = [d.stem for d in self.config.path.iterdir()] children: TREE = {} @@ -37,6 +39,7 @@ class OutputSynthesis(LazyNode[JSON, bytes, bytes]): def __init__(self, context: ContextServer, config: FileStudyTreeConfig): super().__init__(context, config) + @override def get_lazy_content( self, url: t.Optional[t.List[str]] = None, @@ -45,6 +48,7 @@ def get_lazy_content( ) -> str: return f"matrix://{self.config.path.name}" # prefix used by the front to parse the back-end response + @override def load( self, url: t.Optional[t.List[str]] = None, @@ -59,9 +63,11 @@ def load( del output["index"] return t.cast(JSON, output) + @override def dump(self, data: bytes, url: t.Optional[t.List[str]] = None) -> None: raise MustNotModifyOutputException(self.config.path.name) + @override def check_errors(self, data: str, url: t.Optional[t.List[str]] = None, raising: bool = False) -> t.List[str]: if not self.config.path.exists(): msg = f"{self.config.path} not exist" @@ -70,9 +76,11 @@ def check_errors(self, data: str, url: t.Optional[t.List[str]] = None, raising: return [msg] return [] + @override def normalize(self) -> None: pass # shouldn't be normalized as it's an output file + @override def denormalize(self) -> None: pass # shouldn't be denormalized as it's an output file @@ -81,6 +89,7 @@ class DigestSynthesis(OutputSynthesis): def __init__(self, context: ContextServer, config: FileStudyTreeConfig): super().__init__(context, config) + @override def load( self, url: t.Optional[t.List[str]] = None, diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/mcind/mcind.py b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/mcind/mcind.py index e6cac924ea..ea1c5d2c58 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/mcind/mcind.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/mode/mcind/mcind.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig, Simulation from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer @@ -29,6 +30,7 @@ def __init__( super().__init__(context, config) self.simulation = simulation + @override def build(self) -> TREE: children: TREE = { f"{scn:05d}": OutputSimulationModeCommon(self.context, self.config.next_file(f"{scn:05d}")) diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/simulation.py b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/simulation.py index 129b05f147..12ecb930d9 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/simulation.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/simulation.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig, Simulation from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer @@ -42,6 +43,7 @@ def __init__( super().__init__(context, config) self.simulation = simulation + @override def build(self) -> TREE: children: TREE = { "about-the-study": OutputSimulationAbout(self.context, self.config.next_file("about-the-study")), diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/ts_generator/ts_generator.py b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/ts_generator/ts_generator.py index 73e0ae6ed0..b99307d01d 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/ts_generator/ts_generator.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/ts_generator/ts_generator.py @@ -12,6 +12,8 @@ from typing import Any, Callable +from typing_extensions import override + from antarest.study.storage.rawstudy.model.filesystem.common.area_matrix_list import ( AreaMatrixList, AreaMultipleMatrixList, @@ -26,6 +28,7 @@ class OutputSimulationTsGeneratorSimpleMatrixList(FolderNode): + @override def build(self) -> TREE: children: TREE = { "mc-0": AreaMatrixList(self.context, self.config.next_file("mc-0")), @@ -54,6 +57,7 @@ def __init__( super().__init__(context, config) self.klass = klass + @override def build(self) -> TREE: children: TREE = { "mc-0": AreaMultipleMatrixList( @@ -67,6 +71,7 @@ def build(self) -> TREE: class OutputSimulationTsGenerator(FolderNode): + @override def build(self) -> TREE: children: TREE = {} for output_type in ["load", "solar", "wind"]: diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/ts_numbers/ts_numbers.py b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/ts_numbers/ts_numbers.py index 2d5f9d729b..4c84635f4a 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/ts_numbers/ts_numbers.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/ts_numbers/ts_numbers.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.common.area_matrix_list import ( AreaMatrixList, @@ -68,6 +69,7 @@ class OutputSimulationTsNumbers(FolderNode): └── turbinage.txt """ + @override def build(self) -> TREE: children: TREE = {} for output_type in ["hydro", "load", "solar", "wind"]: diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/ts_numbers/ts_numbers_data.py b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/ts_numbers/ts_numbers_data.py index 670a6526ca..45512a2af2 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/ts_numbers/ts_numbers_data.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/ts_numbers/ts_numbers_data.py @@ -13,6 +13,8 @@ import logging from typing import List, Optional, Union +from typing_extensions import override + from antarest.core.model import JSON from antarest.study.storage.rawstudy.model.filesystem.lazy_node import LazyNode @@ -20,6 +22,7 @@ class TsNumbersVector(LazyNode[List[int], List[int], JSON]): + @override def load( self, url: Optional[List[str]] = None, @@ -44,6 +47,7 @@ def load( logger.warning(f"Missing file {self.config.path}") return [] + @override def dump( self, data: Union[str, bytes, List[int]], @@ -55,6 +59,7 @@ def dump( for d in data: fh.write(f"{d}\n") + @override def check_errors( self, data: JSON, @@ -63,10 +68,12 @@ def check_errors( ) -> List[str]: return [] + @override def normalize(self) -> None: # this is not normalizable pass + @override def denormalize(self) -> None: # this is not normalizable pass diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/xpansion/sensitivity.py b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/xpansion/sensitivity.py index 6e3e5a4ee5..8383dbbcad 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/xpansion/sensitivity.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/xpansion/sensitivity.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -17,6 +18,7 @@ class Sensitivity(FolderNode): + @override def build(self) -> TREE: return { "out": JsonFileNode(self.context, self.config.next_file("sensitivity_out.json")), diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/xpansion/xpansion.py b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/xpansion/xpansion.py index 9635e8be82..2815bac089 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/xpansion/xpansion.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/output/simulation/xpansion/xpansion.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -16,6 +17,7 @@ class Xpansion(FolderNode): + @override def build(self) -> TREE: return { "out": JsonFileNode(self.context, self.config.next_file("out.json")), diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/settings/resources/resources.py b/antarest/study/storage/rawstudy/model/filesystem/root/settings/resources/resources.py index 347d90f609..4ff236f7be 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/settings/resources/resources.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/settings/resources/resources.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -16,6 +17,7 @@ class Resources(FolderNode): + @override def build(self) -> TREE: children: TREE = {"study": RawFileNode(self.context, self.config.next_file("study.ico"))} return children diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/settings/scenariobuilder.py b/antarest/study/storage/rawstudy/model/filesystem/root/settings/scenariobuilder.py index bf0501d531..f1960d9984 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/settings/scenariobuilder.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/settings/scenariobuilder.py @@ -14,6 +14,7 @@ import typing as t import typing_extensions as te +from typing_extensions import override from antarest.study.model import ( STUDY_VERSION_8, @@ -124,6 +125,7 @@ def _populate_hydro_generation_power_rules(self, rules: _Rules) -> None: for area_id in self.config.areas: rules[f"hgp,{area_id},0"] = _TSNumber + @override def _get_filtering_kwargs(self, url: t.List[str]) -> t.Dict[str, str]: # If the URL contains 2 elements, we can filter the options based on the generator type. if len(url) == 2: diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/settings/settings.py b/antarest/study/storage/rawstudy/model/filesystem/root/settings/settings.py index 61d32f6012..f63b7cb69d 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/settings/settings.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/settings/settings.py @@ -9,6 +9,7 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE @@ -20,6 +21,7 @@ class Settings(FolderNode): + @override def build(self) -> TREE: children: TREE = { "resources": Resources(self.context, self.config.next_file("resources")), diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/settings/simulations/simulations.py b/antarest/study/storage/rawstudy/model/filesystem/root/settings/simulations/simulations.py index 5a8e832e11..64e074623e 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/settings/simulations/simulations.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/settings/simulations/simulations.py @@ -9,12 +9,14 @@ # SPDX-License-Identifier: MPL-2.0 # # This file is part of the Antares project. +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE class SettingsSimulations(FolderNode): + @override def build(self) -> TREE: children: TREE = {} return children diff --git a/antarest/study/storage/rawstudy/model/filesystem/root/user/expansion/sensitivity.py b/antarest/study/storage/rawstudy/model/filesystem/root/user/expansion/sensitivity.py index da42e73c64..2a2ea7c9c3 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/root/user/expansion/sensitivity.py +++ b/antarest/study/storage/rawstudy/model/filesystem/root/user/expansion/sensitivity.py @@ -12,12 +12,15 @@ from typing import List +from typing_extensions import override + from antarest.study.storage.rawstudy.model.filesystem.folder_node import FolderNode from antarest.study.storage.rawstudy.model.filesystem.inode import TREE from antarest.study.storage.rawstudy.model.filesystem.json_file_node import JsonFileNode class SensitivityConfig(FolderNode): + @override def build(self) -> TREE: types = {"epsilon": float, "capex": bool, "projection": List[str]} return { diff --git a/antarest/study/storage/rawstudy/raw_study_service.py b/antarest/study/storage/rawstudy/raw_study_service.py index e90b9b33ed..11572d6710 100644 --- a/antarest/study/storage/rawstudy/raw_study_service.py +++ b/antarest/study/storage/rawstudy/raw_study_service.py @@ -20,6 +20,7 @@ from uuid import uuid4 from antares.study.version import StudyVersion +from typing_extensions import override from antarest.core.config import Config from antarest.core.exceptions import StudyDeletionNotAllowed @@ -142,6 +143,7 @@ def update_name_and_version_from_raw_meta(self, metadata: RawStudy) -> bool: ) return False + @override def exists(self, study: RawStudy) -> bool: """ Check if the study exists in the filesystem. @@ -158,6 +160,7 @@ def exists(self, study: RawStudy) -> bool: path = self.get_study_path(study) return path.joinpath("study.antares").is_file() + @override def get_raw( self, metadata: RawStudy, @@ -177,12 +180,14 @@ def get_raw( study_path = self.get_study_path(metadata) return self.study_factory.create_from_fs(study_path, metadata.id, output_dir, use_cache=use_cache) + @override def get_synthesis(self, metadata: RawStudy, params: t.Optional[RequestParameters] = None) -> FileStudyTreeConfigDTO: self._check_study_exists(metadata) study_path = self.get_study_path(metadata) study = self.study_factory.create_from_fs(study_path, metadata.id) return FileStudyTreeConfigDTO.from_build_config(study.config) + @override def create(self, metadata: RawStudy) -> RawStudy: """ Create a new empty study based on the given metadata. @@ -216,6 +221,7 @@ def create(self, metadata: RawStudy) -> RawStudy: return metadata + @override def copy( self, src_meta: RawStudy, @@ -273,6 +279,7 @@ def copy( return dest_study + @override def delete(self, metadata: RawStudy) -> None: """ Delete study @@ -290,6 +297,7 @@ def delete(self, metadata: RawStudy) -> None: else: raise StudyDeletionNotAllowed(metadata.id) + @override def delete_output(self, metadata: RawStudy, output_name: str) -> None: """ Delete output folder @@ -338,6 +346,7 @@ def import_study(self, metadata: RawStudy, stream: t.BinaryIO) -> Study: metadata.path = str(study_path) return metadata + @override def export_study_flat( self, metadata: RawStudy, @@ -379,6 +388,7 @@ def check_errors( study = self.study_factory.create_from_fs(path, metadata.id) return study.tree.check_errors(study.tree.get()) + @override def set_reference_output(self, study: RawStudy, output_id: str, status: bool) -> None: self.patch_service.set_reference_output(study, output_id, status) remove_from_cache(self.cache, study.id) @@ -422,6 +432,7 @@ def find_archive_path(self, study: RawStudy) -> Path: return path raise FileNotFoundError(f"Study {study.id} archiving process is corrupted (no archive file found).") + @override def get_study_path(self, metadata: Study) -> Path: """ Get study path @@ -435,6 +446,7 @@ def get_study_path(self, metadata: Study) -> Path: return self.find_archive_path(metadata) return Path(metadata.path) + @override def initialize_additional_data(self, raw_study: RawStudy) -> bool: try: study = self.study_factory.create_from_fs( diff --git a/antarest/study/storage/rawstudy/watcher.py b/antarest/study/storage/rawstudy/watcher.py index 3166bbe40f..75541c009b 100644 --- a/antarest/study/storage/rawstudy/watcher.py +++ b/antarest/study/storage/rawstudy/watcher.py @@ -19,6 +19,7 @@ from typing import List, Optional from filelock import FileLock +from typing_extensions import override from antarest.core.config import Config from antarest.core.interfaces.service import IService @@ -70,6 +71,7 @@ def __init__( self.should_stop = False self.allowed_to_start = not config.storage.watcher_lock or Watcher._get_lock(config.storage.watcher_lock_delay) + @override def start(self, threaded: bool = True) -> None: self.should_stop = False if self.allowed_to_start: @@ -97,6 +99,7 @@ def _get_lock(lock_delay: int) -> bool: logger.info("Watcher doesn't get lock") return False + @override def _loop(self) -> None: try: logger.info( diff --git a/antarest/study/storage/variantstudy/business/command_extractor.py b/antarest/study/storage/variantstudy/business/command_extractor.py index 0f034f8a37..dbac37aff9 100644 --- a/antarest/study/storage/variantstudy/business/command_extractor.py +++ b/antarest/study/storage/variantstudy/business/command_extractor.py @@ -15,6 +15,7 @@ import typing as t import numpy as np +from typing_extensions import override from antarest.core.model import JSON from antarest.core.utils.utils import StopWatch @@ -67,6 +68,7 @@ def __init__(self, matrix_service: ISimpleMatrixService, patch_service: PatchSer patch_service=self.patch_service, ) + @override def extract_area(self, study: FileStudy, area_id: str) -> t.Tuple[t.List[ICommand], t.List[ICommand]]: stopwatch = StopWatch() study_tree = study.tree @@ -147,6 +149,7 @@ def extract_area(self, study: FileStudy, area_id: str) -> t.Tuple[t.List[IComman stopwatch.log_elapsed(lambda x: logger.info(f"Hydro command extraction done in {x}s")) return study_commands, links_commands + @override def extract_link( self, study: FileStudy, @@ -249,12 +252,15 @@ def _extract_cluster(self, study: FileStudy, area_id: str, cluster_id: str, rene ) return study_commands + @override def extract_cluster(self, study: FileStudy, area_id: str, thermal_id: str) -> t.List[ICommand]: return self._extract_cluster(study, area_id, thermal_id, False) + @override def extract_renewables_cluster(self, study: FileStudy, area_id: str, renewables_id: str) -> t.List[ICommand]: return self._extract_cluster(study, area_id, renewables_id, True) + @override def extract_hydro(self, study: FileStudy, area_id: str) -> t.List[ICommand]: study_tree = study.tree commands = [ @@ -316,6 +322,7 @@ def extract_hydro(self, study: FileStudy, area_id: str) -> t.List[ICommand]: return commands + @override def extract_district(self, study: FileStudy, district_id: str) -> t.List[ICommand]: study_commands: t.List[ICommand] = [] study_config = study.config @@ -337,6 +344,7 @@ def extract_district(self, study: FileStudy, district_id: str) -> t.List[IComman ) return study_commands + @override def extract_comments(self, study: FileStudy) -> t.List[ICommand]: study_tree = study.tree content = t.cast(bytes, study_tree.get(["settings", "comments"])) @@ -347,6 +355,7 @@ def extract_comments(self, study: FileStudy) -> t.List[ICommand]: ) ] + @override def extract_binding_constraint( self, study: FileStudy, @@ -400,6 +409,7 @@ def extract_binding_constraint( return [create_cmd] + @override def generate_update_config(self, study_tree: FileStudyTree, url: t.List[str]) -> ICommand: data = study_tree.get(url) return UpdateConfig( @@ -409,6 +419,7 @@ def generate_update_config(self, study_tree: FileStudyTree, url: t.List[str]) -> study_version=study_tree.config.version, ) + @override def generate_update_raw_file(self, study_tree: FileStudyTree, url: t.List[str]) -> ICommand: data = study_tree.get(url) return UpdateRawFile( @@ -418,6 +429,7 @@ def generate_update_raw_file(self, study_tree: FileStudyTree, url: t.List[str]) study_version=study_tree.config.version, ) + @override def generate_update_comments( self, study_tree: FileStudyTree, @@ -443,6 +455,7 @@ def generate_update_playlist( study_version=study_tree.config.version, ) + @override def generate_replace_matrix( self, study_tree: FileStudyTree, diff --git a/antarest/study/storage/variantstudy/model/command/create_area.py b/antarest/study/storage/variantstudy/model/command/create_area.py index 1eefb00d1a..d05cc931d4 100644 --- a/antarest/study/storage/variantstudy/model/command/create_area.py +++ b/antarest/study/storage/variantstudy/model/command/create_area.py @@ -13,6 +13,7 @@ import typing as t from pydantic import Field +from typing_extensions import override from antarest.core.model import JSON from antarest.study.model import STUDY_VERSION_6_5, STUDY_VERSION_8_1, STUDY_VERSION_8_3, STUDY_VERSION_8_6 @@ -74,6 +75,7 @@ class CreateArea(ICommand): # or if we don't want to support this feature. metadata: t.Dict[str, str] = Field(default_factory=dict, description="Area metadata: country and tag list") + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> t.Tuple[CommandOutput, t.Dict[str, t.Any]]: if self.command_context.generator_matrix_constants is None: raise ValueError() @@ -102,6 +104,7 @@ def _apply_config(self, study_data: FileStudyTreeConfig) -> t.Tuple[CommandOutpu {"area_id": area_id}, ) + @override def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = None) -> CommandOutput: config = study_data.config @@ -287,21 +290,26 @@ def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = return output + @override def to_dto(self) -> CommandDTO: return CommandDTO( action=CommandName.CREATE_AREA.value, args={"area_name": self.area_name}, study_version=self.study_version ) + @override def match_signature(self) -> str: return str(self.command_name.value + MATCH_SIGNATURE_SEPARATOR + self.area_name) + @override def match(self, other: ICommand, equal: bool = False) -> bool: if not isinstance(other, CreateArea): return False return self.area_name == other.area_name + @override def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: return [] + @override def get_inner_matrices(self) -> t.List[str]: return [] diff --git a/antarest/study/storage/variantstudy/model/command/create_binding_constraint.py b/antarest/study/storage/variantstudy/model/command/create_binding_constraint.py index 93a65ff25a..b41fb2dd93 100644 --- a/antarest/study/storage/variantstudy/model/command/create_binding_constraint.py +++ b/antarest/study/storage/variantstudy/model/command/create_binding_constraint.py @@ -17,6 +17,7 @@ import numpy as np from antares.study.version import StudyVersion from pydantic import Field, field_validator, model_validator +from typing_extensions import override from antarest.core.model import LowerCaseStr from antarest.core.serialization import AntaresBaseModel @@ -221,6 +222,7 @@ class AbstractBindingConstraintCommand(OptionalProperties, BindingConstraintMatr coeffs: t.Optional[t.Dict[str, t.List[float]]] = None + @override def to_dto(self) -> CommandDTO: json_command = self.model_dump(mode="json", exclude={"command_context"}) args = {} @@ -248,6 +250,7 @@ def to_dto(self) -> CommandDTO: action=self.command_name.value, args=args, version=self.version, study_version=self.study_version ) + @override def get_inner_matrices(self) -> t.List[str]: matrix_service = self.command_context.matrix_service return [ @@ -416,6 +419,7 @@ class CreateBindingConstraint(AbstractBindingConstraintCommand): # Properties of the `CREATE_BINDING_CONSTRAINT` command: name: str + @override def _apply_config(self, study_data_config: FileStudyTreeConfig) -> t.Tuple[CommandOutput, t.Dict[str, t.Any]]: bd_id = transform_name_to_id(self.name) group = self.group or DEFAULT_GROUP @@ -431,6 +435,7 @@ def _apply_config(self, study_data_config: FileStudyTreeConfig) -> t.Tuple[Comma ) return CommandOutput(status=True), {} + @override def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = None) -> CommandOutput: binding_constraints = study_data.tree.get(["input", "bindingconstraints", "bindingconstraints"]) new_key = str(len(binding_constraints)) @@ -449,14 +454,17 @@ def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = ) return super().apply_binding_constraint(study_data, binding_constraints, new_key, bd_id) + @override def to_dto(self) -> CommandDTO: dto = super().to_dto() dto.args["name"] = self.name # type: ignore return dto + @override def match_signature(self) -> str: return str(self.command_name.value + MATCH_SIGNATURE_SEPARATOR + self.name) + @override def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: from antarest.study.storage.variantstudy.model.command.update_binding_constraint import UpdateBindingConstraint @@ -492,6 +500,7 @@ def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: return [UpdateBindingConstraint.model_validate(args)] + @override def match(self, other: "ICommand", equal: bool = False) -> bool: if not isinstance(other, self.__class__): return False diff --git a/antarest/study/storage/variantstudy/model/command/create_cluster.py b/antarest/study/storage/variantstudy/model/command/create_cluster.py index 779d85ee3c..de4eb70ffc 100644 --- a/antarest/study/storage/variantstudy/model/command/create_cluster.py +++ b/antarest/study/storage/variantstudy/model/command/create_cluster.py @@ -13,6 +13,7 @@ import typing as t from pydantic import Field, model_validator +from typing_extensions import override from antarest.core.model import JSON, LowerCaseStr from antarest.core.utils.utils import assert_this @@ -73,6 +74,7 @@ def validate_model(cls, values: t.Dict[str, t.Any]) -> t.Dict[str, t.Any]: return values + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> t.Tuple[CommandOutput, t.Dict[str, t.Any]]: # Search the Area in the configuration if self.area_id not in study_data.areas: @@ -107,6 +109,7 @@ def _apply_config(self, study_data: FileStudyTreeConfig) -> t.Tuple[CommandOutpu {"cluster_id": cluster.id}, ) + @override def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = None) -> CommandOutput: output, data = self._apply_config(study_data.config) if not output.status: @@ -144,6 +147,7 @@ def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = return output + @override def to_dto(self) -> CommandDTO: return CommandDTO( action=self.command_name.value, @@ -157,6 +161,7 @@ def to_dto(self) -> CommandDTO: study_version=self.study_version, ) + @override def match_signature(self) -> str: return str( self.command_name.value @@ -166,6 +171,7 @@ def match_signature(self) -> str: + self.cluster_name ) + @override def match(self, other: ICommand, equal: bool = False) -> bool: if not isinstance(other, CreateCluster): return False @@ -181,6 +187,7 @@ def match(self, other: ICommand, equal: bool = False) -> bool: and self.modulation == other.modulation ) + @override def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: other = t.cast(CreateCluster, other) from antarest.study.storage.variantstudy.model.command.replace_matrix import ReplaceMatrix @@ -220,6 +227,7 @@ def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: ) return commands + @override def get_inner_matrices(self) -> t.List[str]: matrices: t.List[str] = [] if self.prepro: diff --git a/antarest/study/storage/variantstudy/model/command/create_district.py b/antarest/study/storage/variantstudy/model/command/create_district.py index e8ed7f69d9..668ace46f1 100644 --- a/antarest/study/storage/variantstudy/model/command/create_district.py +++ b/antarest/study/storage/variantstudy/model/command/create_district.py @@ -14,6 +14,7 @@ from typing import Any, Dict, List, Optional, Tuple, cast from pydantic import field_validator +from typing_extensions import override from antarest.core.model import LowerCaseStr from antarest.study.storage.rawstudy.model.filesystem.config.field_validators import transform_name_to_id @@ -50,6 +51,7 @@ class CreateDistrict(ICommand): output: bool = True comments: str = "" + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> Tuple[CommandOutput, Dict[str, Any]]: district_id = transform_name_to_id(self.name) if district_id in study_data.sets: @@ -75,6 +77,7 @@ def _apply_config(self, study_data: FileStudyTreeConfig) -> Tuple[CommandOutput, "item_key": item_key, } + @override def _apply(self, study_data: FileStudy, listener: Optional[ICommandListener] = None) -> CommandOutput: output, data = self._apply_config(study_data.config) if not output.status: @@ -94,6 +97,7 @@ def _apply(self, study_data: FileStudy, listener: Optional[ICommandListener] = N return output + @override def to_dto(self) -> CommandDTO: return CommandDTO( action=CommandName.CREATE_DISTRICT.value, @@ -107,9 +111,11 @@ def to_dto(self) -> CommandDTO: study_version=self.study_version, ) + @override def match_signature(self) -> str: return str(self.command_name.value + MATCH_SIGNATURE_SEPARATOR + self.name) + @override def match(self, other: ICommand, equal: bool = False) -> bool: if not isinstance(other, CreateDistrict): return False @@ -124,6 +130,7 @@ def match(self, other: ICommand, equal: bool = False) -> bool: and self.comments == other.comments ) + @override def _create_diff(self, other: "ICommand") -> List["ICommand"]: other = cast(CreateDistrict, other) district_id = transform_name_to_id(self.name) @@ -147,5 +154,6 @@ def _create_diff(self, other: "ICommand") -> List["ICommand"]: ) ] + @override def get_inner_matrices(self) -> List[str]: return [] diff --git a/antarest/study/storage/variantstudy/model/command/create_link.py b/antarest/study/storage/variantstudy/model/command/create_link.py index 1e46c20bf1..134491ce8c 100644 --- a/antarest/study/storage/variantstudy/model/command/create_link.py +++ b/antarest/study/storage/variantstudy/model/command/create_link.py @@ -14,6 +14,7 @@ from antares.study.version import StudyVersion from pydantic import ValidationInfo, field_validator, model_validator +from typing_extensions import override from antarest.core.exceptions import LinkValidationError from antarest.core.utils.utils import assert_this @@ -65,6 +66,7 @@ def validate_areas(self) -> "AbstractLinkCommand": return self + @override def to_dto(self) -> CommandDTO: args = { "area1": self.area1, @@ -76,6 +78,7 @@ def to_dto(self) -> CommandDTO: args[attr] = strip_matrix_protocol(value) return CommandDTO(action=self.command_name.value, args=args, study_version=self.study_version) + @override def match(self, other: ICommand, equal: bool = False) -> bool: if not isinstance(other, self.__class__): return False @@ -90,11 +93,13 @@ def match(self, other: ICommand, equal: bool = False) -> bool: and self.indirect == other.indirect ) + @override def match_signature(self) -> str: return str( self.command_name.value + MATCH_SIGNATURE_SEPARATOR + self.area1 + MATCH_SIGNATURE_SEPARATOR + self.area2 ) + @override def _create_diff(self, other: "ICommand") -> List["ICommand"]: other = cast(AbstractLinkCommand, other) @@ -123,6 +128,7 @@ def _create_diff(self, other: "ICommand") -> List["ICommand"]: ) return commands + @override def get_inner_matrices(self) -> List[str]: list_matrices = [] for attr in MATRIX_ATTRIBUTES: @@ -200,6 +206,7 @@ def _create_link_in_config(self, area_from: str, area_to: str, study_data: FileS ], ) + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> Tuple[CommandOutput, Dict[str, Any]]: if self.area1 not in study_data.areas: return ( @@ -254,6 +261,7 @@ def _apply_config(self, study_data: FileStudyTreeConfig) -> Tuple[CommandOutput, {"area_from": area_from, "area_to": area_to}, ) + @override def _apply(self, study_data: FileStudy, listener: Optional[ICommandListener] = None) -> CommandOutput: version = study_data.config.version output, data = self._apply_config(study_data.config) @@ -283,17 +291,22 @@ def _apply(self, study_data: FileStudy, listener: Optional[ICommandListener] = N return output + @override def to_dto(self) -> CommandDTO: return super().to_dto() + @override def match_signature(self) -> str: return super().match_signature() + @override def match(self, other: ICommand, equal: bool = False) -> bool: return super().match(other, equal) + @override def _create_diff(self, other: "ICommand") -> List["ICommand"]: return super()._create_diff(other) + @override def get_inner_matrices(self) -> List[str]: return super().get_inner_matrices() diff --git a/antarest/study/storage/variantstudy/model/command/create_renewables_cluster.py b/antarest/study/storage/variantstudy/model/command/create_renewables_cluster.py index 6c0c6381c8..057791ca10 100644 --- a/antarest/study/storage/variantstudy/model/command/create_renewables_cluster.py +++ b/antarest/study/storage/variantstudy/model/command/create_renewables_cluster.py @@ -13,6 +13,7 @@ import typing as t from pydantic import model_validator +from typing_extensions import override from antarest.core.model import JSON, LowerCaseStr from antarest.study.storage.rawstudy.model.filesystem.config.model import Area, EnrModelling, FileStudyTreeConfig @@ -53,6 +54,7 @@ def validate_model(cls, values: t.Dict[str, t.Any]) -> t.Dict[str, t.Any]: values["parameters"] = create_renewable_properties(values["study_version"], **args) return values + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> t.Tuple[CommandOutput, t.Dict[str, t.Any]]: if EnrModelling(study_data.enr_modelling) != EnrModelling.CLUSTERS: # Since version 8.1 of the solver, we can use renewable clusters @@ -100,6 +102,7 @@ def _apply_config(self, study_data: FileStudyTreeConfig) -> t.Tuple[CommandOutpu {"cluster_id": cluster.id}, ) + @override def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = None) -> CommandOutput: output, data = self._apply_config(study_data.config) if not output.status: @@ -127,6 +130,7 @@ def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = return output + @override def to_dto(self) -> CommandDTO: return CommandDTO( action=self.command_name.value, @@ -138,6 +142,7 @@ def to_dto(self) -> CommandDTO: study_version=self.study_version, ) + @override def match_signature(self) -> str: return str( self.command_name.value @@ -147,6 +152,7 @@ def match_signature(self) -> str: + self.cluster_name ) + @override def match(self, other: ICommand, equal: bool = False) -> bool: if not isinstance(other, CreateRenewablesCluster): return False @@ -157,6 +163,7 @@ def match(self, other: ICommand, equal: bool = False) -> bool: other_params = other.parameters.model_dump(mode="json", by_alias=True) return simple_match and self_params == other_params + @override def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: other = t.cast(CreateRenewablesCluster, other) from antarest.study.storage.variantstudy.model.command.update_config import UpdateConfig @@ -175,5 +182,6 @@ def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: ) return commands + @override def get_inner_matrices(self) -> t.List[str]: return [] diff --git a/antarest/study/storage/variantstudy/model/command/create_st_storage.py b/antarest/study/storage/variantstudy/model/command/create_st_storage.py index 8d15d5c625..42a3e66253 100644 --- a/antarest/study/storage/variantstudy/model/command/create_st_storage.py +++ b/antarest/study/storage/variantstudy/model/command/create_st_storage.py @@ -14,6 +14,7 @@ import numpy as np from pydantic import Field, ValidationInfo, model_validator +from typing_extensions import override from antarest.core.model import JSON, LowerCaseStr from antarest.matrixstore.model import MatrixData @@ -170,6 +171,7 @@ def validate_matrices(cls, values: t.Union[t.Dict[str, t.Any], ValidationInfo]) new_values[field] = cls.validate_field(new_values.get(field, None), new_values, field) return new_values + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> t.Tuple[CommandOutput, t.Dict[str, t.Any]]: """ Applies configuration changes to the study data: add the short-term storage in the storages list. @@ -229,6 +231,7 @@ def _apply_config(self, study_data: FileStudyTreeConfig) -> t.Tuple[CommandOutpu {"storage_id": storage_id}, ) + @override def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = None) -> CommandOutput: """ Applies the study data to update storage configurations and saves the changes. @@ -263,6 +266,7 @@ def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = return output + @override def to_dto(self) -> CommandDTO: """ Converts the current object to a Data Transfer Object (DTO) @@ -281,6 +285,7 @@ def to_dto(self) -> CommandDTO: study_version=self.study_version, ) + @override def match_signature(self) -> str: """Returns the command signature.""" return str( @@ -291,6 +296,7 @@ def match_signature(self) -> str: + self.storage_id ) + @override def match(self, other: "ICommand", equal: bool = False) -> bool: """ Checks if the current instance matches another `ICommand` object. @@ -310,6 +316,7 @@ def match(self, other: "ICommand", equal: bool = False) -> bool: else: return self.area_id == other.area_id and self.storage_id == other.storage_id + @override def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: """ Creates a list of commands representing the differences between @@ -350,6 +357,7 @@ def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: ) return commands + @override def get_inner_matrices(self) -> t.List[str]: """ Retrieves the list of matrix IDs. diff --git a/antarest/study/storage/variantstudy/model/command/create_user_resource.py b/antarest/study/storage/variantstudy/model/command/create_user_resource.py index a62fd1f8b0..0afb321e54 100644 --- a/antarest/study/storage/variantstudy/model/command/create_user_resource.py +++ b/antarest/study/storage/variantstudy/model/command/create_user_resource.py @@ -12,6 +12,8 @@ import typing as t from enum import StrEnum +from typing_extensions import override + from antarest.core.exceptions import ChildNotFoundError from antarest.core.model import JSON from antarest.core.serialization import AntaresBaseModel @@ -50,9 +52,11 @@ class CreateUserResource(ICommand): data: CreateUserResourceData + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> t.Tuple[CommandOutput, t.Dict[str, t.Any]]: return CommandOutput(status=True, message="ok"), {} + @override def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = None) -> CommandOutput: url = [item for item in self.data.path.split("/") if item] study_tree = study_data.tree @@ -74,6 +78,7 @@ def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = return CommandOutput(status=False, message=f"the given resource already exists: {self.data.path}") return CommandOutput(status=True, message="ok") + @override def to_dto(self) -> CommandDTO: return CommandDTO( action=self.command_name.value, @@ -81,6 +86,7 @@ def to_dto(self) -> CommandDTO: study_version=self.study_version, ) + @override def match_signature(self) -> str: return str( self.command_name.value @@ -90,13 +96,16 @@ def match_signature(self) -> str: + self.data.resource_type.value ) + @override def match(self, other: ICommand, equal: bool = False) -> bool: if not isinstance(other, CreateUserResource): return False return self.data.path == other.data.path and self.data.resource_type == other.data.resource_type + @override def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: return [other] + @override def get_inner_matrices(self) -> t.List[str]: return [] diff --git a/antarest/study/storage/variantstudy/model/command/generate_thermal_cluster_timeseries.py b/antarest/study/storage/variantstudy/model/command/generate_thermal_cluster_timeseries.py index e6e4bf17a1..9f8631b608 100644 --- a/antarest/study/storage/variantstudy/model/command/generate_thermal_cluster_timeseries.py +++ b/antarest/study/storage/variantstudy/model/command/generate_thermal_cluster_timeseries.py @@ -21,6 +21,7 @@ from antares.tsgen.duration_generator import ProbabilityLaw from antares.tsgen.random_generator import MersenneTwisterRNG from antares.tsgen.ts_generator import OutageGenerationParameters, ThermalCluster, TimeseriesGenerator +from typing_extensions import override from antarest.study.storage.rawstudy.model.filesystem.config.model import Area, FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.config.thermal import LocalTSGenerationBehavior @@ -47,9 +48,11 @@ class GenerateThermalClusterTimeSeries(ICommand): command_name: CommandName = CommandName.GENERATE_THERMAL_CLUSTER_TIMESERIES version: int = 1 + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> OutputTuple: return CommandOutput(status=True, message="Nothing to do"), {} + @override def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = None) -> CommandOutput: study_path = study_data.config.study_path with tempfile.TemporaryDirectory(suffix=TS_GEN_SUFFIX, prefix=TS_GEN_PREFIX, dir=study_path.parent) as path: @@ -141,22 +144,27 @@ def _build_timeseries( e.args = (f"Area {area_id}, cluster {thermal.id.lower()}: " + e.args[0],) raise + @override def to_dto(self) -> CommandDTO: return CommandDTO(action=self.command_name.value, args={}, study_version=self.study_version) + @override def match_signature(self) -> str: return str(self.command_name.value) + @override def match(self, other: "ICommand", equal: bool = False) -> bool: # Only used inside the cli app that no one uses I believe. if not isinstance(other, GenerateThermalClusterTimeSeries): return False return True + @override def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: # Only used inside the cli app that no one uses I believe. raise NotImplementedError() + @override def get_inner_matrices(self) -> t.List[str]: # This is used to get used matrices and not remove them inside the garbage collector loop. return [] diff --git a/antarest/study/storage/variantstudy/model/command/remove_area.py b/antarest/study/storage/variantstudy/model/command/remove_area.py index af536c37df..4d7c33e922 100644 --- a/antarest/study/storage/variantstudy/model/command/remove_area.py +++ b/antarest/study/storage/variantstudy/model/command/remove_area.py @@ -14,6 +14,8 @@ import logging import typing as t +from typing_extensions import override + from antarest.core.exceptions import ChildNotFoundError from antarest.core.model import JSON from antarest.study.model import ( @@ -64,6 +66,7 @@ def _remove_area_from_sets_in_config(self, study_data_config: FileStudyTreeConfi set_.areas.remove(self.id) study_data_config.sets[id_] = set_ + @override def _apply_config(self, study_data_config: FileStudyTreeConfig) -> t.Tuple[CommandOutput, t.Dict[str, t.Any]]: del study_data_config.areas[self.id] @@ -223,6 +226,7 @@ def _remove_area_from_scenario_builder(self, study_data: FileStudy) -> None: study_data.tree.save(rulesets, ["settings", "scenariobuilder"]) # noinspection SpellCheckingInspection + @override def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = None) -> CommandOutput: study_data.tree.delete(["input", "areas", self.id]) study_data.tree.delete(["input", "hydro", "common", "capacity", f"maxpower_{self.id}"]) @@ -281,6 +285,7 @@ def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = return output + @override def to_dto(self) -> CommandDTO: return CommandDTO( action=CommandName.REMOVE_AREA.value, @@ -290,14 +295,18 @@ def to_dto(self) -> CommandDTO: study_version=self.study_version, ) + @override def match_signature(self) -> str: return str(self.command_name.value + MATCH_SIGNATURE_SEPARATOR + self.id) + @override def match(self, other: ICommand, equal: bool = False) -> bool: return isinstance(other, RemoveArea) and self.id == other.id + @override def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: return [] + @override def get_inner_matrices(self) -> t.List[str]: return [] diff --git a/antarest/study/storage/variantstudy/model/command/remove_binding_constraint.py b/antarest/study/storage/variantstudy/model/command/remove_binding_constraint.py index 916ba5ea70..55dd971133 100644 --- a/antarest/study/storage/variantstudy/model/command/remove_binding_constraint.py +++ b/antarest/study/storage/variantstudy/model/command/remove_binding_constraint.py @@ -12,6 +12,8 @@ from typing import Any, Dict, List, Optional, Tuple +from typing_extensions import override + from antarest.core.model import JSON from antarest.study.model import STUDY_VERSION_8_7 from antarest.study.storage.rawstudy.model.filesystem.config.binding_constraint import DEFAULT_GROUP @@ -35,6 +37,7 @@ class RemoveBindingConstraint(ICommand): # Properties of the `REMOVE_BINDING_CONSTRAINT` command: id: str + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> Tuple[CommandOutput, Dict[str, Any]]: if self.id not in [bind.id for bind in study_data.bindings]: return ( @@ -44,6 +47,7 @@ def _apply_config(self, study_data: FileStudyTreeConfig) -> Tuple[CommandOutput, study_data.bindings.remove(next(iter([bind for bind in study_data.bindings if bind.id == self.id]))) return CommandOutput(status=True), {} + @override def _apply(self, study_data: FileStudy, listener: Optional[ICommandListener] = None) -> CommandOutput: if self.id not in [bind.id for bind in study_data.config.bindings]: return CommandOutput(status=False, message=f"Binding constraint not found: '{self.id}'") @@ -76,6 +80,7 @@ def _apply(self, study_data: FileStudy, listener: Optional[ICommandListener] = N return self._apply_config(study_data.config)[0] + @override def to_dto(self) -> CommandDTO: return CommandDTO( action=CommandName.REMOVE_BINDING_CONSTRAINT.value, @@ -85,16 +90,20 @@ def to_dto(self) -> CommandDTO: study_version=self.study_version, ) + @override def match_signature(self) -> str: return str(self.command_name.value + MATCH_SIGNATURE_SEPARATOR + self.id) + @override def match(self, other: ICommand, equal: bool = False) -> bool: if not isinstance(other, RemoveBindingConstraint): return False return self.id == other.id + @override def _create_diff(self, other: "ICommand") -> List["ICommand"]: return [] + @override def get_inner_matrices(self) -> List[str]: return [] diff --git a/antarest/study/storage/variantstudy/model/command/remove_cluster.py b/antarest/study/storage/variantstudy/model/command/remove_cluster.py index b64bbdfc39..396d0d5d7b 100644 --- a/antarest/study/storage/variantstudy/model/command/remove_cluster.py +++ b/antarest/study/storage/variantstudy/model/command/remove_cluster.py @@ -13,6 +13,7 @@ import typing as t from pydantic import Field +from typing_extensions import override from antarest.core.model import LowerCaseStr from antarest.study.storage.rawstudy.model.filesystem.config.model import Area, FileStudyTreeConfig @@ -43,6 +44,7 @@ class RemoveCluster(ICommand): area_id: str cluster_id: LowerCaseStr = Field(description="Cluster ID", pattern=r"[a-z0-9_(),& -]+") + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> t.Tuple[CommandOutput, t.Dict[str, t.Any]]: """ Applies configuration changes to the study data: remove the thermal clusters from the storages list. @@ -101,6 +103,7 @@ def _remove_cluster_from_scenario_builder(self, study_data: FileStudy) -> None: study_data.tree.save(rulesets, ["settings", "scenariobuilder"]) + @override def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = None) -> CommandOutput: """ Applies the study data to update thermal cluster configurations and saves the changes: @@ -142,6 +145,7 @@ def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = # deleting the files and folders. return self._apply_config(study_data.config)[0] + @override def to_dto(self) -> CommandDTO: return CommandDTO( action=self.command_name.value, @@ -149,6 +153,7 @@ def to_dto(self) -> CommandDTO: study_version=self.study_version, ) + @override def match_signature(self) -> str: return str( self.command_name.value @@ -158,14 +163,17 @@ def match_signature(self) -> str: + self.area_id ) + @override def match(self, other: ICommand, equal: bool = False) -> bool: if not isinstance(other, RemoveCluster): return False return self.cluster_id == other.cluster_id and self.area_id == other.area_id + @override def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: return [] + @override def get_inner_matrices(self) -> t.List[str]: return [] diff --git a/antarest/study/storage/variantstudy/model/command/remove_district.py b/antarest/study/storage/variantstudy/model/command/remove_district.py index 9e842d9872..2b675aed3f 100644 --- a/antarest/study/storage/variantstudy/model/command/remove_district.py +++ b/antarest/study/storage/variantstudy/model/command/remove_district.py @@ -12,6 +12,8 @@ from typing import Any, Dict, List, Optional, Tuple +from typing_extensions import override + from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.factory import FileStudy from antarest.study.storage.variantstudy.model.command.common import CommandName, CommandOutput @@ -36,15 +38,18 @@ class RemoveDistrict(ICommand): id: str + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> Tuple[CommandOutput, Dict[str, Any]]: del study_data.sets[self.id] return CommandOutput(status=True, message=self.id), dict() + @override def _apply(self, study_data: FileStudy, listener: Optional[ICommandListener] = None) -> CommandOutput: output, _ = self._apply_config(study_data.config) study_data.tree.delete(["input", "areas", "sets", self.id]) return output + @override def to_dto(self) -> CommandDTO: return CommandDTO( action=CommandName.REMOVE_DISTRICT.value, @@ -54,16 +59,20 @@ def to_dto(self) -> CommandDTO: study_version=self.study_version, ) + @override def match_signature(self) -> str: return str(self.command_name.value + MATCH_SIGNATURE_SEPARATOR + self.id) + @override def match(self, other: ICommand, equal: bool = False) -> bool: if not isinstance(other, RemoveDistrict): return False return self.id == other.id + @override def _create_diff(self, other: "ICommand") -> List["ICommand"]: return [] + @override def get_inner_matrices(self) -> List[str]: return [] diff --git a/antarest/study/storage/variantstudy/model/command/remove_link.py b/antarest/study/storage/variantstudy/model/command/remove_link.py index 9b372e906f..668c5ff2b9 100644 --- a/antarest/study/storage/variantstudy/model/command/remove_link.py +++ b/antarest/study/storage/variantstudy/model/command/remove_link.py @@ -13,6 +13,7 @@ import typing as t from pydantic import field_validator, model_validator +from typing_extensions import override from antarest.study.model import STUDY_VERSION_8_2 from antarest.study.storage.rawstudy.model.filesystem.config.field_validators import transform_name_to_id @@ -91,6 +92,7 @@ def _check_link_exists(self, study_cfg: FileStudyTreeConfig) -> OutputTuple: return CommandOutput(status=bool(data), message=message), data + @override def _apply_config(self, study_cfg: FileStudyTreeConfig) -> OutputTuple: """ Update the study configuration by removing the link between the source and target areas. @@ -126,6 +128,7 @@ def _remove_link_from_scenario_builder(self, study_data: FileStudy) -> None: study_data.tree.save(rulesets, ["settings", "scenariobuilder"]) + @override def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = None) -> CommandOutput: """ Update the configuration and the study data by removing the link between the source and target areas. @@ -152,6 +155,7 @@ def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = return self._apply_config(study_data.config)[0] + @override def to_dto(self) -> CommandDTO: return CommandDTO( action=CommandName.REMOVE_LINK.value, @@ -159,17 +163,21 @@ def to_dto(self) -> CommandDTO: study_version=self.study_version, ) + @override def match_signature(self) -> str: sep = MATCH_SIGNATURE_SEPARATOR return f"{self.command_name.value}{sep}{self.area1}{sep}{self.area2}" + @override def match(self, other: ICommand, equal: bool = False) -> bool: if not isinstance(other, RemoveLink): return False return self.area1 == other.area1 and self.area2 == other.area2 + @override def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: return [] + @override def get_inner_matrices(self) -> t.List[str]: return [] diff --git a/antarest/study/storage/variantstudy/model/command/remove_renewables_cluster.py b/antarest/study/storage/variantstudy/model/command/remove_renewables_cluster.py index 592b75bfbb..8e62dabbfa 100644 --- a/antarest/study/storage/variantstudy/model/command/remove_renewables_cluster.py +++ b/antarest/study/storage/variantstudy/model/command/remove_renewables_cluster.py @@ -12,10 +12,10 @@ import typing as t -from pydantic import Field, field_validator +from pydantic import Field +from typing_extensions import override from antarest.core.model import LowerCaseStr -from antarest.study.storage.rawstudy.model.filesystem.config.field_validators import validate_id_against_name from antarest.study.storage.rawstudy.model.filesystem.config.model import Area, FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.factory import FileStudy from antarest.study.storage.variantstudy.model.command.common import CommandName, CommandOutput @@ -41,6 +41,7 @@ class RemoveRenewablesCluster(ICommand): area_id: str cluster_id: LowerCaseStr = Field(description="Cluster ID", pattern=r"[a-z0-9_(),& -]+") + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> t.Tuple[CommandOutput, t.Dict[str, t.Any]]: """ Applies configuration changes to the study data: remove the renewable clusters from the storages list. @@ -97,6 +98,7 @@ def _remove_cluster_from_scenario_builder(self, study_data: FileStudy) -> None: study_data.tree.save(rulesets, ["settings", "scenariobuilder"]) + @override def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = None) -> CommandOutput: """ Applies the study data to update renewable cluster configurations and saves the changes: @@ -135,6 +137,7 @@ def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = # deleting the files and folders. return self._apply_config(study_data.config)[0] + @override def to_dto(self) -> CommandDTO: return CommandDTO( action=self.command_name.value, @@ -142,6 +145,7 @@ def to_dto(self) -> CommandDTO: study_version=self.study_version, ) + @override def match_signature(self) -> str: return str( self.command_name.value @@ -151,13 +155,16 @@ def match_signature(self) -> str: + self.area_id ) + @override def match(self, other: ICommand, equal: bool = False) -> bool: if not isinstance(other, RemoveRenewablesCluster): return False return self.cluster_id == other.cluster_id and self.area_id == other.area_id + @override def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: return [] + @override def get_inner_matrices(self) -> t.List[str]: return [] diff --git a/antarest/study/storage/variantstudy/model/command/remove_st_storage.py b/antarest/study/storage/variantstudy/model/command/remove_st_storage.py index f27ebc5a58..3d10466793 100644 --- a/antarest/study/storage/variantstudy/model/command/remove_st_storage.py +++ b/antarest/study/storage/variantstudy/model/command/remove_st_storage.py @@ -13,6 +13,7 @@ import typing as t from pydantic import Field +from typing_extensions import override from antarest.core.model import LowerCaseStr from antarest.study.model import STUDY_VERSION_8_6 @@ -44,6 +45,7 @@ class RemoveSTStorage(ICommand): area_id: str = Field(description="Area ID", pattern=r"[a-z0-9_(),& -]+") storage_id: LowerCaseStr = Field(description="Short term storage ID", pattern=r"[a-z0-9_(),& -]+") + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> t.Tuple[CommandOutput, t.Dict[str, t.Any]]: """ Applies configuration changes to the study data: remove the storage from the storages list. @@ -101,6 +103,7 @@ def _apply_config(self, study_data: FileStudyTreeConfig) -> t.Tuple[CommandOutpu {}, ) + @override def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = None) -> CommandOutput: """ Applies the study data to update storage configurations and saves the changes: @@ -130,6 +133,7 @@ def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = # deleting the files and folders. return self._apply_config(study_data.config)[0] + @override def to_dto(self) -> CommandDTO: """ Converts the current object to a Data Transfer Object (DTO) @@ -144,6 +148,7 @@ def to_dto(self) -> CommandDTO: study_version=self.study_version, ) + @override def match_signature(self) -> str: """Returns the command signature.""" return str( @@ -154,13 +159,16 @@ def match_signature(self) -> str: + self.storage_id ) + @override def match(self, other: "ICommand", equal: bool = False) -> bool: # always perform a deep comparison, as there are no parameters # or matrices, so that shallow and deep comparisons are identical. return self.__eq__(other) + @override def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: return [] + @override def get_inner_matrices(self) -> t.List[str]: return [] diff --git a/antarest/study/storage/variantstudy/model/command/remove_user_resource.py b/antarest/study/storage/variantstudy/model/command/remove_user_resource.py index 4f24041589..f873ec2d10 100644 --- a/antarest/study/storage/variantstudy/model/command/remove_user_resource.py +++ b/antarest/study/storage/variantstudy/model/command/remove_user_resource.py @@ -12,6 +12,8 @@ import typing as t +from typing_extensions import override + from antarest.core.exceptions import ChildNotFoundError from antarest.core.serialization import AntaresBaseModel from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig @@ -43,9 +45,11 @@ class RemoveUserResource(ICommand): data: RemoveUserResourceData + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> t.Tuple[CommandOutput, t.Dict[str, t.Any]]: return CommandOutput(status=True, message="ok"), {} + @override def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = None) -> CommandOutput: url = [item for item in self.data.path.split("/") if item] study_tree = study_data.tree @@ -62,6 +66,7 @@ def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = return CommandOutput(status=True, message="ok") + @override def to_dto(self) -> CommandDTO: return CommandDTO( action=self.command_name.value, @@ -69,16 +74,20 @@ def to_dto(self) -> CommandDTO: study_version=self.study_version, ) + @override def match_signature(self) -> str: return str(self.command_name.value + MATCH_SIGNATURE_SEPARATOR + self.data.path) + @override def match(self, other: ICommand, equal: bool = False) -> bool: if not isinstance(other, RemoveUserResource): return False return self.data.path == other.data.path + @override def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: return [other] + @override def get_inner_matrices(self) -> t.List[str]: return [] diff --git a/antarest/study/storage/variantstudy/model/command/replace_matrix.py b/antarest/study/storage/variantstudy/model/command/replace_matrix.py index b26dd717f1..c796c47ce2 100644 --- a/antarest/study/storage/variantstudy/model/command/replace_matrix.py +++ b/antarest/study/storage/variantstudy/model/command/replace_matrix.py @@ -13,6 +13,7 @@ import typing as t from pydantic import Field, ValidationInfo, field_validator +from typing_extensions import override from antarest.core.exceptions import ChildNotFoundError from antarest.core.model import JSON @@ -49,6 +50,7 @@ class ReplaceMatrix(ICommand): def matrix_validator(cls, matrix: t.Union[t.List[t.List[MatrixData]], str], values: ValidationInfo) -> str: return validate_matrix(matrix, values.data) + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> t.Tuple[CommandOutput, t.Dict[str, t.Any]]: return ( CommandOutput( @@ -58,6 +60,7 @@ def _apply_config(self, study_data: FileStudyTreeConfig) -> t.Tuple[CommandOutpu {}, ) + @override def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = None) -> CommandOutput: if self.target[0] == "@": self.target = AliasDecoder.decode(self.target, study_data) @@ -89,6 +92,7 @@ def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = output, _ = self._apply_config(study_data.config) return output + @override def to_dto(self) -> CommandDTO: return CommandDTO( action=CommandName.REPLACE_MATRIX.value, @@ -99,9 +103,11 @@ def to_dto(self) -> CommandDTO: study_version=self.study_version, ) + @override def match_signature(self) -> str: return str(self.command_name.value + MATCH_SIGNATURE_SEPARATOR + self.target) + @override def match(self, other: ICommand, equal: bool = False) -> bool: if not isinstance(other, ReplaceMatrix): return False @@ -109,9 +115,11 @@ def match(self, other: ICommand, equal: bool = False) -> bool: return self.target == other.target and self.matrix == other.matrix return self.target == other.target + @override def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: return [other] + @override def get_inner_matrices(self) -> t.List[str]: assert_this(isinstance(self.matrix, str)) return [strip_matrix_protocol(self.matrix)] diff --git a/antarest/study/storage/variantstudy/model/command/update_binding_constraint.py b/antarest/study/storage/variantstudy/model/command/update_binding_constraint.py index 5f0f134494..6c247b4c05 100644 --- a/antarest/study/storage/variantstudy/model/command/update_binding_constraint.py +++ b/antarest/study/storage/variantstudy/model/command/update_binding_constraint.py @@ -12,6 +12,8 @@ import typing as t +from typing_extensions import override + from antarest.core.model import JSON from antarest.study.model import STUDY_VERSION_8_7 from antarest.study.storage.rawstudy.model.filesystem.config.binding_constraint import ( @@ -112,6 +114,7 @@ class UpdateBindingConstraint(AbstractBindingConstraintCommand): # Properties of the `UPDATE_BINDING_CONSTRAINT` command: id: str + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> t.Tuple[CommandOutput, t.Dict[str, t.Any]]: index = next(i for i, bc in enumerate(study_data.bindings) if bc.id == self.id) existing_constraint = study_data.bindings[index] @@ -151,6 +154,7 @@ def _find_binding_config(self, binding_constraints: t.Mapping[str, JSON]) -> t.O return str(index), binding_config return None + @override def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = None) -> CommandOutput: binding_constraints = study_data.tree.get(["input", "bindingconstraints", "bindingconstraints"]) @@ -200,6 +204,7 @@ def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = return super().apply_binding_constraint(study_data, binding_constraints, index, self.id, old_groups=old_groups) + @override def to_dto(self) -> CommandDTO: matrices = ["values"] + [m.value for m in TermMatrices] matrix_service = self.command_context.matrix_service @@ -214,12 +219,15 @@ def to_dto(self) -> CommandDTO: action=self.command_name.value, args=json_command, version=self.version, study_version=self.study_version ) + @override def match_signature(self) -> str: return str(self.command_name.value + MATCH_SIGNATURE_SEPARATOR + self.id) + @override def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: return [other] + @override def match(self, other: "ICommand", equal: bool = False) -> bool: if not isinstance(other, self.__class__): return False diff --git a/antarest/study/storage/variantstudy/model/command/update_comments.py b/antarest/study/storage/variantstudy/model/command/update_comments.py index 108363a3a3..b5300d171c 100644 --- a/antarest/study/storage/variantstudy/model/command/update_comments.py +++ b/antarest/study/storage/variantstudy/model/command/update_comments.py @@ -12,6 +12,8 @@ from typing import Any, Dict, List, Optional, Tuple +from typing_extensions import override + from antarest.core.model import JSON from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.factory import FileStudy @@ -37,6 +39,7 @@ class UpdateComments(ICommand): comments: str + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> Tuple[CommandOutput, Dict[str, Any]]: return ( CommandOutput( @@ -46,6 +49,7 @@ def _apply_config(self, study_data: FileStudyTreeConfig) -> Tuple[CommandOutput, dict(), ) + @override def _apply(self, study_data: FileStudy, listener: Optional[ICommandListener] = None) -> CommandOutput: replace_comment_data: JSON = {"settings": {"comments": self.comments.encode("utf-8")}} @@ -54,6 +58,7 @@ def _apply(self, study_data: FileStudy, listener: Optional[ICommandListener] = N output, _ = self._apply_config(study_data.config) return output + @override def to_dto(self) -> CommandDTO: return CommandDTO( action=CommandName.UPDATE_COMMENTS.value, @@ -63,16 +68,20 @@ def to_dto(self) -> CommandDTO: study_version=self.study_version, ) + @override def match_signature(self) -> str: return str(self.command_name.value) + @override def match(self, other: ICommand, equal: bool = False) -> bool: if not isinstance(other, UpdateComments): return False return not equal or (self.comments == other.comments and equal) + @override def _create_diff(self, other: "ICommand") -> List["ICommand"]: return [other] + @override def get_inner_matrices(self) -> List[str]: return [] diff --git a/antarest/study/storage/variantstudy/model/command/update_config.py b/antarest/study/storage/variantstudy/model/command/update_config.py index 258c64e80d..ef054383f0 100644 --- a/antarest/study/storage/variantstudy/model/command/update_config.py +++ b/antarest/study/storage/variantstudy/model/command/update_config.py @@ -13,6 +13,7 @@ import typing as t import typing_extensions as te +from typing_extensions import override from antarest.core.model import JSON from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig @@ -54,6 +55,7 @@ class UpdateConfig(ICommand): target: str data: _Data + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> t.Tuple[CommandOutput, t.Dict[str, t.Any]]: # The renewable-generation-modelling parameter must be reflected in the config if self.target.startswith("settings"): @@ -64,6 +66,7 @@ def _apply_config(self, study_data: FileStudyTreeConfig) -> t.Tuple[CommandOutpu return CommandOutput(status=True, message="ok"), {} + @override def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = None) -> CommandOutput: url = self.target.split("/") tree_node = study_data.tree.get_node(url) @@ -78,6 +81,7 @@ def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = output, _ = self._apply_config(study_data.config) return output + @override def to_dto(self) -> CommandDTO: return CommandDTO( action=CommandName.UPDATE_CONFIG.value, @@ -88,9 +92,11 @@ def to_dto(self) -> CommandDTO: study_version=self.study_version, ) + @override def match_signature(self) -> str: return str(self.command_name.value + MATCH_SIGNATURE_SEPARATOR + self.target) + @override def match(self, other: ICommand, equal: bool = False) -> bool: if not isinstance(other, UpdateConfig): return False @@ -99,8 +105,10 @@ def match(self, other: ICommand, equal: bool = False) -> bool: return simple_match return simple_match and self.data == other.data + @override def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: return [other] + @override def get_inner_matrices(self) -> t.List[str]: return [] diff --git a/antarest/study/storage/variantstudy/model/command/update_district.py b/antarest/study/storage/variantstudy/model/command/update_district.py index 2819e822a8..5d0b706484 100644 --- a/antarest/study/storage/variantstudy/model/command/update_district.py +++ b/antarest/study/storage/variantstudy/model/command/update_district.py @@ -12,6 +12,8 @@ from typing import Any, Dict, List, Optional, Tuple +from typing_extensions import override + from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.factory import FileStudy from antarest.study.storage.variantstudy.model.command.common import CommandName, CommandOutput @@ -41,6 +43,7 @@ class UpdateDistrict(ICommand): output: Optional[bool] = None comments: Optional[str] = None + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> Tuple[CommandOutput, Dict[str, Any]]: base_set = study_data.sets[self.id] if self.id not in study_data.sets: @@ -67,6 +70,7 @@ def _apply_config(self, study_data: FileStudyTreeConfig) -> Tuple[CommandOutput, "item_key": item_key, } + @override def _apply(self, study_data: FileStudy, listener: Optional[ICommandListener] = None) -> CommandOutput: output, data = self._apply_config(study_data.config) if not output.status: @@ -90,6 +94,7 @@ def _apply(self, study_data: FileStudy, listener: Optional[ICommandListener] = N return output + @override def to_dto(self) -> CommandDTO: return CommandDTO( action=CommandName.UPDATE_DISTRICT.value, @@ -103,9 +108,11 @@ def to_dto(self) -> CommandDTO: study_version=self.study_version, ) + @override def match_signature(self) -> str: return str(self.command_name.value + MATCH_SIGNATURE_SEPARATOR + self.id) + @override def match(self, other: ICommand, equal: bool = False) -> bool: if not isinstance(other, UpdateDistrict): return False @@ -120,8 +127,10 @@ def match(self, other: ICommand, equal: bool = False) -> bool: and self.comments == other.comments ) + @override def _create_diff(self, other: "ICommand") -> List["ICommand"]: return [other] + @override def get_inner_matrices(self) -> List[str]: return [] diff --git a/antarest/study/storage/variantstudy/model/command/update_link.py b/antarest/study/storage/variantstudy/model/command/update_link.py index ceab38ed56..16a13fa7e3 100644 --- a/antarest/study/storage/variantstudy/model/command/update_link.py +++ b/antarest/study/storage/variantstudy/model/command/update_link.py @@ -11,6 +11,8 @@ # This file is part of the Antares project. import typing as t +from typing_extensions import override + from antarest.study.business.model.link_model import LinkInternal from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.factory import FileStudy @@ -32,6 +34,7 @@ class UpdateLink(AbstractLinkCommand): command_name: CommandName = CommandName.UPDATE_LINK version: int = 1 + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> OutputTuple: return ( CommandOutput( @@ -41,6 +44,7 @@ def _apply_config(self, study_data: FileStudyTreeConfig) -> OutputTuple: {}, ) + @override def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = None) -> CommandOutput: version = study_data.config.version @@ -65,14 +69,18 @@ def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = return output + @override def to_dto(self) -> CommandDTO: return super().to_dto() + @override def match_signature(self) -> str: return super().match_signature() + @override def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: return super()._create_diff(other) + @override def get_inner_matrices(self) -> t.List[str]: return super().get_inner_matrices() diff --git a/antarest/study/storage/variantstudy/model/command/update_playlist.py b/antarest/study/storage/variantstudy/model/command/update_playlist.py index 7840019672..3c19b9a2f4 100644 --- a/antarest/study/storage/variantstudy/model/command/update_playlist.py +++ b/antarest/study/storage/variantstudy/model/command/update_playlist.py @@ -12,6 +12,8 @@ from typing import Any, Dict, List, Optional, Tuple +from typing_extensions import override + from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.factory import FileStudy from antarest.study.storage.rawstudy.model.helpers import FileStudyHelpers @@ -40,6 +42,7 @@ class UpdatePlaylist(ICommand): weights: Optional[Dict[int, float]] = None reverse: bool = False + @override def _apply(self, study_data: FileStudy, listener: Optional[ICommandListener] = None) -> CommandOutput: FileStudyHelpers.set_playlist( study_data, @@ -50,9 +53,11 @@ def _apply(self, study_data: FileStudy, listener: Optional[ICommandListener] = N ) return CommandOutput(status=True) + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> Tuple[CommandOutput, Dict[str, Any]]: return CommandOutput(status=True), {} + @override def to_dto(self) -> CommandDTO: return CommandDTO( action=CommandName.UPDATE_PLAYLIST.value, @@ -65,9 +70,11 @@ def to_dto(self) -> CommandDTO: study_version=self.study_version, ) + @override def match_signature(self) -> str: return CommandName.UPDATE_PLAYLIST.name + @override def match(self, other: "ICommand", equal: bool = False) -> bool: if not isinstance(other, UpdatePlaylist): return False @@ -80,8 +87,10 @@ def match(self, other: "ICommand", equal: bool = False) -> bool: ) return True + @override def _create_diff(self, other: "ICommand") -> List["ICommand"]: return [other] + @override def get_inner_matrices(self) -> List[str]: return [] diff --git a/antarest/study/storage/variantstudy/model/command/update_raw_file.py b/antarest/study/storage/variantstudy/model/command/update_raw_file.py index 4090b5ead1..97446dadcc 100644 --- a/antarest/study/storage/variantstudy/model/command/update_raw_file.py +++ b/antarest/study/storage/variantstudy/model/command/update_raw_file.py @@ -13,6 +13,8 @@ import base64 from typing import Any, Dict, List, Optional, Tuple +from typing_extensions import override + from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig from antarest.study.storage.rawstudy.model.filesystem.factory import FileStudy from antarest.study.storage.rawstudy.model.filesystem.raw_file_node import RawFileNode @@ -39,6 +41,7 @@ class UpdateRawFile(ICommand): target: str b64Data: str + @override def __repr__(self) -> str: cls = self.__class__.__name__ target = self.target @@ -48,9 +51,11 @@ def __repr__(self) -> str: except (ValueError, TypeError): return f"{cls}(target={target!r}, b64Data={self.b64Data!r})" + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> Tuple[CommandOutput, Dict[str, Any]]: return CommandOutput(status=True, message="ok"), {} + @override def _apply(self, study_data: FileStudy, listener: Optional[ICommandListener] = None) -> CommandOutput: url = self.target.split("/") tree_node = study_data.tree.get_node(url) @@ -63,6 +68,7 @@ def _apply(self, study_data: FileStudy, listener: Optional[ICommandListener] = N study_data.tree.save(base64.decodebytes(self.b64Data.encode("utf-8")), url) return CommandOutput(status=True, message="ok") + @override def to_dto(self) -> CommandDTO: return CommandDTO( action=self.command_name.value, @@ -70,9 +76,11 @@ def to_dto(self) -> CommandDTO: study_version=self.study_version, ) + @override def match_signature(self) -> str: return str(self.command_name.value + MATCH_SIGNATURE_SEPARATOR + self.target) + @override def match(self, other: ICommand, equal: bool = False) -> bool: if not isinstance(other, UpdateRawFile): return False @@ -81,8 +89,10 @@ def match(self, other: ICommand, equal: bool = False) -> bool: return simple_match return simple_match and self.b64Data == other.b64Data + @override def _create_diff(self, other: "ICommand") -> List["ICommand"]: return [other] + @override def get_inner_matrices(self) -> List[str]: return [] diff --git a/antarest/study/storage/variantstudy/model/command/update_scenario_builder.py b/antarest/study/storage/variantstudy/model/command/update_scenario_builder.py index 89b12b5e51..4328bcf1fe 100644 --- a/antarest/study/storage/variantstudy/model/command/update_scenario_builder.py +++ b/antarest/study/storage/variantstudy/model/command/update_scenario_builder.py @@ -13,6 +13,7 @@ import typing as t import numpy as np +from typing_extensions import override from antarest.core.requests import CaseInsensitiveDict from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig @@ -54,6 +55,7 @@ class UpdateScenarioBuilder(ICommand): data: t.Union[t.Dict[str, t.Any], t.Mapping[str, t.Any], t.MutableMapping[str, t.Any]] + @override def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = None) -> CommandOutput: """ Apply the command to the study data. @@ -95,17 +97,21 @@ def _apply(self, study_data: FileStudy, listener: t.Optional[ICommandListener] = study_data.tree.save(curr_cfg, url) # type: ignore return CommandOutput(status=True) + @override def _apply_config(self, study_data: FileStudyTreeConfig) -> t.Tuple[CommandOutput, t.Dict[str, t.Any]]: return CommandOutput(status=True), {} + @override def to_dto(self) -> CommandDTO: return CommandDTO( action=CommandName.UPDATE_SCENARIO_BUILDER.value, args={"data": self.data}, study_version=self.study_version ) + @override def match_signature(self) -> str: return CommandName.UPDATE_SCENARIO_BUILDER.value + @override def match(self, other: "ICommand", equal: bool = False) -> bool: if not isinstance(other, UpdateScenarioBuilder): return False @@ -113,8 +119,10 @@ def match(self, other: "ICommand", equal: bool = False) -> bool: return self.data == other.data return True + @override def _create_diff(self, other: "ICommand") -> t.List["ICommand"]: return [other] + @override def get_inner_matrices(self) -> t.List[str]: return [] diff --git a/antarest/study/storage/variantstudy/model/dbmodel.py b/antarest/study/storage/variantstudy/model/dbmodel.py index 180b7105ef..b855a75cf9 100644 --- a/antarest/study/storage/variantstudy/model/dbmodel.py +++ b/antarest/study/storage/variantstudy/model/dbmodel.py @@ -17,6 +17,7 @@ from sqlalchemy import Column, DateTime, ForeignKey, Integer, String # type: ignore from sqlalchemy.orm import relationship # type: ignore +from typing_extensions import override from antarest.core.persistence import Base from antarest.core.serialization import from_json @@ -43,6 +44,7 @@ class VariantStudySnapshot(Base): # type: ignore "polymorphic_identity": "variant_study_snapshot", } + @override def __str__(self) -> str: return f"[Snapshot] id={self.id}, created_at={self.created_at}" @@ -82,6 +84,7 @@ def to_dto(self) -> CommandDTO: updated_at=self.updated_at, ) + @override def __str__(self) -> str: return ( f"CommandBlock(id={self.id!r}," @@ -125,6 +128,7 @@ class VariantStudy(Study): cascade="all, delete, delete-orphan", ) + @override def __str__(self) -> str: return super().__str__() + f", snapshot={self.snapshot}" diff --git a/antarest/study/storage/variantstudy/repository.py b/antarest/study/storage/variantstudy/repository.py index fc4fcc3809..a090b84c5a 100644 --- a/antarest/study/storage/variantstudy/repository.py +++ b/antarest/study/storage/variantstudy/repository.py @@ -13,6 +13,7 @@ import typing as t from sqlalchemy.orm import Session, joinedload # type: ignore +from typing_extensions import override from antarest.core.interfaces.cache import ICache from antarest.core.utils.fastapi_sqlalchemy import db @@ -37,6 +38,7 @@ def __init__(self, cache_service: ICache, session: t.Optional[Session] = None): super().__init__(cache_service) self._session = session + @override @property def session(self) -> Session: """ diff --git a/antarest/study/storage/variantstudy/variant_study_service.py b/antarest/study/storage/variantstudy/variant_study_service.py index 75154dae95..82fed1d98e 100644 --- a/antarest/study/storage/variantstudy/variant_study_service.py +++ b/antarest/study/storage/variantstudy/variant_study_service.py @@ -24,6 +24,7 @@ from antares.study.version import StudyVersion from fastapi import HTTPException from filelock import FileLock +from typing_extensions import override from antarest.core.config import Config from antarest.core.exceptions import ( @@ -532,6 +533,7 @@ def _get_variants_parents(self, id: str, params: RequestParameters) -> t.List[St return output_list + @override def get( self, metadata: VariantStudy, @@ -561,6 +563,7 @@ def get( use_cache=use_cache, ) + @override def get_file( self, metadata: VariantStudy, @@ -870,6 +873,7 @@ def get_study_task(self, study_id: str, params: RequestParameters) -> TaskDTO: return self.task_service.status_task(task_id=task_id, request_params=params, with_logs=True) raise StudyValidationError(f"Variant study '{study_id}' has no generation task") + @override def create(self, study: VariantStudy) -> VariantStudy: """ Create an empty new study. @@ -879,6 +883,7 @@ def create(self, study: VariantStudy) -> VariantStudy: """ raise NotImplementedError() + @override def exists(self, metadata: VariantStudy) -> bool: """ Check if the study snapshot exists and is up-to-date. @@ -894,6 +899,7 @@ def exists(self, metadata: VariantStudy) -> bool: and (self.get_study_path(metadata) / "study.antares").is_file() ) + @override def copy( self, src_meta: VariantStudy, @@ -991,6 +997,7 @@ def _get_snapshot_last_executed_command_index( return last_executed_command_index if last_executed_command_index >= 0 else None return None + @override def get_raw( self, metadata: VariantStudy, @@ -1015,6 +1022,7 @@ def get_raw( use_cache=use_cache, ) + @override def get_study_sim_result(self, study: VariantStudy) -> t.List[StudySimResultDTO]: """ Get global result information @@ -1025,6 +1033,7 @@ def get_study_sim_result(self, study: VariantStudy) -> t.List[StudySimResultDTO] self._safe_generation(study, timeout=600) return super().get_study_sim_result(study=study) + @override def set_reference_output(self, metadata: VariantStudy, output_id: str, status: bool) -> None: """ Set an output to the reference output of a study @@ -1037,6 +1046,7 @@ def set_reference_output(self, metadata: VariantStudy, output_id: str, status: b self.patch_service.set_reference_output(metadata, output_id, status) remove_from_cache(self.cache, metadata.id) + @override def delete(self, metadata: VariantStudy) -> None: """ Delete study @@ -1049,6 +1059,7 @@ def delete(self, metadata: VariantStudy) -> None: shutil.rmtree(study_path) remove_from_cache(self.cache, metadata.id) + @override def delete_output(self, metadata: VariantStudy, output_id: str) -> None: """ Delete a simulation output @@ -1062,6 +1073,7 @@ def delete_output(self, metadata: VariantStudy, output_id: str) -> None: shutil.rmtree(output_path, ignore_errors=True) remove_from_cache(self.cache, metadata.id) + @override def get_study_path(self, metadata: Study) -> Path: """ Get study path @@ -1073,6 +1085,7 @@ def get_study_path(self, metadata: Study) -> Path: """ return Path(metadata.path) / SNAPSHOT_RELATIVE_PATH + @override def export_study_flat( self, metadata: VariantStudy, @@ -1096,6 +1109,7 @@ def export_study_flat( output_src_path, ) + @override def get_synthesis( self, metadata: VariantStudy, @@ -1118,6 +1132,7 @@ def get_synthesis( raise VariantGenerationError(f"Error during light generation of {metadata.id}") + @override def initialize_additional_data(self, variant_study: VariantStudy) -> bool: try: if self.exists(variant_study): diff --git a/antarest/tools/lib.py b/antarest/tools/lib.py index a8cef7e371..03d59a1949 100644 --- a/antarest/tools/lib.py +++ b/antarest/tools/lib.py @@ -22,6 +22,7 @@ import numpy as np from antares.study.version import StudyVersion from httpx import Client +from typing_extensions import override from antarest.core.cache.business.local_chache import LocalCache from antarest.core.config import CacheConfig @@ -76,6 +77,7 @@ def __init__( self.session = session self.host = host + @override def apply_commands( self, commands: List[CommandDTO], @@ -139,6 +141,7 @@ def render_template(self, study_version: StudyVersion = NEW_DEFAULT_STUDY_VERSIO sets_ini = self.output_path.joinpath("input/areas/sets.ini") sets_ini.write_bytes(b"") + @override def apply_commands(self, commands: List[CommandDTO], matrices_dir: Path) -> GenerationResultInfoDTO: stopwatch = StopWatch() matrix_content_repository = MatrixContentRepository( diff --git a/antarest/worker/archive_worker.py b/antarest/worker/archive_worker.py index 6ce01acf85..23bcd3aa69 100644 --- a/antarest/worker/archive_worker.py +++ b/antarest/worker/archive_worker.py @@ -13,6 +13,8 @@ import logging from pathlib import Path +from typing_extensions import override + from antarest.core.config import Config from antarest.core.interfaces.eventbus import IEventBus from antarest.core.serialization import AntaresBaseModel @@ -57,6 +59,7 @@ def __init__( [f"{ArchiveWorker.TASK_TYPE}_{workspace}"], ) + @override def _execute_task(self, task_info: WorkerTaskCommand) -> TaskResult: logger.info(f"Executing task {task_info.model_dump_json()}") try: diff --git a/antarest/worker/worker.py b/antarest/worker/worker.py index f73dd551f0..6de459efbb 100644 --- a/antarest/worker/worker.py +++ b/antarest/worker/worker.py @@ -16,6 +16,8 @@ from concurrent.futures import Future, ThreadPoolExecutor from typing import Any, Dict, List, Union +from typing_extensions import override + from antarest.core.interfaces.eventbus import Event, EventType, IEventBus from antarest.core.interfaces.service import IService from antarest.core.model import PermissionInfo, PublicMode @@ -99,6 +101,7 @@ def __init__( thread_name_prefix="worker_task_", ) + @override def _loop(self) -> None: for task_type in self.accept: self.event_bus.add_queue_consumer(self._listen_for_tasks, task_type) diff --git a/pyproject.toml b/pyproject.toml index 6073d80770..3b3b9a3849 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -39,6 +39,7 @@ exclude = "antarest/fastapi_jwt_auth/*" strict = true files = "antarest" plugins = "pydantic.mypy" +enable_error_code = "explicit-override" [[tool.mypy.overrides]] module = ["antarest/fastapi_jwt_auth.*"]