Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

v3.13.0 - Olentzero's gift #430

Merged
merged 8 commits into from
Dec 24, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions docs/advanced_usage.rst
Original file line number Diff line number Diff line change
Expand Up @@ -145,6 +145,8 @@ A submodel of a translated model can be run as a standalone model. This can be d
.. automethod:: pysd.py_backend.model.Model.select_submodel
:noindex:

.. note::
This method will mutate the original model. If you want to have a copy of the model with the selected variables/modules you can use :py:data:`inplace=False` argument.

In order to preview the needed exogenous variables, the :py:meth:`.get_dependencies` method can be used:

Expand Down
8 changes: 8 additions & 0 deletions docs/getting_started.rst
Original file line number Diff line number Diff line change
Expand Up @@ -272,3 +272,11 @@ We can easily access the current value of a model component using square bracket
If you try to get the current values of a lookup variable, the previous method will fail, as lookup variables take arguments. However, it is possible to get the full series of a lookup or data object with :py:meth:`.get_series_data` method::

>>> model.get_series_data('Growth lookup')


Copying a model
---------------
Sometimes, you may want to run several versions of a model. For this purpose, copying an already-loaded model to make changes while keeping an untouched one is useful. The :py:meth:`.copy` method will help do that; it will load a new model from the translated file and apply to it the same changes that have been applied to the original model (modifying components, selecting submodels, etc.). You can also load a copy of the source model (without applying) any change setting the argument :py:data:`reload=True`.

.. warning::
The copy function will load a new model from the file and apply the same changes to it. If any of these changes have replaced a variable with a function that references other variables in the model, the copy will not work properly since the function will still reference the variables in the original model, in which case the function should be redefined.
1 change: 1 addition & 0 deletions docs/python_api/model_class.rst
Original file line number Diff line number Diff line change
Expand Up @@ -11,3 +11,4 @@ Macro class
-----------
.. autoclass:: pysd.py_backend.model.Macro
:members:
:exclude-members: export
36 changes: 36 additions & 0 deletions docs/whats_new.rst
Original file line number Diff line number Diff line change
@@ -1,5 +1,41 @@
What's New
==========
v3.13.0 (2023/12/25)
--------------------
New Features
~~~~~~~~~~~~
- Include new method :py:meth:`pysd.py_backend.model.Model.copy` which allows copying a model (:issue:`131`). (`@enekomartinmartinez <https://github.com/enekomartinmartinez>`_)
- :py:meth:`pysd.py_backend.model.Model.select_submodel` now takes an optional argument `inplace` when set to :py:data:`False` it will return a modified copy of the model instead of modifying the original model (:issue:`131`). (`@enekomartinmartinez <https://github.com/enekomartinmartinez>`_)
- :py:meth:`pysd.py_backend.model.Model.export` will now save also time component information if changed (e.g. final time, time step...). (`@enekomartinmartinez <https://github.com/enekomartinmartinez>`_)

Breaking changes
~~~~~~~~~~~~~~~~

Deprecations
~~~~~~~~~~~~

Bug fixes
~~~~~~~~~
- Set the pointer of :py:class:`pysd.py_backend.statefuls.DelayFixed` to 0 during initialization (:issue:`427`). (`@enekomartinmartinez <https://github.com/enekomartinmartinez>`_)
- :py:meth:`pysd.py_backend.model.Model.export` now works with Macros. (`@enekomartinmartinez <https://github.com/enekomartinmartinez>`_)

Documentation
~~~~~~~~~~~~~
- Improve documentation of methods in :py:class:`pysd.py_backend.model.Model` and :py:class:`pysd.py_backend.model.Macro` includying cross-references and rewrite the one from :py:meth:`pysd.py_backend.model.Macro.set_components`. (`@enekomartinmartinez <https://github.com/enekomartinmartinez>`_)
- Include documentation about the new method :py:meth:`pysd.py_backend.model.Model.copy` and update documentation from :py:meth:`pysd.py_backend.model.Model.select_submodel`. (`@enekomartinmartinez <https://github.com/enekomartinmartinez>`_)

Performance
~~~~~~~~~~~
- Improved performace of :py:class:`pysd.py_backend.output.DataFrameHandler` by creating the dataframe at the end of the run (:issue:`374` and :issue:`330`). (`@easyas314159 <https://github.com/easyas314159>`_ and `@enekomartinmartinez <https://github.com/enekomartinmartinez>`_)

Internal Changes
~~~~~~~~~~~~~~~~
- Move old :py:meth:`pysd.py_backend.model.Macro.set_components` to :py:meth:`pysd.py_backend.model.Macro._set_components`, and create new method with the same name without the `new` argument. (`@enekomartinmartinez <https://github.com/enekomartinmartinez>`_)
- Move old :py:meth:`pysd.py_backend.model.Macro.set_stateful` to :py:meth:`pysd.py_backend.model.Macro._set_stateful`. (`@enekomartinmartinez <https://github.com/enekomartinmartinez>`_)
- Make integration tests filter only specific warnings. (`@enekomartinmartinez <https://github.com/enekomartinmartinez>`_)
- Include warnings in :py:meth:`pysd.py_backend.model.Macro.set_components` when changing the behaviour of the component (:issue:`58`). (`@enekomartinmartinez <https://github.com/enekomartinmartinez>`_)


