JAX-based POD-RBF for autodiff-enabled reduced order modeling.
Project description
POD-RBF
A Python package for building a Reduced Order Model (ROM) from high-dimensional data using a Proper Orthogonal Decomposition - Radial Basis Function (POD-RBF) Network.
Given a 'snapshot' matrix of the data points with varying parameters, this code contains functions to find the truncated POD basis and interpolate using a RBF network for new parameters.
This code is based on the following papers implementing the method:
- Solving inverse heat conduction problems using trained POD-RBF network inverse method - Ostrowski, Bialecki, Kassab (2008)
- RBF-trained POD-accelerated CFD analysis of wind loads on PV systems - Huayamave, Ceballos, Barriento, Seigneur, Barkaszi, Divo, and Kassab (2017)
- Real-Time Thermomechanical Modeling of PV Cell Fabrication via a POD-Trained RBF Interpolation Network - Das, Khoury, Divo, Huayamave, Ceballos, Eaglin, Kassab, Payne, Yelundur, and Seigneur (2020)
Features:
- JAX-based - enables autodifferentiation for gradient optimization, sensitivity analysis, and inverse problems
- Shape parameter optimization for the Radial Basis Functions (RBFs)
- Algorithm switching based on memory requirements (eigenvalue decomposition vs. SVD)
Installation
pip install pod-rbf
# or
uv add pod-rbf
Example
In the example folder you can find two examples of that demonstrates how to use the package. The first is a simple heat conduction problem on a unit square.
The other example will be demonstrated step-by-step here. We seek to build a ROM of the 2D lid-driven cavity problem. For the impatient, here is the full code to run this example. I will break down each line in the sections below.
If you wish to build a ROM with multiple parameters, see this basic 2-parameter example.
import pod_rbf
import jax.numpy as jnp
import numpy as np
Re = np.linspace(0, 1000, num=11)
Re[0] = 1
# make snapshot matrix from csv files
train_snapshot = pod_rbf.build_snapshot_matrix("examples/lid-driven-cavity/data/train/")
# train the model (keeps 99% energy in POD modes by default)
result = pod_rbf.train(train_snapshot, Re)
# inference on an unseen parameter
sol = pod_rbf.inference_single(result.state, jnp.array(450.0))
Building the snapshot matrix
First, we need to build the snapshot matrix, X, which contains the data we are training on. It must be of the form where each column is the k-th 'snapshot' of the solution field given some parameter, p_k, with n samples in the snapshot at locations x_n. A single snapshot is below
and the snapshot matrix would then look like
where m is the total number of snapshots.
For example, suppose our lid-driven cavity was solved on a mesh with 400 cells and we varied the parameter of interest (Re number in this case) 10 times. We would have a matrix of size (n,m) = (400,10).
For our example, solutions were generated using STAR-CCM+ for Reynolds numbers of 1-1000 in increments of 100. The Re number will serve as our single parameter in this case. Snapshots were generated as a separate .csv file for each. To make it easier to combine them all into the snapshot matrix, there is a function which takes the path and file pattern. The same syntax is borrowed from the ffmpeg tool - that is, if you had files named as sample_001.csv, sample_002.csv ... you would input sample_%03d.csv. The files for this example are named as re-%04d.csv so we would issue a command as
>>> import pod_rbf
>>> train_snapshot = pod_rbf.build_snapshot_matrix("examples/lid-driven-cavity/data/train/")
Note: if you are using this approach where each snapshot is contained in a different csv file, please group all of them into a directory of their own.
If you notice, these files are contained in the train folder, as I also generated some more snapshots for validation (which as you probably guessed is in the /data/validation folder). now we need to generate the array of input parameters that correspond to each snapshot.
>>> Re = np.linspace(0, 1000, num=11)
>>> Re[0] = 1
Note: it is extremely important that each input parameter maps to the same element number of the
snapshot matrix. For example if the 5th column (index 4) then the input parameter used to generate
that snapshot should be what you find in the 5th element (index 4) of the array, e.g.
train_snapshot[:,4] -> Re[4]. The csv files are loaded in alpha-numeric order so that is why
the input parameter array goes from 1 -> 1000.
where Re is an array of input parameters that we are training the model on. Next, we train
the model with a single function call. We choose to keep 99% of the energy in POD modes (this is
the default, so you don't have to set that).
>>> result = pod_rbf.train(train_snapshot, Re)
>>> # Or with custom config:
>>> config = pod_rbf.TrainConfig(energy_threshold=0.99)
>>> result = pod_rbf.train(train_snapshot, Re, config)
Now that the weights and truncated POD basis have been calculated and stored in result.state, we
can inference on the model using any input parameter.
>>> import jax.numpy as jnp
>>> sol = pod_rbf.inference_single(result.state, jnp.array(450.0))
and we can plot the results comparing the inference and target below
and for Reynold's number of 50:
Saving and loading models
You can save and load the trained model state:
>>> pod_rbf.save_model("model.pkl", result.state)
>>> state = pod_rbf.load_model("model.pkl")
>>> sol = pod_rbf.inference_single(state, jnp.array(450.0))
Autodifferentiation
Since POD-RBF is built on JAX, you can compute gradients for optimization and inverse problems:
>>> import jax
>>> grad_fn = jax.grad(lambda p: jnp.sum(pod_rbf.inference_single(result.state, p)**2))
>>> gradient = grad_fn(jnp.array(450.0))
Project details
Release history Release notifications | RSS feed
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
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file pod_rbf-2.0.1.tar.gz.
File metadata
- Download URL: pod_rbf-2.0.1.tar.gz
- Upload date:
- Size: 2.5 MB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
86a3bf3dc7319f5126ca39d3a12bf97df7e1f5c7f9911f3d71827c0fc5204344
|
|
| MD5 |
4d8db165fe46368056eb37c2094d5969
|
|
| BLAKE2b-256 |
2de85b0f8d10dd64313e271b03b8f5172f29ec8afea152592ff5c4f628223bc0
|
Provenance
The following attestation bundles were made for pod_rbf-2.0.1.tar.gz:
Publisher:
publish.yml on kylebeggs/POD-RBF
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
pod_rbf-2.0.1.tar.gz -
Subject digest:
86a3bf3dc7319f5126ca39d3a12bf97df7e1f5c7f9911f3d71827c0fc5204344 - Sigstore transparency entry: 763991803
- Sigstore integration time:
-
Permalink:
kylebeggs/POD-RBF@fc112967f298505d379fb2ab4ffe6d70ed40cdb0 -
Branch / Tag:
refs/tags/v2.0.1 - Owner: https://github.com/kylebeggs
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@fc112967f298505d379fb2ab4ffe6d70ed40cdb0 -
Trigger Event:
push
-
Statement type:
File details
Details for the file pod_rbf-2.0.1-py3-none-any.whl.
File metadata
- Download URL: pod_rbf-2.0.1-py3-none-any.whl
- Upload date:
- Size: 25.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
739216a367dc0e657b78c59b76f68ffb418758a73ca09a126c7d98ee5b29a16a
|
|
| MD5 |
9e9a3aabfee57069b0f4e263539ea04b
|
|
| BLAKE2b-256 |
dc7cdabbabe0033fc7f532bd959cdbf60da7db6c3428f4a2e9f38e5cd88a10ae
|
Provenance
The following attestation bundles were made for pod_rbf-2.0.1-py3-none-any.whl:
Publisher:
publish.yml on kylebeggs/POD-RBF
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
pod_rbf-2.0.1-py3-none-any.whl -
Subject digest:
739216a367dc0e657b78c59b76f68ffb418758a73ca09a126c7d98ee5b29a16a - Sigstore transparency entry: 763991805
- Sigstore integration time:
-
Permalink:
kylebeggs/POD-RBF@fc112967f298505d379fb2ab4ffe6d70ed40cdb0 -
Branch / Tag:
refs/tags/v2.0.1 - Owner: https://github.com/kylebeggs
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@fc112967f298505d379fb2ab4ffe6d70ed40cdb0 -
Trigger Event:
push
-
Statement type: