From 07cf7cac22554b3c6cd63c5eb66766ccb5fbd2ea Mon Sep 17 00:00:00 2001 From: belthlemar Date: Thu, 29 Feb 2024 15:12:48 +0100 Subject: [PATCH 1/3] fix(outputs): build outputs config even when using cache --- .../rawstudy/model/filesystem/config/files.py | 4 +- .../rawstudy/model/filesystem/factory.py | 5 +- .../variant_blueprint/test_variant_manager.py | 54 +++++++++++++++++++ .../filesystem/config/test_config_files.py | 4 +- 4 files changed, 62 insertions(+), 5 deletions(-) diff --git a/antarest/study/storage/rawstudy/model/filesystem/config/files.py b/antarest/study/storage/rawstudy/model/filesystem/config/files.py index 3727f320ec..3248b6560a 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/config/files.py +++ b/antarest/study/storage/rawstudy/model/filesystem/config/files.py @@ -74,7 +74,7 @@ def build(study_path: Path, study_id: str, output_path: t.Optional[Path] = None) version=_parse_version(study_path), areas=_parse_areas(study_path), sets=_parse_sets(study_path), - outputs=_parse_outputs(outputs_dir), + outputs=parse_outputs(outputs_dir), bindings=_parse_bindings(study_path), store_new_set=sns, archive_input_series=asi, @@ -232,7 +232,7 @@ def _parse_areas(root: Path) -> t.Dict[str, Area]: return {transform_name_to_id(a): parse_area(root, a) for a in areas} -def _parse_outputs(output_path: Path) -> t.Dict[str, Simulation]: +def parse_outputs(output_path: Path) -> t.Dict[str, Simulation]: if not output_path.is_dir(): return {} sims = {} diff --git a/antarest/study/storage/rawstudy/model/filesystem/factory.py b/antarest/study/storage/rawstudy/model/filesystem/factory.py index 1899ec1bb4..040e747629 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/factory.py +++ b/antarest/study/storage/rawstudy/model/filesystem/factory.py @@ -10,7 +10,7 @@ from antarest.core.interfaces.cache import CacheConstants, ICache from antarest.matrixstore.service import ISimpleMatrixService from antarest.matrixstore.uri_resolver_service import UriResolverService -from antarest.study.storage.rawstudy.model.filesystem.config.files import build +from antarest.study.storage.rawstudy.model.filesystem.config.files import build, parse_outputs from antarest.study.storage.rawstudy.model.filesystem.config.model import FileStudyTreeConfig, FileStudyTreeConfigDTO from antarest.study.storage.rawstudy.model.filesystem.context import ContextServer from antarest.study.storage.rawstudy.model.filesystem.root.filestudytree import FileStudyTree @@ -93,6 +93,9 @@ def _create_from_fs_unsafe( if from_cache is not None: logger.info(f"Study {study_id} read from cache") config = FileStudyTreeConfigDTO.parse_obj(from_cache).to_build_config() + if output_path: + config.output_path = output_path + config.outputs = parse_outputs(output_path) return FileStudy(config, FileStudyTree(self.context, config)) start_time = time.time() config = build(path, study_id, output_path) diff --git a/tests/integration/variant_blueprint/test_variant_manager.py b/tests/integration/variant_blueprint/test_variant_manager.py index 8a300e75da..4b066cdf73 100644 --- a/tests/integration/variant_blueprint/test_variant_manager.py +++ b/tests/integration/variant_blueprint/test_variant_manager.py @@ -1,10 +1,13 @@ +import io import logging +import time import typing as t import pytest from starlette.testclient import TestClient from antarest.core.tasks.model import TaskDTO, TaskStatus +from tests.integration.assets import ASSETS_DIR @pytest.fixture(name="base_study_id") @@ -251,3 +254,54 @@ def test_recursive_variant_tree(client: TestClient, admin_access_token: str): # Asserts that we do not trigger a Recursive Exception res = client.get(f"/v1/studies/{parent_id}/variants", headers=admin_headers) assert res.status_code == 200 + + +def test_outputs(client: TestClient, admin_access_token: str, tmp_path: str) -> None: + # ======================= + # SET UP + # ======================= + + admin_headers = {"Authorization": f"Bearer {admin_access_token}"} + res = client.post(f"/v1/studies?name=foo", headers=admin_headers) + parent_id = res.json() + res = client.post(f"/v1/studies/{parent_id}/variants?name=variant_foo", headers=admin_headers) + variant_id = res.json() + + # Only done to generate the variant folder + res = client.post(f"/v1/launcher/run/{variant_id}", headers=admin_headers) + job_id = res.json()["job_id"] + status = client.get(f"/v1/launcher/jobs/{job_id}", headers=admin_headers).json()["status"] + while status != "failed": + time.sleep(0.2) + status = client.get(f"/v1/launcher/jobs/{job_id}", headers=admin_headers).json()["status"] + + # Import an output to the study folder + output_path_zip = ASSETS_DIR / "output_adq.zip" + client.post( + f"/v1/studies/{variant_id}/output", + headers=admin_headers, + files={"output": io.BytesIO(output_path_zip.read_bytes())}, + ) + + # ======================= + # ASSERTS GENERATING THE VARIANT DOES NOT `HIDE` OUTPUTS FROM THE ENDPOINT + # ======================= + + # Get output + res = client.get(f"/v1/studies/{variant_id}/outputs", headers=admin_headers).json() + assert len(res) == 1 + + # Generates the study + res = client.put(f"/v1/studies/{variant_id}/generate?denormalize=false&from_scratch=true", headers=admin_headers) + task_id = res.json() + # Wait for task completion + res = client.get(f"/v1/tasks/{task_id}", headers=admin_headers, params={"wait_for_completion": True}) + assert res.status_code == 200 + task_result = TaskDTO.parse_obj(res.json()) + assert task_result.status == TaskStatus.COMPLETED + assert task_result.result is not None + assert task_result.result.success + + # Get outputs again + res = client.get(f"/v1/studies/{variant_id}/outputs", headers=admin_headers).json() + assert len(res) == 1 diff --git a/tests/storage/repository/filesystem/config/test_config_files.py b/tests/storage/repository/filesystem/config/test_config_files.py index 7cbab645bc..d8d157f01f 100644 --- a/tests/storage/repository/filesystem/config/test_config_files.py +++ b/tests/storage/repository/filesystem/config/test_config_files.py @@ -11,12 +11,12 @@ ) from antarest.study.storage.rawstudy.model.filesystem.config.files import ( _parse_links, - _parse_outputs, _parse_renewables, _parse_sets, _parse_st_storage, _parse_thermal, build, + parse_outputs, ) from antarest.study.storage.rawstudy.model.filesystem.config.model import ( Area, @@ -227,7 +227,7 @@ def test_parse_outputs__nominal(tmp_path: Path, assets_name: str, expected: Dict with ZipFile(pkg_dir) as zf: zf.extractall(tmp_path) output_path = tmp_path.joinpath("output") - actual = _parse_outputs(output_path) + actual = parse_outputs(output_path) assert actual == expected From 70dc01754a672aecf04df0fa9618c4a694cd196e Mon Sep 17 00:00:00 2001 From: Laurent LAPORTE Date: Mon, 25 Mar 2024 20:33:54 +0100 Subject: [PATCH 2/3] test(outputs): improve unit test `test_outputs` --- .../variant_blueprint/test_variant_manager.py | 50 ++++++++++++------- 1 file changed, 31 insertions(+), 19 deletions(-) diff --git a/tests/integration/variant_blueprint/test_variant_manager.py b/tests/integration/variant_blueprint/test_variant_manager.py index 4b066cdf73..a0e4a68108 100644 --- a/tests/integration/variant_blueprint/test_variant_manager.py +++ b/tests/integration/variant_blueprint/test_variant_manager.py @@ -243,33 +243,34 @@ def test_comments(client: TestClient, admin_access_token: str, variant_id: str) assert res.json() == comment -def test_recursive_variant_tree(client: TestClient, admin_access_token: str): +def test_recursive_variant_tree(client: TestClient, admin_access_token: str, base_study_id: str) -> None: admin_headers = {"Authorization": f"Bearer {admin_access_token}"} - base_study_res = client.post("/v1/studies?name=foo", headers=admin_headers) - base_study_id = base_study_res.json() - parent_id = base_study_res.json() - for k in range(150): - res = client.post(f"/v1/studies/{base_study_id}/variants?name=variant_{k}", headers=admin_headers) + parent_id = base_study_id + for k in range(200): + res = client.post( + f"/v1/studies/{base_study_id}/variants", + headers=admin_headers, + params={"name": f"variant_{k}"}, + ) base_study_id = res.json() + # Asserts that we do not trigger a Recursive Exception res = client.get(f"/v1/studies/{parent_id}/variants", headers=admin_headers) - assert res.status_code == 200 + assert res.status_code == 200, res.json() -def test_outputs(client: TestClient, admin_access_token: str, tmp_path: str) -> None: +def test_outputs(client: TestClient, admin_access_token: str, variant_id: str, tmp_path: str) -> None: # ======================= # SET UP # ======================= admin_headers = {"Authorization": f"Bearer {admin_access_token}"} - res = client.post(f"/v1/studies?name=foo", headers=admin_headers) - parent_id = res.json() - res = client.post(f"/v1/studies/{parent_id}/variants?name=variant_foo", headers=admin_headers) - variant_id = res.json() # Only done to generate the variant folder res = client.post(f"/v1/launcher/run/{variant_id}", headers=admin_headers) + res.raise_for_status() job_id = res.json()["job_id"] + status = client.get(f"/v1/launcher/jobs/{job_id}", headers=admin_headers).json()["status"] while status != "failed": time.sleep(0.2) @@ -277,31 +278,42 @@ def test_outputs(client: TestClient, admin_access_token: str, tmp_path: str) -> # Import an output to the study folder output_path_zip = ASSETS_DIR / "output_adq.zip" - client.post( + res = client.post( f"/v1/studies/{variant_id}/output", headers=admin_headers, files={"output": io.BytesIO(output_path_zip.read_bytes())}, ) + res.raise_for_status() # ======================= # ASSERTS GENERATING THE VARIANT DOES NOT `HIDE` OUTPUTS FROM THE ENDPOINT # ======================= # Get output - res = client.get(f"/v1/studies/{variant_id}/outputs", headers=admin_headers).json() - assert len(res) == 1 + res = client.get(f"/v1/studies/{variant_id}/outputs", headers=admin_headers) + assert res.status_code == 200, res.json() + outputs = res.json() + assert len(outputs) == 1 # Generates the study - res = client.put(f"/v1/studies/{variant_id}/generate?denormalize=false&from_scratch=true", headers=admin_headers) + res = client.put( + f"/v1/studies/{variant_id}/generate", + headers=admin_headers, + params={"denormalize": False, "from_scratch": True}, + ) + res.raise_for_status() task_id = res.json() + # Wait for task completion res = client.get(f"/v1/tasks/{task_id}", headers=admin_headers, params={"wait_for_completion": True}) - assert res.status_code == 200 + res.raise_for_status() task_result = TaskDTO.parse_obj(res.json()) assert task_result.status == TaskStatus.COMPLETED assert task_result.result is not None assert task_result.result.success # Get outputs again - res = client.get(f"/v1/studies/{variant_id}/outputs", headers=admin_headers).json() - assert len(res) == 1 + res = client.get(f"/v1/studies/{variant_id}/outputs", headers=admin_headers) + assert res.status_code == 200, res.json() + outputs = res.json() + assert len(outputs) == 1 From dc0e93fb3fb50b5eb5edd3c4a838d8fab9cbc3d1 Mon Sep 17 00:00:00 2001 From: Laurent LAPORTE Date: Mon, 25 Mar 2024 21:11:05 +0100 Subject: [PATCH 3/3] test: correct implementation of the `get_matrix_id` mock --- tests/variantstudy/conftest.py | 6 +++++- tests/variantstudy/test_command_factory.py | 6 +++++- 2 files changed, 10 insertions(+), 2 deletions(-) diff --git a/tests/variantstudy/conftest.py b/tests/variantstudy/conftest.py index 1bcbb13cb1..b08851b07b 100644 --- a/tests/variantstudy/conftest.py +++ b/tests/variantstudy/conftest.py @@ -81,7 +81,11 @@ def get_matrix_id(matrix: t.Union[t.List[t.List[float]], str]) -> str: Get the matrix ID from a matrix or a matrix link. """ if isinstance(matrix, str): - return matrix.lstrip("matrix://") + # str.removeprefix() is not available in Python 3.8 + prefix = "matrix://" + if matrix.startswith(prefix): + return matrix[len(prefix) :] + return matrix elif isinstance(matrix, list): return create(matrix) else: diff --git a/tests/variantstudy/test_command_factory.py b/tests/variantstudy/test_command_factory.py index 64ec3b799f..a0324a2722 100644 --- a/tests/variantstudy/test_command_factory.py +++ b/tests/variantstudy/test_command_factory.py @@ -403,7 +403,11 @@ def setup_class(self): @pytest.mark.unit_test def test_command_factory(self, command_dto: CommandDTO): def get_matrix_id(matrix: str) -> str: - return matrix.lstrip("matrix://") + # str.removeprefix() is not available in Python 3.8 + prefix = "matrix://" + if matrix.startswith(prefix): + return matrix[len(prefix) :] + return matrix command_factory = CommandFactory( generator_matrix_constants=Mock(spec=GeneratorMatrixConstants),