From 32d6275e2909a11e7a333ea4466d3d5af7650ce3 Mon Sep 17 00:00:00 2001 From: Laurent LAPORTE Date: Fri, 1 Mar 2024 10:52:42 +0100 Subject: [PATCH] fix(study-filesystem): change `JsonReader` to return an empty dictionary when the file is missing --- .../model/filesystem/json_file_node.py | 40 ++++++++++++++++--- .../repository/filesystem/test_folder_node.py | 5 +-- 2 files changed, 36 insertions(+), 9 deletions(-) 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 ee14da91ea..46c7970dd8 100644 --- a/antarest/study/storage/rawstudy/model/filesystem/json_file_node.py +++ b/antarest/study/storage/rawstudy/model/filesystem/json_file_node.py @@ -1,6 +1,6 @@ import json from pathlib import Path -from typing import Any, Dict, Optional, cast +import typing as t from antarest.core.model import JSON from antarest.study.storage.rawstudy.ini_reader import IReader @@ -11,13 +11,41 @@ class JsonReader(IReader): - def read(self, path: Any) -> JSON: - if isinstance(path, Path): - return cast(JSON, json.loads(path.read_text(encoding="utf-8"))) - return cast(JSON, json.loads(path)) + """ + JSON file reader. + """ + + def read(self, path: t.Any) -> JSON: + content: t.Union[str, bytes] + + if isinstance(path, (Path, str)): + try: + with open(path, mode="r", encoding="utf-8") as f: + content = f.read() + except FileNotFoundError: + # If the file is missing, an empty dictionary is returned, + # to mimic the behavior of `configparser.ConfigParser`. + return {} + + elif hasattr(path, "read"): + with path: + content = path.read() + + else: # pragma: no cover + raise TypeError(repr(type(path))) + + try: + return t.cast(JSON, json.loads(content)) + except json.JSONDecodeError as exc: + err_msg = f"Failed to parse JSON file '{path}'" + raise ValueError(err_msg) from exc class JsonWriter(IniWriter): + """ + JSON file writer. + """ + def write(self, data: JSON, path: Path) -> None: with open(path, "w") as fh: json.dump(data, fh) @@ -28,6 +56,6 @@ def __init__( self, context: ContextServer, config: FileStudyTreeConfig, - types: Optional[Dict[str, Any]] = None, + types: t.Optional[t.Dict[str, t.Any]] = None, ) -> None: super().__init__(context, config, types, JsonReader(), JsonWriter()) diff --git a/tests/storage/repository/filesystem/test_folder_node.py b/tests/storage/repository/filesystem/test_folder_node.py index b44680eac6..d08360e223 100644 --- a/tests/storage/repository/filesystem/test_folder_node.py +++ b/tests/storage/repository/filesystem/test_folder_node.py @@ -121,9 +121,8 @@ def test_get_user_expansion_sensitivity_sensitivity_in(tmp_path: Path) -> None: # Add the "sensitivity" directory tmp_path.joinpath("user/expansion/sensitivity").mkdir() - # fixme: we should have `{}` - with pytest.raises(FileNotFoundError, match=r"No such file or directory"): - file_study.tree.get(url) + actual = file_study.tree.get(url) + assert actual == {} # Add the "sensitivity_in.json" file sensitivity_obj = {"epsilon": 10000.0, "projection": ["pv", "battery"], "capex": True}