A block based time domain system simulation framework.
Project description
PathSim: A Time-Domain System Simulation Framework
Overview
PathSim is a minimalistic and flexible block-based time-domain system simulation framework in Python with basic automatic differentiation capabilities. It provides a modular and intuitive approach to modeling and simulating complex interconnected dynamical systems. It is similar to Matlab Simulink in spirit but works very differently under the hood.
Key features of PathSim include:
- Decentralized architecture where each dynamical block has their own numerical integration engine.
- The system is solved directly on the computational graph instead of compiling a unified differential algebraic system.
- This has some advantages such as hot-swappable blocks during simulation and reading simulation results directly from the scopes.
- The block execution is decoupled from the data transfer, which enables parallelization (future) and linear computational complexity scaling for sparsely connected systems.
- Support for MIMO (Multiple Input, Multiple Output) blocks, enabling the creation of complex interconnected system topologies.
- Fixed-point iteration approach with path length estimation to efficiently resolve algebraic loops.
- Wide range of numerical solvers, including implicit and explicit multi-stage, and adaptive Runge-Kutta methods such as
RKDP54
orESDIRK54
. - Modular and hierarchical modeling with (nested) subsystems.
- Automatic differentiation for differentiable system simulations.
- Library of pre-defined blocks, including mathematical operations, integrators, delays, transfer functions, and more.
- Easy extensibility, allowing users to define custom blocks by subclassing the base
Block
class and implementing just a handful of methods.
Installation
The latest release version of pathsim available on PyPi and installable via pip:
$ pip install pathsim
Example - Harmonic Oscillator
Here's an example that demonstrates how to create a basic simulation. The main components of the package are:
Simulation
: The main class that handles the blocks, connections, and the simulation loop.Connection
: The class that defines the connections between blocks.- Various block classes from the
blocks
module, such asIntegrator
,Amplifier
,Adder
,Scope
, etc.
In this example, we create a simulation of the harmonic oscillator (a spring mass damper 2nd order system) initial value problem. The ODE that defines it is give by
$$ \ddot{x} + \frac{c}{m} \dot{x} + \frac{k}{m} x = 0 $$
where $c$ is the damping, $k$ the spring constant and $m$ the mass. And initial conditions $x_0$ and $v_0$ for position and velocity.
The ODE above can be translated to a block diagram using integrators, amplifiers and adders in the following way:
The topology of the block diagram above can be directly defined as blocks and connections in the PathSim
framework. First we initialize the blocks needed to represent the dynamical systems with their respective arguments such as initial conditions and gain values, then the blocks are connected using Connection
objects, forming two feedback loops. The Simulation
instance manages the blocks and connections and advances the system in time with the timestep (dt
). The log
flag for logging the simulation progress is also set. Finally, we run the simulation for some number of seconds and plot the results using the plot()
method of the scope block.
from pathsim import Simulation
from pathsim import Connection
from pathsim.blocks import Integrator, Amplifier, Adder, Scope
from pathsim.solvers import SSPRK22 # 2nd order fixed timestep, this is also the default
#initial position and velocity
x0, v0 = 2, 5
#parameters (mass, damping, spring constant)
m, c, k = 0.8, 0.2, 1.5
# Create blocks
I1 = Integrator(v0) # integrator for velocity
I2 = Integrator(x0) # integrator for position
A1 = Amplifier(-c/m)
A2 = Amplifier(-k/m)
P1 = Adder()
Sc = Scope(labels=["v(t)", "x(t)"])
blocks = [I1, I2, A1, A2, P1, Sc]
# Create connections
connections = [
Connection(I1, I2, A1, Sc), # one to many connection
Connection(I2, A2, Sc[1]),
Connection(A1, P1), # default connection to port 0
Connection(A2, P1[1]), # specific connection to port 1
Connection(P1, I1)
]
# Create a simulation instance from the blocks and connections
Sim = Simulation(blocks, connections, dt=0.05, log=True, Solver=SSPRK22)
# Run the simulation for 50 seconds
Sim.run(duration=50.0)
# Plot the results directly from the scope
Sc.plot()
# Read the results from the scope for further processing
time, data = Sc.read()
2024-11-07 21:18:04,345 - INFO - LOGGING enabled
2024-11-07 21:18:04,345 - INFO - SOLVER SSPRK22 adaptive=False implicit=False
2024-11-07 21:18:04,346 - INFO - PATH LENGTH ESTIMATE 2, 'iterations_min' set to 2
2024-11-07 21:18:04,346 - INFO - RESET
2024-11-07 21:18:04,347 - INFO - RUN duration=50.0
2024-11-07 21:18:04,347 - INFO - STARTING progress tracker
2024-11-07 21:18:04,348 - INFO - progress=0%
2024-11-07 21:18:04,367 - INFO - progress=10%
2024-11-07 21:18:04,385 - INFO - progress=20%
2024-11-07 21:18:04,403 - INFO - progress=30%
2024-11-07 21:18:04,422 - INFO - progress=40%
2024-11-07 21:18:04,440 - INFO - progress=50%
2024-11-07 21:18:04,458 - INFO - progress=60%
2024-11-07 21:18:04,476 - INFO - progress=70%
2024-11-07 21:18:04,493 - INFO - progress=80%
2024-11-07 21:18:04,511 - INFO - progress=90%
2024-11-07 21:18:04,529 - INFO - progress=100%
2024-11-07 21:18:04,529 - INFO - FINISHED steps(total)=1001(1001) runtime=182.51ms
Example - Differentiable Simulation
PathSim also includes a rudimentary automatic differentiation framework based on a dual number system with overloaded operators. This makes the system simulation fully differentiable with respect to a predefined set of parameters. For now it only works with the explicit integrators. To demonstrate this lets consider the following linear feedback system.
The source term is a scaled unit step function (scaled by $A$). The parameters we want to differentiate the time domain response by are the feedback term $a$, the initial condition $x_0$ and the amplitude of the source term $A$.
from pathsim import Simulation, Connection
from pathsim.blocks import Source, Integrator, Amplifier, Adder, Scope
#AD module
from pathsim.diff import Parameter
#parameters
A = Parameter(1)
a = Parameter(-1)
x0 = Parameter(2)
#simulation timestep
dt = 0.01
#step function
tau = 3
def s(t):
return A*int(t>tau)
#blocks that define the system
Src = Source(s)
Int = Integrator(x0)
Amp = Amplifier(a)
Add = Adder()
Sco = Scope(labels=["step", "response"])
blocks = [Src, Int, Amp, Add, Sco]
#the connections between the blocks
connections = [
Connection(Src, Add[0], Sco[0]),
Connection(Amp, Add[1]),
Connection(Add, Int),
Connection(Int, Amp, Sco[1])
]
#initialize simulation with the blocks, connections, timestep and logging enabled
Sim = Simulation(blocks, connections, dt=dt, log=True)
#run the simulation for some time
Sim.run(4*tau)
Sco.plot()
2024-11-07 21:18:06,513 - INFO - LOGGING enabled
2024-11-07 21:18:06,514 - INFO - SOLVER SSPRK22 adaptive=False implicit=False
2024-11-07 21:18:06,514 - INFO - PATH LENGTH ESTIMATE 2, 'iterations_min' set to 2
2024-11-07 21:18:06,514 - INFO - RESET
2024-11-07 21:18:06,515 - INFO - RUN duration=12
2024-11-07 21:18:06,516 - INFO - STARTING progress tracker
2024-11-07 21:18:06,517 - INFO - progress=0%
2024-11-07 21:18:06,573 - INFO - progress=10%
2024-11-07 21:18:06,629 - INFO - progress=20%
2024-11-07 21:18:06,680 - INFO - progress=30%
2024-11-07 21:18:06,732 - INFO - progress=40%
2024-11-07 21:18:06,784 - INFO - progress=50%
2024-11-07 21:18:06,838 - INFO - progress=60%
2024-11-07 21:18:06,891 - INFO - progress=70%
2024-11-07 21:18:06,945 - INFO - progress=80%
2024-11-07 21:18:06,999 - INFO - progress=90%
2024-11-07 21:18:07,051 - INFO - progress=100%
2024-11-07 21:18:07,052 - INFO - FINISHED steps(total)=1201(1201) runtime=535.60ms
Now the recorded data is of type Parameter
and we can evaluate the automatically computed partial derivatives at each timestep. For example
$\partial x(t) / \partial a$ the response with respect to the linear feedback parameter.
import matplotlib.pyplot as plt
#read data from the scope
time, [step, data] = Sco.read()
#evaluate partial derivatives
dxda = list(map(lambda x: x.d(a), data)) # w.r.t. feedback
dxdx0 = list(map(lambda x: x.d(x0), data)) # w.r.t. initial condition
dxdA = list(map(lambda x: x.d(A), data)) # w.r.t. source amplitude
fig, ax = plt.subplots(nrows=1, tight_layout=True, figsize=(8, 4), dpi=120)
ax.plot(time, dxda, label="$dx/da$")
ax.plot(time, dxdx0, label="$dx/dx_0$")
ax.plot(time, dxdA, label="$dx/dA$")
ax.set_xlabel("time [s]")
ax.grid(True)
ax.legend(fancybox=False);
Contributing and Future
There are some things I want to explore with PathSim eventually, and your help is highly appreciated! If you want to contribute, send me a message and we can discuss how!
Some of the possible directions for future features are:
- better
__repr__
for the blocks maybe in json format OR just add ajson
method to the blocks and to the connections that builds a netlist representation to save to and load from an interpretable file (compatibility with other system description languages) - implement event handling mechanism, including scheduled events and event detection, this should interface with the
get
,set
andreset
methods of the solvers - include discrete time blocks and integrate them into the event handling mechanism
- more extensive testing and validation (as always)
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
Built Distribution
File details
Details for the file pathsim-0.4.6.tar.gz
.
File metadata
- Download URL: pathsim-0.4.6.tar.gz
- Upload date:
- Size: 79.8 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/5.1.1 CPython/3.9.20
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 79269194f37ff54e1494fffe64f8a53ea1731b5c346a34f4535fdd5e82fe51b2 |
|
MD5 | 739ac5341f8b794319e7bd12bae2b296 |
|
BLAKE2b-256 | ca211708440a84bd89e43099ff2a08a57c0e5cdff3a2dad8218a3edec8df1484 |
File details
Details for the file pathsim-0.4.6-py3-none-any.whl
.
File metadata
- Download URL: pathsim-0.4.6-py3-none-any.whl
- Upload date:
- Size: 136.3 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/5.1.1 CPython/3.9.20
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 8dc22ea7af97e282d1bab2338c7bf6d213b66ba989bfc4c49e79ada26c03a7a6 |
|
MD5 | 1926ad3d7d99489a628da630601a89d6 |
|
BLAKE2b-256 | 882d4272c145369526dc226d6ebfed6cdee765e188c26948e64ecd27cef58ad6 |