From 45cb21131c34576e58897b25159f8ae10529fcdf Mon Sep 17 00:00:00 2001 From: Jean-Robin Date: Wed, 27 Nov 2024 13:35:13 +0100 Subject: [PATCH] Feature/#2017 editor (#2265) * protect writing data when data node is locked. * clean parameters for write, append and upload methods. * fix linter * Increase test coverage * Remove todos * Fix wrong import --- taipy/core/data/_file_datanode_mixin.py | 76 ++-- taipy/core/data/data_node.py | 27 +- taipy/gui_core/_context.py | 21 +- tests/core/data/test_csv_data_node.py | 21 +- tests/core/data/test_data_node.py | 112 +++++ tests/gui_core/test_context_is_editable.py | 14 +- tests/gui_core/test_context_is_readable.py | 14 +- tests/gui_core/test_context_on_file_action.py | 209 ++++++++++ .../test_context_tabular_data_edit.py | 388 ++++++++++++++++++ tests/gui_core/test_context_update_data.py | 211 ++++++++++ 10 files changed, 1033 insertions(+), 60 deletions(-) create mode 100644 tests/gui_core/test_context_on_file_action.py create mode 100644 tests/gui_core/test_context_tabular_data_edit.py create mode 100644 tests/gui_core/test_context_update_data.py diff --git a/taipy/core/data/_file_datanode_mixin.py b/taipy/core/data/_file_datanode_mixin.py index 83a5542c34..e7c12463a2 100644 --- a/taipy/core/data/_file_datanode_mixin.py +++ b/taipy/core/data/_file_datanode_mixin.py @@ -21,12 +21,19 @@ from .._entity._reload import _self_reload from ..common._utils import _normalize_path -from ..reason import InvalidUploadFile, NoFileToDownload, NotAFile, ReasonCollection, UploadFileCanNotBeRead +from ..reason import ( + DataNodeEditInProgress, + InvalidUploadFile, + NoFileToDownload, + NotAFile, + ReasonCollection, + UploadFileCanNotBeRead, +) from .data_node import DataNode from .data_node_id import Edit -class _FileDataNodeMixin(object): +class _FileDataNodeMixin: """Mixin class designed to handle file-based data nodes.""" __EXTENSION_MAP = {"csv": "csv", "excel": "xlsx", "parquet": "parquet", "pickle": "p", "json": "json"} @@ -102,53 +109,76 @@ def _get_downloadable_path(self) -> str: return "" - def _upload(self, path: str, upload_checker: Optional[Callable[[str, Any], bool]] = None) -> ReasonCollection: + def _upload(self, + path: str, + upload_checker: Optional[Callable[[str, Any], bool]] = None, + editor_id: Optional[str] = None, + comment: Optional[str] = None, + **kwargs: Any) -> ReasonCollection: """Upload a file data to the data node. Arguments: path (str): The path of the file to upload to the data node. - upload_checker (Optional[Callable[[str, Any], bool]]): A function to check if the upload is allowed. - The function takes the title of the upload data and the data itself as arguments and returns - True if the upload is allowed, otherwise False. + upload_checker (Optional[Callable[[str, Any], bool]]): A function to check if the + upload is allowed. The function takes the title of the upload data and the data + itself as arguments and returns True if the upload is allowed, otherwise False. + editor_id (Optional[str]): The ID of the user who is uploading the file. + comment (Optional[str]): A comment to add to the edit history of the data node. + **kwargs: Additional keyword arguments. These arguments are stored in the edit + history of the data node. In particular, an `editor_id` or a `comment` can be + passed. The `editor_id` is the ID of the user who is uploading the file, and the + `comment` is a comment to add to the edit history. Returns: - True if the upload was successful, otherwise False. + True if the upload was successful, the reasons why the upload was not successful + otherwise. """ from ._data_manager_factory import _DataManagerFactory - reason_collection = ReasonCollection() - - upload_path = pathlib.Path(path) + reasons = ReasonCollection() + if (editor_id + and self.edit_in_progress # type: ignore[attr-defined] + and self.editor_id != editor_id # type: ignore[attr-defined] + and (not self.editor_expiration_date # type: ignore[attr-defined] + or self.editor_expiration_date > datetime.now())): # type: ignore[attr-defined] + reasons._add_reason(self.id, DataNodeEditInProgress(self.id)) # type: ignore[attr-defined] + return reasons + up_path = pathlib.Path(path) try: - upload_data = self._read_from_path(str(upload_path)) + upload_data = self._read_from_path(str(up_path)) except Exception as err: - self.__logger.error(f"Error while uploading {upload_path.name} to data node {self.id}:") # type: ignore[attr-defined] + self.__logger.error(f"Error uploading `{up_path.name}` to data " + f"node `{self.id}`:") # type: ignore[attr-defined] self.__logger.error(f"Error: {err}") - reason_collection._add_reason(self.id, UploadFileCanNotBeRead(upload_path.name, self.id)) # type: ignore[attr-defined] - return reason_collection + reasons._add_reason(self.id, UploadFileCanNotBeRead(up_path.name, self.id)) # type: ignore[attr-defined] + return reasons if upload_checker is not None: try: - can_upload = upload_checker(upload_path.name, upload_data) + can_upload = upload_checker(up_path.name, upload_data) except Exception as err: self.__logger.error( - f"Error while checking if {upload_path.name} can be uploaded to data node {self.id}" # type: ignore[attr-defined] - f" using the upload checker {upload_checker.__name__}: {err}" - ) + f"Error with the upload checker `{upload_checker.__name__}` " + f"while checking `{up_path.name}` file for upload to the data " + f"node `{self.id}`:") # type: ignore[attr-defined] + self.__logger.error(f"Error: {err}") can_upload = False if not can_upload: - reason_collection._add_reason(self.id, InvalidUploadFile(upload_path.name, self.id)) # type: ignore[attr-defined] - return reason_collection + reasons._add_reason(self.id, InvalidUploadFile(up_path.name, self.id)) # type: ignore[attr-defined] + return reasons - shutil.copy(upload_path, self.path) + shutil.copy(up_path, self.path) - self.track_edit(timestamp=datetime.now()) # type: ignore[attr-defined] + self.track_edit(timestamp=datetime.now(), # type: ignore[attr-defined] + editor_id=editor_id, + comment=comment, **kwargs) self.unlock_edit() # type: ignore[attr-defined] + _DataManagerFactory._build_manager()._set(self) # type: ignore[arg-type] - return reason_collection + return reasons def _read_from_path(self, path: Optional[str] = None, **read_kwargs) -> Any: raise NotImplementedError diff --git a/taipy/core/data/data_node.py b/taipy/core/data/data_node.py index 06da7cddf5..ad15e61072 100644 --- a/taipy/core/data/data_node.py +++ b/taipy/core/data/data_node.py @@ -421,35 +421,52 @@ def read(self) -> Any: ) return None - def append(self, data, editor_id: Optional[str] = None, **kwargs: Any): + def append(self, data, editor_id: Optional[str] = None, comment: Optional[str] = None, **kwargs: Any): """Append some data to this data node. Arguments: data (Any): The data to write to this data node. editor_id (str): An optional identifier of the editor. + comment (str): An optional comment to attach to the edit document. **kwargs (Any): Extra information to attach to the edit document corresponding to this write. """ from ._data_manager_factory import _DataManagerFactory - + if (editor_id + and self.edit_in_progress + and self.editor_id != editor_id + and (not self.editor_expiration_date or self.editor_expiration_date > datetime.now())): + raise DataNodeIsBeingEdited(self.id, self.editor_id) self._append(data) - self.track_edit(editor_id=editor_id, **kwargs) + self.track_edit(editor_id=editor_id, comment=comment, **kwargs) self.unlock_edit() _DataManagerFactory._build_manager()._set(self) - def write(self, data, job_id: Optional[JobId] = None, **kwargs: Any): + def write(self, + data, + job_id: Optional[JobId] = None, + editor_id: Optional[str] = None, + comment: Optional[str] = None, + **kwargs: Any): """Write some data to this data node. Arguments: data (Any): The data to write to this data node. job_id (JobId): An optional identifier of the job writing the data. + editor_id (str): An optional identifier of the editor writing the data. + comment (str): An optional comment to attach to the edit document. **kwargs (Any): Extra information to attach to the edit document corresponding to this write. """ from ._data_manager_factory import _DataManagerFactory + if (editor_id + and self.edit_in_progress + and self.editor_id != editor_id + and (not self.editor_expiration_date or self.editor_expiration_date > datetime.now())): + raise DataNodeIsBeingEdited(self.id, self.editor_id) self._write(data) - self.track_edit(job_id=job_id, **kwargs) + self.track_edit(job_id=job_id, editor_id=editor_id, comment=comment, **kwargs) self.unlock_edit() _DataManagerFactory._build_manager()._set(self) diff --git a/taipy/gui_core/_context.py b/taipy/gui_core/_context.py index ad7897e18a..739929ec5e 100644 --- a/taipy/gui_core/_context.py +++ b/taipy/gui_core/_context.py @@ -1020,10 +1020,10 @@ def get_data_node_history(self, id: str): def __check_readable_editable(self, state: State, id: str, ent_type: str, var: t.Optional[str]): if not (reason := is_readable(t.cast(ScenarioId, id))): - _GuiCoreContext.__assign_var(state, var, f"{ent_type} {id} is not readable: {_get_reason(reason)}.") + _GuiCoreContext.__assign_var(state, var, f"{ent_type} {id} is not readable: {_get_reason(reason)}") return False if not (reason := is_editable(t.cast(ScenarioId, id))): - _GuiCoreContext.__assign_var(state, var, f"{ent_type} {id} is not editable: {_get_reason(reason)}.") + _GuiCoreContext.__assign_var(state, var, f"{ent_type} {id} is not editable: {_get_reason(reason)}") return False return True @@ -1033,7 +1033,7 @@ def update_data(self, state: State, id: str, payload: t.Dict[str, str]): if args is None or not isinstance(args, list) or len(args) < 1 or not isinstance(args[0], dict): return data = t.cast(dict, args[0]) - error_var = payload.get("error_id") + error_var = data.get("error_id") entity_id = t.cast(str, data.get(_GuiCoreContext.__PROP_ENTITY_ID)) if not self.__check_readable_editable(state, entity_id, "Data node", error_var): return @@ -1049,9 +1049,9 @@ def update_data(self, state: State, id: str, payload: t.Dict[str, str]): else float(val) if data.get("type") == "float" else data.get("value"), + editor_id=self.gui._get_client_id(), comment=t.cast(dict, data.get(_GuiCoreContext.__PROP_ENTITY_COMMENT)), ) - entity.unlock_edit(self.gui._get_client_id()) _GuiCoreContext.__assign_var(state, error_var, "") except Exception as e: _GuiCoreContext.__assign_var(state, error_var, f"Error updating Data node value. {e}") @@ -1135,7 +1135,9 @@ def tabular_data_edit(self, state: State, var_name: str, payload: dict): # noqa "Error updating data node tabular value: type does not support at[] indexer.", ) if new_data is not None: - datanode.write(new_data, comment=user_data.get(_GuiCoreContext.__PROP_ENTITY_COMMENT)) + datanode.write(new_data, + editor_id=self.gui._get_client_id(), + comment=user_data.get(_GuiCoreContext.__PROP_ENTITY_COMMENT)) _GuiCoreContext.__assign_var(state, error_var, "") except Exception as e: _GuiCoreContext.__assign_var(state, error_var, f"Error updating data node tabular value. {e}") @@ -1222,6 +1224,8 @@ def on_dag_select(self, state: State, id: str, payload: t.Dict[str, str]): def on_file_action(self, state: State, id: str, payload: t.Dict[str, t.Any]): args = t.cast(list, payload.get("args")) + if args is None or not isinstance(args, list) or len(args) < 1 or not isinstance(args[0], dict): + return act_payload = t.cast(t.Dict[str, str], args[0]) dn_id = t.cast(DataNodeId, act_payload.get("id")) error_id = act_payload.get("error_id", "") @@ -1229,11 +1233,10 @@ def on_file_action(self, state: State, id: str, payload: t.Dict[str, t.Any]): try: dn = t.cast(_FileDataNodeMixin, core_get(dn_id)) if act_payload.get("action") == "export": - path = dn._get_downloadable_path() - if path: + if reason := dn.is_downloadable(): + path = dn._get_downloadable_path() self.gui._download(Path(path), dn_id) else: - reason = dn.is_downloadable() state.assign( error_id, "Data unavailable: " @@ -1246,6 +1249,8 @@ def on_file_action(self, state: State, id: str, payload: t.Dict[str, t.Any]): reason := dn._upload( act_payload.get("path", ""), t.cast(t.Callable[[str, t.Any], bool], checker) if callable(checker) else None, + editor_id=self.gui._get_client_id(), + comment=None ) ): state.assign(error_id, f"Data unavailable: {reason.reasons}") diff --git a/tests/core/data/test_csv_data_node.py b/tests/core/data/test_csv_data_node.py index 95b78520c6..360fd62a56 100644 --- a/tests/core/data/test_csv_data_node.py +++ b/tests/core/data/test_csv_data_node.py @@ -251,6 +251,22 @@ def test_upload(self, csv_file, tmpdir_factory): assert dn.last_edit_date > old_last_edit_date assert dn.path == _normalize_path(old_csv_path) # The path of the dn should not change + def test_upload_fails_if_data_node_locked(self, csv_file, tmpdir_factory): + old_csv_path = tmpdir_factory.mktemp("data").join("df.csv").strpath + old_data = pd.DataFrame([{"a": 0, "b": 1, "c": 2}, {"a": 3, "b": 4, "c": 5}]) + + dn = CSVDataNode("foo", Scope.SCENARIO, properties={"path": old_csv_path, "exposed_type": "pandas"}) + dn.write(old_data) + upload_content = pd.read_csv(csv_file) + dn.lock_edit("editor_id_1") + + reasons = dn._upload(csv_file, editor_id="editor_id_2") + assert not reasons + + assert dn._upload(csv_file, editor_id="editor_id_1") + + assert_frame_equal(dn.read(), upload_content) # The content of the dn should change to the uploaded content + def test_upload_with_upload_check_with_exception(self, csv_file, tmpdir_factory, caplog): old_csv_path = tmpdir_factory.mktemp("data").join("df.csv").strpath dn = CSVDataNode("foo", Scope.SCENARIO, properties={"path": old_csv_path, "exposed_type": "pandas"}) @@ -261,8 +277,9 @@ def check_with_exception(upload_path, upload_data): reasons = dn._upload(csv_file, upload_checker=check_with_exception) assert bool(reasons) is False assert ( - f"Error while checking if df.csv can be uploaded to data node {dn.id} using " - "the upload checker check_with_exception: An error with check_with_exception" in caplog.text + f"Error with the upload checker `check_with_exception` " + f"while checking `df.csv` file for upload to the data " + f"node `{dn.id}`:" in caplog.text ) def test_upload_with_upload_check_pandas(self, csv_file, tmpdir_factory): diff --git a/tests/core/data/test_data_node.py b/tests/core/data/test_data_node.py index 9959cf36de..5442b63065 100644 --- a/tests/core/data/test_data_node.py +++ b/tests/core/data/test_data_node.py @@ -14,6 +14,8 @@ from time import sleep from unittest import mock +import freezegun +import pandas as pd import pytest import taipy.core as tp @@ -752,6 +754,116 @@ def test_change_data_node_name(self): dn.properties["name"] = "baz" assert dn.name == "baz" + def test_locked_data_node_write_should_fail_with_wrong_editor(self): + dn_config = Config.configure_data_node("A") + dn = _DataManager._bulk_get_or_create([dn_config])[dn_config] + dn.lock_edit("editor_1") + + # Should raise exception for wrong editor + with pytest.raises(DataNodeIsBeingEdited): + dn.write("data", editor_id="editor_2") + + # Should succeed with correct editor + dn.write("data", editor_id="editor_1") + assert dn.read() == "data" + + def test_locked_data_node_write_should_fail_before_expiration_date_and_succeed_after(self): + dn_config = Config.configure_data_node("A") + dn = _DataManager._bulk_get_or_create([dn_config])[dn_config] + + lock_time = datetime.now() + with freezegun.freeze_time(lock_time): + dn.lock_edit("editor_1") + + with freezegun.freeze_time(lock_time + timedelta(minutes=29)): + # Should raise exception for wrong editor and expiration date NOT passed + with pytest.raises(DataNodeIsBeingEdited): + dn.write("data", editor_id="editor_2") + + with freezegun.freeze_time(lock_time + timedelta(minutes=31)): + # Should succeed with wrong editor but expiration date passed + dn.write("data", editor_id="editor_2") + assert dn.read() == "data" + + def test_locked_data_node_append_should_fail_with_wrong_editor(self): + dn_config = Config.configure_csv_data_node("A") + dn = _DataManager._bulk_get_or_create([dn_config])[dn_config] + first_line = pd.DataFrame(data={'col1': [1], 'col2': [3]}) + second_line = pd.DataFrame(data={'col1': [2], 'col2': [4]}) + data = pd.DataFrame(data={'col1': [1, 2], 'col2': [3, 4]}) + dn.write(first_line) + assert first_line.equals(dn.read()) + + dn.lock_edit("editor_1") + + with pytest.raises(DataNodeIsBeingEdited): + dn.append(second_line, editor_id="editor_2") + + dn.append(second_line, editor_id="editor_1") + assert dn.read().equals(data) + + def test_locked_data_node_append_should_fail_before_expiration_date_and_succeed_after(self): + dn_config = Config.configure_csv_data_node("A") + dn = _DataManager._bulk_get_or_create([dn_config])[dn_config] + first_line = pd.DataFrame(data={'col1': [1], 'col2': [3]}) + second_line = pd.DataFrame(data={'col1': [2], 'col2': [4]}) + data = pd.DataFrame(data={'col1': [1, 2], 'col2': [3, 4]}) + dn.write(first_line) + assert first_line.equals(dn.read()) + + lock_time = datetime.now() + with freezegun.freeze_time(lock_time): + dn.lock_edit("editor_1") + + with freezegun.freeze_time(lock_time + timedelta(minutes=29)): + # Should raise exception for wrong editor and expiration date NOT passed + with pytest.raises(DataNodeIsBeingEdited): + dn.append(second_line, editor_id="editor_2") + + with freezegun.freeze_time(lock_time + timedelta(minutes=31)): + # Should succeed with wrong editor but expiration date passed + dn.append(second_line, editor_id="editor_2") + assert dn.read().equals(data) + + def test_orchestrator_write_without_editor_id(self): + dn_config = Config.configure_data_node("A") + dn = _DataManager._bulk_get_or_create([dn_config])[dn_config] + dn.lock_edit("editor_1") + + # Orchestrator write without editor_id should succeed + dn.write("orchestrator_data") + assert dn.read() == "orchestrator_data" + + def test_editor_fails_writing_a_data_node_locked_by_orchestrator(self): + dn_config = Config.configure_data_node("A") + dn = _DataManager._bulk_get_or_create([dn_config])[dn_config] + dn.lock_edit() # Locked by orchestrator + + with pytest.raises(DataNodeIsBeingEdited): + dn.write("data", editor_id="editor_1") + + # Orchestrator write without editor_id should succeed + dn.write("orchestrator_data", job_id=JobId("job_1")) + assert dn.read() == "orchestrator_data" + + def test_editor_fails_appending_a_data_node_locked_by_orchestrator(self): + dn_config = Config.configure_csv_data_node("A") + dn = _DataManager._bulk_get_or_create([dn_config])[dn_config] + first_line = pd.DataFrame(data={'col1': [1], 'col2': [3]}) + second_line = pd.DataFrame(data={'col1': [2], 'col2': [4]}) + data = pd.DataFrame(data={'col1': [1, 2], 'col2': [3, 4]}) + dn.write(first_line) + assert first_line.equals(dn.read()) + dn = _DataManager._bulk_get_or_create([dn_config])[dn_config] + dn.lock_edit() # Locked by orchestrator + + with pytest.raises(DataNodeIsBeingEdited): + dn.append(second_line, editor_id="editor_1") + assert dn.read().equals(first_line) + + dn.append(second_line, job_id=JobId("job_1")) + assert dn.read().equals(data) + def test_track_edit(self): dn_config = Config.configure_data_node("A") data_node = _DataManager._bulk_get_or_create([dn_config])[dn_config] diff --git a/tests/gui_core/test_context_is_editable.py b/tests/gui_core/test_context_is_editable.py index 623d83fbc9..bc0f9a6161 100644 --- a/tests/gui_core/test_context_is_editable.py +++ b/tests/gui_core/test_context_is_editable.py @@ -254,10 +254,9 @@ def test_update_data(self): MockState(assign=assign), "", { - "args": [ - {"id": a_datanode.id}, - ], - "error_id": "error_var", + "args": [{ + "id": a_datanode.id, + "error_id": "error_var"}], }, ) assign.assert_called() @@ -269,12 +268,7 @@ def test_update_data(self): gui_core_context.update_data( MockState(assign=assign), "", - { - "args": [ - {"id": a_datanode.id}, - ], - "error_id": "error_var", - }, + {"args": [{"id": a_datanode.id, "error_id": "error_var"}]}, ) assign.assert_called_once() assert assign.call_args.args[0] == "error_var" diff --git a/tests/gui_core/test_context_is_readable.py b/tests/gui_core/test_context_is_readable.py index 3fe9f22184..e495a238ca 100644 --- a/tests/gui_core/test_context_is_readable.py +++ b/tests/gui_core/test_context_is_readable.py @@ -398,12 +398,7 @@ def test_update_data(self): gui_core_context.update_data( MockState(assign=assign), "", - t.cast(dict, { - "args": [ - {"id": a_datanode.id}, - ], - "error_id": "error_var", - }), + t.cast(dict, {"args": [{"id": a_datanode.id, "error_id": "error_var"}]}) ) assign.assert_called() assert assign.call_args_list[0].args[0] == "error_var" @@ -414,12 +409,7 @@ def test_update_data(self): gui_core_context.update_data( MockState(assign=assign), "", - t.cast(dict, { - "args": [ - {"id": a_datanode.id}, - ], - "error_id": "error_var", - }), + t.cast(dict, {"args": [{"id": a_datanode.id, "error_id": "error_var"}]}) ) assign.assert_called_once() assert assign.call_args.args[0] == "error_var" diff --git a/tests/gui_core/test_context_on_file_action.py b/tests/gui_core/test_context_on_file_action.py new file mode 100644 index 0000000000..3605532734 --- /dev/null +++ b/tests/gui_core/test_context_on_file_action.py @@ -0,0 +1,209 @@ +# Copyright 2021-2024 Avaiga Private Limited +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with +# the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on +# an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the +# specific language governing permissions and limitations under the License. + +import typing as t +from pathlib import Path +from unittest.mock import Mock, patch + +import pytest + +from taipy import DataNode, Gui, Scope +from taipy.core.data import PickleDataNode +from taipy.core.data._data_manager_factory import _DataManagerFactory +from taipy.core.data._file_datanode_mixin import _FileDataNodeMixin +from taipy.core.reason import Reason, ReasonCollection +from taipy.gui_core._context import _GuiCoreContext + +dn = PickleDataNode("dn_config_id", + scope = Scope.GLOBAL, + properties={"default_path": "pa/th"}) + +def core_get(entity_id): + if entity_id == dn.id: + return dn + return None + + +def not_downloadable (): + return ReasonCollection()._add_reason(dn.id, Reason("foo")) + + +def downloadable(): + return ReasonCollection() + + +def not_readable(entity_id): + return ReasonCollection()._add_reason(entity_id, Reason("foo")) + + +def readable(entity_id): + return ReasonCollection() + + +def mock_checker(**kwargs): + return True + + +def check_fails(**kwargs): + raise Exception("Failed") + + +def upload_fails (a, b, editor_id, comment): + return ReasonCollection()._add_reason(dn.id, Reason("bar")) + + +def download_fails (a, b, editor_id, comment): + return ReasonCollection()._add_reason(dn.id, Reason("bar")) + + +class MockState: + def __init__(self, **kwargs) -> None: + self.assign = kwargs.get("assign") + + +class TestGuiCoreContext_on_file_action: + + @pytest.fixture(scope="class", autouse=True) + def set_entities(self): + _DataManagerFactory._build_manager()._set(dn) + + def test_does_not_fail_if_wrong_args(self): + gui_core_context = _GuiCoreContext(Mock(Gui)) + gui_core_context.on_file_action(state=Mock(), id="", payload={}) + gui_core_context.on_file_action(state=Mock(), id="", payload={"args": "wrong_args"}) + gui_core_context.on_file_action(state=Mock(), id="", payload={"args": ["wrong_args"]}) + + def test_datanode_not_readable(self): + with patch("taipy.gui_core._context.is_readable", side_effect=not_readable): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + mockGui = Mock(Gui) + mockGui._get_client_id = lambda: "a_client_id" + gui_core_context = _GuiCoreContext(mockGui) + assign = Mock() + gui_core_context.on_file_action( + state=MockState(assign=assign), + id="", + payload={"args": [{"id": dn.id, "error_id": "error_var"}]}, + ) + mock_core_get.assert_not_called() + mock_write.assert_not_called() + assign.assert_called_once_with("error_var", "foo.") + + def test_upload_file_without_checker(self): + with patch("taipy.gui_core._context.is_readable", side_effect=readable): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(_FileDataNodeMixin, "_upload") as mock_upload: + mockGui = Mock(Gui) + mockGui._get_client_id = lambda: "a_client_id" + gui_core_context = _GuiCoreContext(mockGui) + assign = Mock() + gui_core_context.on_file_action( + state=MockState(assign=assign), + id="", + payload={"args": [{"id": dn.id, "error_id": "error_var", "path": "pa/th"}]}, + ) + mock_core_get.assert_called_once_with(dn.id) + mock_upload.assert_called_once_with( + "pa/th", + None, + editor_id="a_client_id", + comment=None) + assign.assert_not_called() + + def test_upload_file_with_checker(self): + with patch("taipy.gui_core._context.is_readable", side_effect=readable): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(_FileDataNodeMixin, "_upload") as mock_upload: + mockGui = Mock(Gui) + mockGui._get_client_id = lambda: "a_client_id" + mockGui._get_user_function = lambda _ : _ + gui_core_context = _GuiCoreContext(mockGui) + assign = Mock() + gui_core_context.on_file_action( + state=MockState(assign=assign), + id="", + payload={"args": [ + {"id": dn.id, "error_id": "error_var", "path": "pa/th", "upload_check": mock_checker}]}, + ) + mock_core_get.assert_called_once_with(dn.id) + mock_upload.assert_called_once_with( + "pa/th", + t.cast(t.Callable[[str, t.Any], bool], mock_checker), + editor_id="a_client_id", + comment=None) + assign.assert_not_called() + + def test_upload_file_with_failing_checker(self): + with patch("taipy.gui_core._context.is_readable", side_effect=readable): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(_FileDataNodeMixin, "_upload", side_effect=upload_fails) as mock_upload: + mockGui = Mock(Gui) + mockGui._get_client_id = lambda: "a_client_id" + mockGui._get_user_function = lambda _ : _ + gui_core_context = _GuiCoreContext(mockGui) + assign = Mock() + gui_core_context.on_file_action( + state=MockState(assign=assign), + id="", + payload={"args": [ + {"id": dn.id, "error_id": "error_var", "path": "pa/th", "upload_check": check_fails}]}, + ) + mock_core_get.assert_called_once_with(dn.id) + mock_upload.assert_called_once_with( + "pa/th", + t.cast(t.Callable[[str, t.Any], bool], check_fails), + editor_id="a_client_id", + comment=None) + assign.assert_called_once_with("error_var", "Data unavailable: bar.") + + def test_download_file_not_downloadable(self): + with patch.object(_FileDataNodeMixin, "is_downloadable", side_effect=not_downloadable): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(_FileDataNodeMixin, "_get_downloadable_path") as mock_download: + mockGui = Mock(Gui) + mockGui._get_client_id = lambda: "a_client_id" + mockGui._get_user_function = lambda _ : _ + gui_core_context = _GuiCoreContext(mockGui) + assign = Mock() + gui_core_context.on_file_action( + state=MockState(assign=assign), + id="", + payload={"args": [ + {"id": dn.id, + "action": "export", + "error_id": "error_var"}]}, + ) + mock_core_get.assert_called_once_with(dn.id) + mock_download.assert_not_called() + assign.assert_called_once_with("error_var", "Data unavailable: foo.") + + def test_download(self): + with patch.object(_FileDataNodeMixin, "is_downloadable", side_effect=downloadable): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(_FileDataNodeMixin, "_get_downloadable_path") as mock_download: + mockGui = Mock(Gui) + mockGui._get_client_id = lambda: "a_client_id" + mockGui._download.return_value = None + gui_core_context = _GuiCoreContext(mockGui) + assign = Mock() + gui_core_context.on_file_action( + state=MockState(assign=assign), + id="", + payload={"args": [ + {"id": dn.id, + "action": "export", + "error_id": "error_var"}]}, + ) + mock_core_get.assert_called_once_with(dn.id) + mock_download.assert_called_once() + mockGui._download.assert_called_once_with(Path(dn._get_downloadable_path()), dn.id) + assign.assert_not_called() diff --git a/tests/gui_core/test_context_tabular_data_edit.py b/tests/gui_core/test_context_tabular_data_edit.py new file mode 100644 index 0000000000..60f89c6541 --- /dev/null +++ b/tests/gui_core/test_context_tabular_data_edit.py @@ -0,0 +1,388 @@ +# Copyright 2021-2024 Avaiga Private Limited +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with +# the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on +# an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the +# specific language governing permissions and limitations under the License. + +from unittest.mock import Mock, patch + +import pandas as pd + +from taipy import DataNode, Gui +from taipy.common.config.common.scope import Scope +from taipy.core.data._data_manager_factory import _DataManagerFactory +from taipy.core.data.pickle import PickleDataNode +from taipy.core.reason import Reason, ReasonCollection +from taipy.gui_core._context import _GuiCoreContext + +dn = PickleDataNode("dn_config_id", scope = Scope.GLOBAL) + + +def core_get(entity_id): + if entity_id == dn.id: + return dn + return None + + +def is_false(entity_id): + return ReasonCollection()._add_reason(entity_id, Reason("foo")) + + +def is_true(entity_id): + return True + +def fails(**kwargs): + raise Exception("Failed") + + +class MockState: + def __init__(self, **kwargs) -> None: + self.assign = kwargs.get("assign") + + +class TestGuiCoreContext_update_data: + + def test_do_not_edit_tabular_data_if_not_readable(self): + _DataManagerFactory._build_manager()._set(dn) + with patch("taipy.gui_core._context.is_readable", side_effect=is_false): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + assign = self.__call_update_data() + + mock_core_get.assert_not_called() + mock_write.assert_not_called() + assign.assert_called_once_with("error_var", f"Data node {dn.id} is not readable: foo.") + + def test_do_not_edit_tabular_data_if_not_editable(self): + _DataManagerFactory._build_manager()._set(dn) + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_false): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + assign = self.__call_update_data() + + mock_core_get.assert_not_called() + mock_write.assert_not_called() + assign.assert_called_once_with("error_var", f"Data node {dn.id} is not editable: foo.") + + def test_edit_pandas_data(self): + dn.write(pd.DataFrame({"a": [1, 2, 3], "b": [4, 5, 6]})) + idx = 0 + col = "a" + new_value = 100 + new_data = pd.DataFrame({"a": [new_value, 2, 3], "b": [4, 5, 6]}) + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + assign = self.__call_update_data(col, idx, new_value) + + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_called_once() + # Cannot use the following line because of the pandas DataFrame comparison + # mock_write.assert_called_once_with(new_data, editor_id="a_client_id", comment=None + # Instead, we will compare the arguments of the call manually + assert mock_write.call_args_list[0].args[0].equals(new_data) + assert mock_write.call_args_list[0].kwargs["editor_id"] == "a_client_id" + assert mock_write.call_args_list[0].kwargs["comment"] is None + assign.assert_called_once_with("error_var", "") + + def __call_update_data(self, col=None, idx=None, new_value=None): + mockGui = Mock(Gui) + mockGui._get_client_id = lambda: "a_client_id" + gui_core_context = _GuiCoreContext(mockGui) + payload = {"user_data": {"dn_id": dn.id}, "error_id": "error_var"} + if idx is not None: + payload["index"] = idx + if col is not None: + payload["col"] = col + if new_value is not None: + payload["value"] = new_value + assign = Mock() + gui_core_context.tabular_data_edit( + state=MockState(assign=assign), + var_name="", + payload=payload, + ) + return assign + + def test_edit_pandas_wrong_idx(self): + data = pd.DataFrame({"a": [1, 2, 3], "b": [4, 5, 6]}) + dn.write(data) + idx = 5 + col = "a" + new_value = 100 + new_data = data.copy() + new_data.at[idx, col] = new_value + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + + assign = self.__call_update_data(col, idx, new_value) + + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_called_once() + # Cannot use the following line because of the pandas DataFrame comparison + # mock_write.assert_called_once_with(new_data, editor_id="a_client_id", comment=None + # Instead, we will compare the arguments of the call manually + assert mock_write.call_args_list[0].args[0].equals(new_data) + assert mock_write.call_args_list[0].kwargs["editor_id"] == "a_client_id" + assert mock_write.call_args_list[0].kwargs["comment"] is None + assign.assert_called_once_with("error_var", "") + + def test_edit_pandas_wrong_col(self): + data = pd.DataFrame({"a": [1, 2, 3], "b": [4, 5, 6]}) + dn.write(data) + idx = 0 + col = "c" + new_value = 100 + new_data = data.copy() + new_data.at[idx, col] = new_value + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + assign = self.__call_update_data(col, idx, new_value) + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_called_once() + # Cannot use the following line because of the pandas DataFrame comparison + # mock_write.assert_called_once_with(new_data, editor_id="a_client_id", comment=None + # Instead, we will compare the arguments of the call manually + assert mock_write.call_args_list[0].args[0].equals(new_data) + assert mock_write.call_args_list[0].kwargs["editor_id"] == "a_client_id" + assert mock_write.call_args_list[0].kwargs["comment"] is None + assign.assert_called_once_with("error_var", "") + + def test_edit_pandas_series(self): + data = pd.Series([1, 2, 3]) + dn.write(data) + idx = 0 + col = "WHATEVER" + new_value = 100 + new_data = pd.Series([100, 2, 3]) + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + assign = self.__call_update_data(col, idx, new_value) + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_called_once() + # Cannot use the following line because of the pandas Series comparison + # mock_write.assert_called_once_with(new_data, editor_id="a_client_id", comment=None + # Instead, we will compare the arguments of the call manually + assert mock_write.call_args_list[0].args[0].equals(new_data) + assert mock_write.call_args_list[0].kwargs["editor_id"] == "a_client_id" + assert mock_write.call_args_list[0].kwargs["comment"] is None + assign.assert_called_once_with("error_var", "") + + def test_edit_pandas_series_wrong_idx(self): + data = pd.Series([1, 2, 3]) + dn.write(data) + idx = 5 + col = "WHATEVER" + new_value = 100 + new_data = data.copy() + new_data.at[idx] = new_value + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + assign = self.__call_update_data(col, idx, new_value) + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_called_once() + # Cannot use the following line because of the pandas Series comparison + # mock_write.assert_called_once_with(new_data, editor_id="a_client_id", comment=None + # Instead, we will compare the arguments of the call manually + assert mock_write.call_args_list[0].args[0].equals(new_data) + assert mock_write.call_args_list[0].kwargs["editor_id"] == "a_client_id" + assert mock_write.call_args_list[0].kwargs["comment"] is None + assign.assert_called_once_with("error_var", "") + + def test_edit_dict(self): + data = {"a": [1, 2, 3], "b": [4, 5, 6]} + dn.write(data) + idx = 0 + col = "a" + new_value = 100 + new_data = {"a": [100, 2, 3], "b": [4, 5, 6]} + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + assign = self.__call_update_data(col, idx, new_value) + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_called_once_with(new_data, editor_id="a_client_id", comment=None) + assign.assert_called_once_with("error_var", "") + + def test_edit_dict_wrong_idx(self): + data = {"a": [1, 2, 3], "b": [4, 5, 6]} + dn.write(data) + idx = 5 + col = "a" + new_value = 100 + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + assign = self.__call_update_data(col, idx, new_value) + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_not_called() + assign.assert_called_once_with( + "error_var", + "Error updating data node tabular value. list assignment index out of range") + + def test_edit_dict_wrong_col(self): + data = {"a": [1, 2, 3], "b": [4, 5, 6]} + dn.write(data) + idx = 0 + col = "c" + new_value = 100 + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + assign = self.__call_update_data(col, idx, new_value) + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_not_called() + assign.assert_called_once_with( + "error_var", + "Error updating Data node: dict values must be list or tuple.") + + def test_edit_dict_of_tuples(self): + data = {"a": (1, 2, 3), "b": (4, 5, 6)} + dn.write(data) + idx = 0 + col = "a" + new_value = 100 + new_data = {"a": (100, 2, 3), "b": (4, 5, 6)} + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + assign = self.__call_update_data(col, idx, new_value) + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_called_once_with(new_data, editor_id="a_client_id", comment=None) + assign.assert_called_once_with("error_var", "") + + def test_edit_dict_of_tuples_wrong_idx(self): + data = {"a": (1, 2, 3), "b": (4, 5, 6)} + dn.write(data) + idx = 5 + col = "a" + new_value = 100 + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + assign = self.__call_update_data(col, idx, new_value) + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_not_called() + assign.assert_called_once_with( + "error_var", + "Error updating data node tabular value. list assignment index out of range") + + def test_edit_dict_of_tuples_wrong_col(self): + data = {"a": (1, 2, 3), "b": (4, 5, 6)} + dn.write(data) + idx = 0 + col = "c" + new_value = 100 + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + assign = self.__call_update_data(col, idx, new_value) + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_not_called() + assign.assert_called_once_with( + "error_var", + "Error updating Data node: dict values must be list or tuple.") + + def test_edit_wrong_dict(self): + data = {"a": 1, "b": 2} + dn.write(data) + idx = 0 + col = "a" + new_value = 100 + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + assign = self.__call_update_data(col, idx, new_value) + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_not_called() + assign.assert_called_once_with( + "error_var", + "Error updating Data node: dict values must be list or tuple.") + + def test_edit_list(self): + data = [[1, 2, 3], [4, 5, 6]] + dn.write(data) + idx = 0 + col = 1 + new_value = 100 + new_data = [[1, 100, 3], [4, 5, 6]] + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + assign = self.__call_update_data(col, idx, new_value) + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_called_once_with(new_data, editor_id="a_client_id", comment=None) + assign.assert_called_once_with("error_var", "") + + def test_edit_list_wrong_idx(self): + data = [[1, 2, 3], [4, 5, 6]] + dn.write(data) + idx = 5 + col = 0 + new_value = 100 + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + assign = self.__call_update_data(col, idx, new_value) + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_not_called() + assign.assert_called_once_with( + "error_var", + "Error updating data node tabular value. list index out of range") + + def test_edit_list_wrong_col(self): + data = [[1, 2, 3], [4, 5, 6]] + dn.write(data) + idx = 0 + col = 5 + new_value = 100 + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + assign = self.__call_update_data(col, idx, new_value) + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_not_called() + assign.assert_called_once_with( + "error_var", + "Error updating data node tabular value. list assignment index out of range") + + def test_edit_tuple(self): + data = ([1, 2, 3], [4, 5, 6]) + dn.write(data) + idx = 0 + col = 1 + new_value = 100 + new_data = ([1, 100, 3], [4, 5, 6]) + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + assign = self.__call_update_data(col, idx, new_value) + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_called_once_with(new_data, editor_id="a_client_id", comment=None) + assign.assert_called_once_with("error_var", "") diff --git a/tests/gui_core/test_context_update_data.py b/tests/gui_core/test_context_update_data.py new file mode 100644 index 0000000000..7d8923ed03 --- /dev/null +++ b/tests/gui_core/test_context_update_data.py @@ -0,0 +1,211 @@ +# Copyright 2021-2024 Avaiga Private Limited +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with +# the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on +# an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the +# specific language governing permissions and limitations under the License. +from datetime import datetime +from unittest.mock import Mock, patch + +import pytest + +from taipy import DataNode, Gui +from taipy.common.config.common.scope import Scope +from taipy.core.data._data_manager_factory import _DataManagerFactory +from taipy.core.data.pickle import PickleDataNode +from taipy.core.reason import Reason, ReasonCollection +from taipy.gui_core._context import _GuiCoreContext + +dn = PickleDataNode("data_node_config_id", Scope.SCENARIO) + + +def core_get(entity_id): + if entity_id == dn.id: + return dn + return None + + +def is_false(entity_id): + return ReasonCollection()._add_reason(entity_id, Reason("foo")) + + +def is_true(entity_id): + return True + +def fails(**kwargs): + raise Exception("Failed") + + +class MockState: + def __init__(self, **kwargs) -> None: + self.assign = kwargs.get("assign") + + +class TestGuiCoreContext_update_data: + + @pytest.fixture(scope="class", autouse=True) + def set_entities(self): + _DataManagerFactory._build_manager()._set(dn) + + def test_does_not_fail_if_wrong_args(self): + gui_core_context = _GuiCoreContext(Mock(Gui)) + gui_core_context.update_data(state=Mock(), id="", payload={}) + gui_core_context.update_data(state=Mock(), id="", payload={"args": "wrong_args"}) + gui_core_context.update_data(state=Mock(), id="", payload={"args": ["wrong_args"]}) + + def test_do_not_update_data_if_not_readable(self): + with patch("taipy.gui_core._context.is_readable", side_effect=is_false): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + mockGui = Mock(Gui) + mockGui._get_client_id = lambda: "a_client_id" + gui_core_context = _GuiCoreContext(mockGui) + assign = Mock() + gui_core_context.update_data( + state=MockState(assign=assign), + id="", + payload={"args": [{"id": dn.id,"error_id": "error_var"}]}, + ) + mock_core_get.assert_not_called() + mock_write.assert_not_called() + assign.assert_called_once_with("error_var", f"Data node {dn.id} is not readable: foo.") + + def test_do_not_update_data_if_not_editable(self): + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_false): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + mockGui = Mock(Gui) + mockGui._get_client_id = lambda: "a_client_id" + gui_core_context = _GuiCoreContext(mockGui) + assign = Mock() + gui_core_context.update_data( + state=MockState(assign=assign), + id="", + payload={"args": [{"id": dn.id,"error_id": "error_var"}]}, + ) + mock_core_get.assert_not_called() + mock_write.assert_not_called() + assign.assert_called_once_with("error_var", f"Data node {dn.id} is not editable: foo.") + + def test_write_str_data_with_editor_and_comment(self): + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + mockGui = Mock(Gui) + mockGui._get_client_id = lambda: "a_client_id" + gui_core_context = _GuiCoreContext(mockGui) + assign = Mock() + gui_core_context.update_data( + state=MockState(assign=assign), + id="", + payload={ + "args": [{ + "id": dn.id, + "value": "data to write", + "comment": "The comment", + "error_id": "error_var"}], + }, + ) + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_called_once_with("data to write", + editor_id="a_client_id", + comment="The comment") + assign.assert_called_once_with("error_var", "") + + def test_write_date_data_with_editor_and_comment(self): + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + mockGui = Mock(Gui) + mockGui._get_client_id = lambda: "a_client_id" + gui_core_context = _GuiCoreContext(mockGui) + assign = Mock() + date = datetime(2000, 1, 1, 0, 0, 0) + gui_core_context.update_data( + state=MockState(assign=assign), + id="", + payload={ + "args": [ + { + "id": dn.id, + "value": "2000-01-01 00:00:00", + "type": "date", + "comment": "The comment", + "error_id": "error_var" + }], + }, + ) + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_called_once_with(date, + editor_id="a_client_id", + comment="The comment") + assign.assert_called_once_with("error_var", "") + + def test_write_int_data_with_editor_and_comment(self): + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + mockGui = Mock(Gui) + mockGui._get_client_id = lambda: "a_client_id" + gui_core_context = _GuiCoreContext(mockGui) + assign = Mock() + gui_core_context.update_data( + state=MockState(assign=assign), + id="", + payload={ + "args": [{"id": dn.id, "value": "1", "type": "int", "error_id": "error_var"}], + }, + ) + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_called_once_with(1, editor_id="a_client_id", comment=None) + assign.assert_called_once_with("error_var", "") + + def test_write_float_data_with_editor_and_comment(self): + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write") as mock_write: + mockGui = Mock(Gui) + mockGui._get_client_id = lambda: "a_client_id" + gui_core_context = _GuiCoreContext(mockGui) + assign = Mock() + gui_core_context.update_data( + state=MockState(assign=assign), + id="", + payload={ + "args": [{"id": dn.id, "value": "1.9", "type": "float", "error_id": "error_var"}], + }, + ) + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_called_once_with(1.9, editor_id="a_client_id", comment=None) + assign.assert_called_once_with("error_var", "") + + def test_fails_and_catch_the_error(self): + with patch("taipy.gui_core._context.is_readable", side_effect=is_true): + with patch("taipy.gui_core._context.is_editable", side_effect=is_true): + with patch("taipy.gui_core._context.core_get", side_effect=core_get) as mock_core_get: + with patch.object(DataNode, "write", side_effect=fails) as mock_write: + mockGui = Mock(Gui) + mockGui._get_client_id = lambda: "a_client_id" + gui_core_context = _GuiCoreContext(mockGui) + assign = Mock() + gui_core_context.update_data( + state=MockState(assign=assign), + id="", + payload={ + "args": [{"id": dn.id, "value": "1.9", "type": "float", "error_id": "error_var"}], + }, + ) + mock_core_get.assert_called_once_with(dn.id) + mock_write.assert_called_once_with(1.9, editor_id="a_client_id", comment=None) + assign.assert_called_once() + assert assign.call_args_list[0].args[0] == "error_var" + assert "Error updating Data node value." in assign.call_args_list[0].args[1]