diff --git a/.github/workflows/ode-toolbox-build.yml b/.github/workflows/ode-toolbox-build.yml
index b95c07ee..4718a1a4 100644
--- a/.github/workflows/ode-toolbox-build.yml
+++ b/.github/workflows/ode-toolbox-build.yml
@@ -43,7 +43,7 @@ jobs:
fail-fast: false
matrix:
with_gsl: ["0", "1"]
- sympy_version: ["==1.4", ""] # empty string for "latest"
+ sympy_version: [""] # empty string for "latest"
steps:
- name: Checkout ODE-toolbox code
diff --git a/doc/index.rst b/doc/index.rst
index c43d5880..a3e486c4 100644
--- a/doc/index.rst
+++ b/doc/index.rst
@@ -348,6 +348,10 @@ The following global options are defined. Note that all are typically formatted
- :python:`"sympy.simplify(expr)"`
- string
- For all expressions ``expr`` that are rewritten internally: the contents of this parameter string are evaluated with ``eval()`` in Python to obtain the final output expression. Override for custom expression simplification steps. Example: ``"sympy.logcombine(sympy.powsimp(sympy.expand(expr)))"``.
+ * - ``forbidden_names``
+ - :python:`["oo", "zoo", "nan", "NaN", "__h"]`
+ - list of strings
+ - For each forbidden name: emit an error if a variable or parameter by this name occurs in the input.
Output
diff --git a/odetoolbox/__init__.py b/odetoolbox/__init__.py
index f365d01a..66ee08ea 100644
--- a/odetoolbox/__init__.py
+++ b/odetoolbox/__init__.py
@@ -27,7 +27,7 @@
from sympy.core.expr import Expr as SympyExpr # works for both sympy 1.4 and 1.8
from .config import Config
-from .sympy_helpers import _find_in_matrix, _is_zero, _is_sympy_type, SympyPrinter
+from .sympy_helpers import _check_numerical_issue, _check_forbidden_name, _find_in_matrix, _is_zero, _is_sympy_type, SympyPrinter
from .system_of_shapes import SystemOfShapes
from .shapes import MalformedInputException, Shape
@@ -109,7 +109,7 @@ def _from_json_to_shapes(indict, parameters=None) -> Tuple[List[Shape], Dict[sym
"""
logging.info("Processing input shapes...")
- shapes = []
+
# first run for grabbing all the variable names. Coefficients might be incorrect.
all_variable_symbols = []
all_parameter_symbols = set()
@@ -124,6 +124,11 @@ def _from_json_to_shapes(indict, parameters=None) -> Tuple[List[Shape], Dict[sym
assert all([_is_sympy_type(sym) for sym in all_variable_symbols])
logging.info("All known variables: " + str(all_variable_symbols) + ", all parameters used in ODEs: " + str(all_parameter_symbols))
+ # validate input for forbidden names
+ for var in set(all_variable_symbols) | all_parameter_symbols:
+ _check_forbidden_name(var)
+
+ # validate parameters
for param in all_parameter_symbols:
if parameters is None:
parameters = dict()
@@ -135,6 +140,7 @@ def _from_json_to_shapes(indict, parameters=None) -> Tuple[List[Shape], Dict[sym
parameters[param] = None
# second run with the now-known list of variable symbols
+ shapes = []
for shape_json in indict["dynamics"]:
shape = Shape.from_json(shape_json, all_variable_symbols=all_variable_symbols, parameters=parameters, _debug=True)
shapes.append(shape)
@@ -210,6 +216,9 @@ def _analysis(indict, disable_stiffness_check: bool = False, disable_analytic_so
assert type(k) is sympy.Symbol
parameters[k] = v
+ _check_forbidden_name(k)
+
+
#
# create Shapes and SystemOfShapes
#
@@ -292,8 +301,12 @@ def _analysis(indict, disable_stiffness_check: bool = False, disable_analytic_so
all_shape_symbols = [str(sympy.Symbol(str(shape.symbol) + Config().differential_order_symbol * i)) for i in range(shape.order)]
for sym in all_shape_symbols:
if sym in solver_json["state_variables"]:
- solver_json["initial_values"][sym] = str(shape.get_initial_value(sym.replace(Config().differential_order_symbol, "'")))
+ iv_expr = shape.get_initial_value(sym.replace(Config().differential_order_symbol, "'"))
+ solver_json["initial_values"][sym] = str(iv_expr)
+ # validate output for numerical problems
+ for var in iv_expr.atoms():
+ _check_numerical_issue(var)
#
# copy the parameter values from the input to the output for convenience; convert into numeric values
@@ -318,7 +331,20 @@ def _analysis(indict, disable_stiffness_check: bool = False, disable_analytic_so
break
if symbol_appears_in_any_expr:
- solver_json["parameters"][param_name] = str(sympy.parsing.sympy_parser.parse_expr(param_expr, global_dict=Shape._sympy_globals).n())
+ sympy_expr = sympy.parsing.sympy_parser.parse_expr(param_expr, global_dict=Shape._sympy_globals)
+
+ # validate output for numerical problems
+ for var in sympy_expr.atoms():
+ _check_numerical_issue(var)
+
+ # convert to numeric value
+ sympy_expr = sympy_expr.n()
+
+ # validate output for numerical problems
+ for var in sympy_expr.atoms():
+ _check_numerical_issue(var)
+
+ solver_json["parameters"][param_name] = str(sympy_expr)
#
diff --git a/odetoolbox/config.py b/odetoolbox/config.py
index 8efb3e8c..6f4fe97e 100644
--- a/odetoolbox/config.py
+++ b/odetoolbox/config.py
@@ -34,7 +34,8 @@ class Config:
"sim_time": 100E-3,
"max_step_size": 999.,
"integration_accuracy_abs": 1E-6,
- "integration_accuracy_rel": 1E-6
+ "integration_accuracy_rel": 1E-6,
+ "forbidden_names": ["oo", "zoo", "nan", "NaN", "__h"]
}
def __getitem__(self, key):
diff --git a/odetoolbox/shapes.py b/odetoolbox/shapes.py
index ba9fc0c1..2f805c26 100644
--- a/odetoolbox/shapes.py
+++ b/odetoolbox/shapes.py
@@ -21,7 +21,7 @@
from __future__ import annotations
-from typing import List, Mapping, Tuple
+from typing import List, Tuple
import functools
import logging
@@ -29,12 +29,10 @@
import sympy
import sympy.parsing.sympy_parser
-from sympy.core.expr import Expr as SympyExpr # works for both sympy 1.4 and 1.8
-from sympy.core.numbers import One as SympyOne
-from sympy.core.numbers import Zero as SympyZero
+from sympy.core.expr import Expr as SympyExpr
from .config import Config
-from .sympy_helpers import _custom_simplify_expr, _is_sympy_type, _is_zero
+from .sympy_helpers import _check_numerical_issue, _check_forbidden_name, _custom_simplify_expr, _is_constant_term, _is_sympy_type, _is_zero
class MalformedInputException(Exception):
@@ -44,17 +42,6 @@ class MalformedInputException(Exception):
pass
-def is_constant_term(term, parameters: Mapping[sympy.Symbol, str] = None):
- r"""
- :return: :python:`True` if and only if this term contains only numerical values and parameters; :python:`False` otherwise.
- """
- if parameters is None:
- parameters = {}
- assert all([type(k) is sympy.Symbol for k in parameters.keys()])
- return type(term) in [sympy.Float, sympy.Integer, SympyZero, SympyOne] \
- or all([sym in parameters.keys() for sym in term.free_symbols])
-
-
class Shape:
r"""
This class provides a canonical representation of a shape function independently of the way in which the user specified the shape. It assumes a differential equation of the general form (where bracketed superscript :math:`{}^{(n)}` indicates the :math:`n`-th derivative with respect to time):
@@ -79,6 +66,7 @@ class Shape:
"Integer": sympy.Integer,
"Float": sympy.Float,
"Function": sympy.Function,
+ "Mul": sympy.Mul,
"Pow": sympy.Pow,
"power": sympy.Pow,
"exp": sympy.exp,
@@ -326,6 +314,9 @@ def from_json(cls, indict, all_variable_symbols=None, parameters=None, _debug=Fa
raise MalformedInputException("In defintion of initial value for variable \"" + iv_symbol + "\": differential order (" + str(iv_order) + ") exceeds that of overall equation order (" + str(order) + ")")
if initial_val_specified[iv_order]:
raise MalformedInputException("Initial value for order " + str(iv_order) + " specified more than once")
+
+ _check_forbidden_name(iv_symbol)
+
initial_val_specified[iv_order] = True
initial_values[iv_symbol + iv_order * "'"] = iv_rhs
@@ -392,13 +383,13 @@ def split_lin_inhom_nonlin(expr, x, parameters=None):
terms = [expr]
for term in terms:
- if is_constant_term(term, parameters=parameters):
+ if _is_constant_term(term, parameters=parameters):
inhom_term += term
else:
# check if the term is linear in any of the symbols in `x`
is_lin = False
for j, sym in enumerate(x):
- if is_constant_term(term / sym, parameters=parameters):
+ if _is_constant_term(term / sym, parameters=parameters):
lin_factors[j] += term / sym
is_lin = True
break
@@ -584,8 +575,21 @@ def from_ode(cls, symbol: str, definition: str, initial_values: dict, all_variab
order: int = len(initial_values)
all_variable_symbols_dict = {str(el): el for el in all_variable_symbols}
definition = sympy.parsing.sympy_parser.parse_expr(definition.replace("'", Config().differential_order_symbol), global_dict=Shape._sympy_globals, local_dict=all_variable_symbols_dict) # minimal global_dict to make no assumptions (e.g. "beta" could otherwise be recognised as a function instead of as a parameter symbol)
+
+ # validate input for forbidden names
+ _initial_values = {k: sympy.parsing.sympy_parser.parse_expr(v, evaluate=False, global_dict=Shape._sympy_globals, local_dict=all_variable_symbols_dict) for k, v in initial_values.items()}
+ for iv_expr in _initial_values.values():
+ for var in iv_expr.atoms():
+ _check_forbidden_name(var)
+
+ # parse input
initial_values = {k: sympy.parsing.sympy_parser.parse_expr(v, global_dict=Shape._sympy_globals, local_dict=all_variable_symbols_dict) for k, v in initial_values.items()}
+ # validate input for numerical issues
+ for iv_expr in initial_values.values():
+ for var in iv_expr.atoms():
+ _check_numerical_issue(var)
+
local_symbols = [symbol + Config().differential_order_symbol * i for i in range(order)]
local_symbols_sympy = [sympy.Symbol(sym_name) for sym_name in local_symbols]
if not symbol in all_variable_symbols:
diff --git a/odetoolbox/sympy_helpers.py b/odetoolbox/sympy_helpers.py
index 0d59f588..ccf443a1 100644
--- a/odetoolbox/sympy_helpers.py
+++ b/odetoolbox/sympy_helpers.py
@@ -1,5 +1,5 @@
#
-# sympy_printer.py
+# sympy_helpers.py
#
# This file is part of the NEST ODE toolbox.
#
@@ -19,6 +19,8 @@
# along with NEST. If not, see .
#
+from typing import Mapping
+
import logging
import sympy
import sys
@@ -26,6 +28,46 @@
from .config import Config
+class NumericalIssueException(Exception):
+ r"""Thrown in case of numerical issues, like division by zero."""
+ pass
+
+
+def _is_constant_term(term, parameters: Mapping[sympy.Symbol, str] = None) -> bool:
+ r"""
+ :return: :python:`True` if and only if this term contains only numerical values and parameters; :python:`False` otherwise.
+ """
+ if parameters is None:
+ parameters = {}
+
+ assert all([type(k) is sympy.Symbol for k in parameters.keys()])
+
+ if sympy.__version__.startswith("1.4"):
+ sympy_zero = sympy.numbers.Zero
+ sympy_one = sympy.numbers.One
+ else:
+ sympy_zero = sympy.core.numbers.Zero
+ sympy_one = sympy.core.numbers.One
+
+ return type(term) in [sympy.Float, sympy.Integer, sympy_zero, sympy_one] \
+ or all([sym in parameters.keys() for sym in term.free_symbols])
+
+
+def _check_numerical_issue(var: str) -> None:
+ forbidden_vars = ["zoo", "oo", "nan", "NaN"]
+ stripped_var_name = str(var).strip("'")
+ if stripped_var_name in forbidden_vars:
+ raise NumericalIssueException("The variable \"" + stripped_var_name + "\" was found. This indicates a numerical problem while solving the system of ODEs. Please check the input for correctness (such as the presence of divisions by zero).")
+
+
+def _check_forbidden_name(var: str) -> None:
+ from .shapes import MalformedInputException
+
+ stripped_var_name = str(var).strip("'")
+ if stripped_var_name in Config().forbidden_names:
+ raise MalformedInputException("Variable by name \"" + stripped_var_name + "\" not allowed; this is a reserved name.")
+
+
def _is_zero(x):
r"""
Check if a sympy expression is equal to zero.
diff --git a/odetoolbox/system_of_shapes.py b/odetoolbox/system_of_shapes.py
index 0a2b7222..3a44a35a 100644
--- a/odetoolbox/system_of_shapes.py
+++ b/odetoolbox/system_of_shapes.py
@@ -195,17 +195,14 @@ def get_sub_system(self, symbols):
def _generate_propagator_matrix(self, A):
- r"""Generate the propagator matrix by matrix exponentiation.
-
- XXX: the default custom simplification expression does not work well with sympy 1.4 here. Consider replacing sympy.simplify() with _custom_simplify_expr() if sympy 1.4 support is dropped.
- """
+ r"""Generate the propagator matrix by matrix exponentiation."""
# naive: calculate propagators in one step
- # P_naive = sympy.simplify(sympy.exp(A * sympy.Symbol(Config().output_timestep_symbol)))
+ # P_naive = _custom_simplify_expr(sympy.exp(A * sympy.Symbol(Config().output_timestep_symbol)))
# optimized: be explicit about block diagonal elements; much faster!
blocks = get_block_diagonal_blocks(np.array(A))
- propagators = [sympy.simplify(sympy.exp(sympy.Matrix(block) * sympy.Symbol(Config().output_timestep_symbol))) for block in blocks]
+ propagators = [_custom_simplify_expr(sympy.exp(sympy.Matrix(block) * sympy.Symbol(Config().output_timestep_symbol))) for block in blocks]
P = sympy.Matrix(scipy.linalg.block_diag(*propagators))
# check the result
diff --git a/tests/test_malformed_input.py b/tests/test_malformed_input.py
new file mode 100644
index 00000000..b26180fe
--- /dev/null
+++ b/tests/test_malformed_input.py
@@ -0,0 +1,66 @@
+#
+# test_malformed_input.py
+#
+# This file is part of the NEST ODE toolbox.
+#
+# Copyright (C) 2017 The NEST Initiative
+#
+# The NEST ODE toolbox is free software: you can redistribute it
+# and/or modify it under the terms of the GNU General Public License
+# as published by the Free Software Foundation, either version 2 of
+# the License, or (at your option) any later version.
+#
+# The NEST ODE toolbox is distributed in the hope that it will be
+# useful, but WITHOUT ANY WARRANTY; without even the implied warranty
+# of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+# General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with NEST. If not, see .
+#
+
+import pytest
+
+from .context import odetoolbox
+from odetoolbox.shapes import MalformedInputException
+from odetoolbox.sympy_helpers import NumericalIssueException
+
+
+class TestMalformedInput:
+ r"""Test for failure when forbidden names are used in the input."""
+
+ @pytest.mark.xfail(strict=True, raises=MalformedInputException)
+ def test_malformed_input_iv(self):
+ indict = {"dynamics": [{"expression": "x' = 0",
+ "initial_value": "zoo"}]}
+ result = odetoolbox.analysis(indict, disable_stiffness_check=True)
+
+ @pytest.mark.xfail(strict=True, raises=MalformedInputException)
+ def test_malformed_input_expr(self):
+ indict = {"dynamics": [{"expression": "x' = 42 * NaN",
+ "initial_value": "0."}]}
+ result = odetoolbox.analysis(indict, disable_stiffness_check=True)
+
+ @pytest.mark.xfail(strict=True, raises=MalformedInputException)
+ def test_malformed_input_sym(self):
+ indict = {"dynamics": [{"expression": "oo' = 0",
+ "initial_value": "0."}]}
+ result = odetoolbox.analysis(indict, disable_stiffness_check=True)
+
+ def test_correct_input(self):
+ indict = {"dynamics": [{"expression": "foo' = 0",
+ "initial_value": "0."}]}
+ result = odetoolbox.analysis(indict, disable_stiffness_check=True)
+
+ @pytest.mark.xfail(strict=True, raises=NumericalIssueException)
+ def test_malformed_input_numerical_iv(self):
+ indict = {"dynamics": [{"expression": "foo' = 0",
+ "initial_value": "1/0"}]}
+ result = odetoolbox.analysis(indict, disable_stiffness_check=True)
+
+ @pytest.mark.xfail(strict=True, raises=NumericalIssueException)
+ def test_malformed_input_numerical_parameter(self):
+ indict = {"dynamics": [{"expression": "foo' = bar",
+ "initial_value": "1"}],
+ "parameters": {"bar": "1/0"}}
+ result = odetoolbox.analysis(indict, disable_stiffness_check=True)