Skip to main content

A Python toolkit for integrating nonsmooth dynamical systems

Project description

solve_nivp

DOI

A Python library for time integration of nonsmooth ODE/DAE systems: models with abrupt changes such as impacts, switching, or inequality constraints. Such models arise in frictional contact mechanics, piecewise and switching behaviour in circuits, sliding-mode control, and discontinuous rules in finance and energy markets. Classical solvers, which assume smoothness, often require regularisation or very small steps due to the inherent stiffness of these models. solve_nivp builds nonsmooth rules directly into the implicit time-stepping scheme, enabling users to encode constraints and advance the state robustly.

Key features

  • Projection-based constraint encoding. Users express set-valued or nonsmooth relations as projections onto convex sets (Coulomb friction cone, sign / normal cone, second-order cone, algebraic constraints). Custom projections need only implement project() and an optional tangent_cone().

  • Nonlinear solvers for nonsmooth problems. A semismooth Newton method with Armijo line search and a variational-inequality (VI) fixed-point iteration, both with standard tolerances, safeguards, and iteration diagnostics.

  • Implicit integrators. Backward Euler, Trapezoidal, theta-method, a composite TR-BE scheme (Bathe-type, second-order), an embedded BE-TR error estimator, and SDIRK2 on this development branch.

  • Adaptive step-size control with Richardson or embedded error estimates.

  • Optional RL add-on. Exposes the time integrator as a Gym-style environment for learning adaptive step-size policies (TD3 / TQC via Stable Baselines 3).

The library is organised around three interchangeable components: projection, nonlinear solver, and integrator, so that swapping algorithms during experimentation is straightforward. Linear-algebra routines operate on dense or sparse arrays in the SciPy ecosystem.

Installation

From PyPI:

pip install solve_nivp

Latest development pre-release from PyPI:

pip install --pre solve_nivp

Optional extras:

pip install solve_nivp[test]   # includes pytest
pip install solve_nivp[rl]     # RL experiments (gymnasium, stable-baselines3)
pip install solve_nivp[docs]   # Sphinx documentation build
pip install solve_nivp[all]    # branch-local heavy extras (PETSc + RL stack)

Developer install (from source)

git clone https://github.com/ERC-INJECT/solve_nivp.git
cd solve_nivp
python3 -m venv .venv && source .venv/bin/activate
pip install -U pip
pip install -e .[test]

Conda environments are also provided for HPC or other user-space installs:

# lean environment for core development, tests, docs, and notebooks
conda env create -f environment.yml

# broader environment for PETSc, RL, and the heavier notebook stack
conda env create -f environment-full.yml

HPC / user-space install

If you are working on an HPC system and want the environment to live in your project or scratch space rather than a shared default location, create it with an explicit prefix:

export ENV_ROOT=/path/to/your/project-or-scratch-space/.conda
export CONDA_PKGS_DIRS="$ENV_ROOT/pkgs"

conda env create -p "$ENV_ROOT/envs/solve_nivp" -f environment.yml
# or, for the broader stack:
# conda env create -p "$ENV_ROOT/envs/solve_nivp-full" -f environment-full.yml

conda activate "$ENV_ROOT/envs/solve_nivp"
pip install -e .

The sliding-block and prestressed-fault examples also depend on a separate poroelasticity repository that is not bundled as a package dependency here. If you need those examples, install that repo into the same environment as well, for example:

pip install -e /path/to/Poroelasticity

Quickstart

import numpy as np
from solve_nivp import solve_nivp

# simple smooth rhs: y' = -y
rhs = lambda t, y: -y

t_span = (0.0, 1.0)
y0 = np.array([1.0])

# identity projection, VI solver via composite integrator
sol = solve_nivp(
    fun=rhs,
    t_span=t_span,
    y0=y0,
    method='composite',
    projection='identity',
    solver='VI',
)

print(sol[0][:5], sol[1][:5])  # t, y samples

See examples/ for notebooks on friction stick–slip, bouncing ball (contact/impact), SOC constraints, and sliding-mode control.

Running tests

pytest -q

Building the documentation

cd docs
make clean html

Open docs/_build/html/index.html.

RL experiments (optional)

The RL_Adaption/ folder contains optional experiments (TD3/TQC) for learned adaptivity on challenging nonsmooth problems. Large artifacts are ignored by Git and not required for core installation or testing.

In these notebooks the reward signal, observation mapping, and policy configuration are defined directly in notebook cells. The core solve_nivp package only supplies the nonsmooth solvers and the AdaptiveStepperEnv wrapper; all RL-specific choices are intentionally left to the user so they can adapt the workflow to their own applications.

Citation

If you use this software, please cite the JOSS paper:

Riley, D. M. & Stefanou, I. (2025). solve_nivp: A Python toolkit for integrating nonsmooth dynamical systems. Journal of Open Source Software. doi:10.21105/joss.09775

BibTeX:

@article{Riley2025solve_nivp,
  author  = {Riley, David Michael and Stefanou, Ioannis},
  title   = {solve\_nivp: A Python toolkit for integrating nonsmooth dynamical systems},
  journal = {Journal of Open Source Software},
  year    = {2025},
  doi     = {10.21105/joss.09775},
  url     = {https://doi.org/10.21105/joss.09775}
}

License

MIT License (see LICENSE).

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

solve_nivp-0.1.3.dev0.tar.gz (238.1 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

solve_nivp-0.1.3.dev0-py3-none-any.whl (258.7 kB view details)

Uploaded Python 3

File details

Details for the file solve_nivp-0.1.3.dev0.tar.gz.

File metadata

  • Download URL: solve_nivp-0.1.3.dev0.tar.gz
  • Upload date:
  • Size: 238.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for solve_nivp-0.1.3.dev0.tar.gz
Algorithm Hash digest
SHA256 bf18fde1af46ee8f819692d7e723ce3ca3b0876847bbe94623e54e3b4824380c
MD5 cb6aa0ea34a19ee365620615d6672787
BLAKE2b-256 07cfa1c30880498c30ccd7947c9f506b45f6e790687616354f39c13c08b25206

See more details on using hashes here.

Provenance

The following attestation bundles were made for solve_nivp-0.1.3.dev0.tar.gz:

Publisher: publish.yml on ERC-INJECT/solve_nivp

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file solve_nivp-0.1.3.dev0-py3-none-any.whl.

File metadata

File hashes

Hashes for solve_nivp-0.1.3.dev0-py3-none-any.whl
Algorithm Hash digest
SHA256 e080822aa70a8e51611dbb4c664791494b2a51694bdecf25fedfa79ea85c2330
MD5 3ff90065191eb610b917ba46e2a73135
BLAKE2b-256 e4363c41aaf3e1dcd15826713bbb0791bca8b856983d9ea6fbb8d00b3fa3f42f

See more details on using hashes here.

Provenance

The following attestation bundles were made for solve_nivp-0.1.3.dev0-py3-none-any.whl:

Publisher: publish.yml on ERC-INJECT/solve_nivp

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page