v3.12.0 (2023/10/02)
--------------------
New Features
Expand Down
2 changes: 1 addition & 1 deletion pysd/_version.py
Original file line number Diff line number Diff line change
@@ -1 +1 @@
__version__ = "3.12.0"
__version__ = "3.13.0"
4 changes: 2 additions & 2 deletions pysd/builders/python/python_model_builder.py
Original file line number Diff line number Diff line change
Expand Up @@ -663,7 +663,7 @@ def build_element(self) -> None:
if ", " in self.type:
warn(
f"Variable '{self.name}' is defined with different types:"
f" '{self.type}'. This may cause bugs when trying to"
f" '{self.type}'. This may cause bugs when trying to "
"change its value or applying other methods from the "
"pysd.py_backend.model.Model class. Running the model "
"without modifying this variable should not cause any "
Expand All @@ -672,7 +672,7 @@ def build_element(self) -> None:
elif ", " in self.subtype:
warn(
f"Variable '{self.name}' is defined with different subtypes:"
f" '{self.subtype}'. This may cause bugs when trying to"
f" '{self.subtype}'. This may cause bugs when trying to "
"change its value or applying other methods from the "
"pysd.py_backend.model.Model class. Running the model "
"without modifying this variable should not cause any "
Expand Down
72 changes: 59 additions & 13 deletions pysd/py_backend/components.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
import random
import inspect
import importlib.util
from copy import deepcopy

import numpy as np

Expand Down Expand Up @@ -129,13 +130,45 @@ def __init__(self):
self._time = None
self.stage = None
self.return_timestamps = None
self._next_return = None
self._control_vars_tracker = {}

def __call__(self):
return self._time

def export(self):
"""Exports time values to a dictionary."""
return {
"control_vars": self._get_control_vars(),
"stage": self.stage,
"_time": self._time,
"return_timestamps": self.return_timestamps,
"_next_return": self._next_return
}

def _get_control_vars(self):
"""
Make control vars changes exportable.
"""
out = {}
for cvar, value in self._control_vars_tracker.items():
if callable(value):
out[cvar] = value()
else:
out[cvar] = value
return out

def _set_time(self, time_dict):
"""Copy values from other Time object, used by Model.copy"""
self.set_control_vars(**time_dict['control_vars'])
for key, value in time_dict.items():
if key == 'control_vars':
continue
setattr(self, key, value)

def set_control_vars(self, **kwargs):
"""
Set the control variables valies
Set the control variables values

Parameters
----------
Expand All @@ -149,6 +182,20 @@ def set_control_vars(self, **kwargs):
saveper: float, callable or None
Saveper.

"""
# filter None values
kwargs = {
key: value for key, value in kwargs.items()
if value is not None
}
# track changes
self._control_vars_tracker.update(kwargs)
self._set_control_vars(**kwargs)

def _set_control_vars(self, **kwargs):
"""
Set the control variables values. Private version to be used
to avoid tracking changes.
"""
def _convert_value(value):
# this function is necessary to avoid copying the pointer in the
Expand All @@ -159,8 +206,7 @@ def _convert_value(value):
return lambda: value

for key, value in kwargs.items():
if value is not None:
setattr(self, key, _convert_value(value))
setattr(self, key, _convert_value(value))

if "initial_time" in kwargs:
self._initial_time = self.initial_time()
Expand All @@ -184,16 +230,16 @@ def in_return(self):

if self.return_timestamps is not None:
# this allows managing float precision error
if self.next_return is None:
if self._next_return is None:
return False
if np.isclose(self._time, self.next_return, prec):
if np.isclose(self._time, self._next_return, prec):
self._update_next_return()
return True
else:
while self.next_return is not None\
and self._time > self.next_return:
while self._next_return is not None\
and self._time > self._next_return:
warn(
f"The returning time stamp '{self.next_return}' "
f"The returning time stamp '{self._next_return}' "
"seems to not be a multiple of the time step. "
"This value will not be saved in the output. "
"Please, modify the returning timestamps or the "
Expand All @@ -218,12 +264,12 @@ def add_return_timestamps(self, return_timestamps):
and len(return_timestamps) > 0:
self.return_timestamps = list(return_timestamps)
self.return_timestamps.sort(reverse=True)
self.next_return = self.return_timestamps.pop()
self._next_return = self.return_timestamps.pop()
elif isinstance(return_timestamps, (float, int)):
self.next_return = return_timestamps
self._next_return = return_timestamps
self.return_timestamps = []
else:
self.next_return = None
self._next_return = None
self.return_timestamps = None

def update(self, value):
Expand All @@ -233,9 +279,9 @@ def update(self, value):
def _update_next_return(self):
""" Update the next_return value """
if self.return_timestamps:
self.next_return = self.return_timestamps.pop()
self._next_return = self.return_timestamps.pop()
else:
self.next_return = None
self._next_return = None

def reset(self):
""" Reset time value to the initial """
Expand Down
Loading