# FluxNet: Conservation-Guaranteed Neural Surrogate Models for PDEs

This repository contains the supplementary material for the ICML 2026 submission: "FluxNet: Structure-Preserving Neural Networks for Conservation Laws via Flux-Based Updates."

## Overview

FluxNet is a family of neural network architectures that guarantee mass conservation and bound constraints for solving partial differential equations (PDEs) governed by conservation laws. Unlike standard neural surrogates that directly predict the next state, FluxNet predicts inter-cell fluxes and updates states through conservative flux differences.

### Key Features

- **Guaranteed Conservation**: Global conserved quantities (mass, momentum, etc.) are preserved exactly during autoregressive rollout
- **Bound Preservation**: Physical bounds (e.g., 0 ≤ ρ ≤ 1) are enforced structurally through specialized flux heads
- **Long-term Stability**: Flux-based updates prevent accumulation of conservation drift
- **Modular Design**: Different flux heads (L, D, P, U, N) for different constraint types

## Repository Structure

```
SupplementaryMaterial/
├── src/
│   ├── models/              # FluxNet variants and baseline models
│   │   ├── fluxnet_l_1d.py  # Lower-bounded 1D FluxNet
│   │   ├── fluxnet_d_1d.py  # Double-bounded 1D FluxNet
│   │   ├── fluxnet_l_2d.py  # Lower-bounded 2D FluxNet
│   │   ├── fluxnet_d_2d.py  # Double-bounded 2D FluxNet
│   │   ├── fluxnet_sw_lap.py # Shallow water FluxNet-LAP
│   │   ├── fno_1d.py        # FNO baseline
│   │   ├── cnn_baseline.py  # CNN baseline
│   │   └── ...
│   ├── training/            # Unified training framework
│   │   ├── trainer_unified.py
│   │   ├── dataloader.py
│   │   └── config.py
│   ├── evaluation/          # Unified evaluation framework
│   │   └── evaluator_unified.py
│   └── utils/
│       └── visualization.py
├── experiments/
│   ├── convection_diffusion/
│   │   └── run_ablation.py
│   ├── traffic_flow/
│   │   └── run_ablation.py
│   ├── shallow_water/
│   │   └── run_ablation.py
│   └── spinodal_decomposition/
│       ├── run_ablation_ndt1.py
│       ├── run_ablation_ndt10.py
│       └── run_ablation_ndt100.py
├── dataset/                 # Place dataset files here
└── results/                 # Experiment results will be saved here
```

## Installation

### Requirements

- Python 3.8+
- PyTorch 1.10+
- CUDA-capable GPU (recommended)

### Setup

```bash
# Create virtual environment
python -m venv venv
source venv/bin/activate  # On Windows: venv\Scripts\activate

# Install dependencies
pip install torch torchvision torchaudio --index-url https://download.pytorch.org/whl/cu118
pip install h5py numpy matplotlib tqdm scikit-learn joblib
```

## Dataset Format

All datasets use HDF5 format. Each file should contain:

### 1D Problems (Convection-Diffusion, Traffic Flow)
- **Convection-Diffusion**: `c` [T, L], `u` [L]
- **Traffic Flow**: `rho` [T, L], `vmax` [L]

### 2D Problems (Shallow Water, Spinodal Decomposition)
- **Shallow Water**: `h` [T, H, W], `mx` [T, H, W], `my` [T, H, W]
- **Spinodal Decomposition**: `phi_data` [T, H, W]

Place your dataset files in:
```
dataset/
├── convection_diffusion/
│   ├── train/
│   ├── val/
│   └── test/
├── traffic_flow/
│   ├── train/
│   ├── val/
│   └── test/
├── shallow_water/
│   ├── train/
│   ├── val/
│   └── test/
└── spinodal_decomposition/
    ├── train/
    ├── val/
    └── test/
```

## Quick Start

### Running Experiments

Each experiment directory contains a `run_ablation.py` script:

```bash
# Traffic flow experiments
python experiments/traffic_flow/run_ablation.py

# Convection-diffusion experiments
python experiments/convection_diffusion/run_ablation.py

# Shallow water experiments
python experiments/shallow_water/run_ablation.py

# Spinodal decomposition experiments (different time steps)
python experiments/spinodal_decomposition/run_ablation_ndt1.py   # Short-term
python experiments/spinodal_decomposition/run_ablation_ndt10.py  # Medium-term
python experiments/spinodal_decomposition/run_ablation_ndt100.py # Long-term
```

### Customizing Experiments

Edit the `SELECTED_MODELS` list in each `run_ablation.py` file to choose which models to run:

```python
SELECTED_MODELS = [
    'FluxNet_D_1D_pf',        # Recommended FluxNet variant
    'FNO_1D_pf',              # FNO baseline
    'CNN_1D_residual_pf',     # CNN baseline
]
```

Modify hyperparameters in the `hparams` dictionary:

```python
hparams = {
    'base_channels': 64,
    'num_blocks': 6,
    'kernel_size': 3,
    'neighborhood_size': 15,
    'learning_rate': 1e-3,
    'num_epochs': 100,
    'batch_size': 16,
    # ...
}
```

## FluxNet Variants

| Model | Constraint Type | Flux Form | Use Cases |
|-------|----------------|-----------|-----------|
| FluxNet-N | None | Unrestricted | Baseline comparison |
| FluxNet-P | Positive flux | softplus ≥ 0 | One-way transport |
| FluxNet-L | Lower bound | Outflow % × available | Concentration, water depth |
| FluxNet-D | Double bound | Dual (outflow + inflow) | Density, phase field |
| FluxNet-U | Upper bound | Inflow % × capacity | Special ablation |
| FluxNet-LAP | Shallow water | L-head + Advection-Pressure | Coupled shallow water |

## Evaluation Metrics

The evaluation framework computes:

**Prediction Accuracy:**
- MAE (Mean Absolute Error)
- RMSE (Root Mean Squared Error)
- MAE@T (at specific time points: T=0.25, 0.5, 0.75, 1.0)

**Conservation Performance:**
- Conservation Drift (relative error in total mass)
- Max Conservation Drift (worst case)

**Bound Violation:**
- Violation Rate (% of grid points violating bounds)
- Conditional Mean OOB Magnitude (average violation magnitude for violating points)

## Results

After running experiments, results are saved in:

```
results/
├── {experiment_name}/
│   ├── {model_name}/
│   │   ├── best_model.pt
│   │   ├── config.json
│   │   ├── loss_curve.pkl
│   │   ├── loss_curve.png
│   │   └── evaluation/
│   │       ├── onestep/
│   │       │   ├── test_set_summary_onestep.json
│   │       │   └── trajectories/
│   │       └── rollout/
│   │           ├── test_set_summary_rollout.json
│   │           └── trajectories/
│   └── ablation_summary.md
```

## Training Modes

### One-Step Training
Trains the model to predict the next state from the current state.

### Pushforward Training
Improves long-term stability by:
1. Computing one-step loss on immediate prediction
2. Rolling out N steps (without gradients)
3. Computing stability loss at step N (with gradients)
4. Total loss = one-step loss + stability loss

Enable in config:
```python
training_config = TrainingConfig(
    use_pushforward=True,
    unroll_steps=5
)
```

## Soft Conservation Loss (Baseline Models Only)

For baseline models (CNN, FNO without flux heads), you can add a soft conservation penalty:

```python
training_config = TrainingConfig(
    soft_conservation_weight=0.1  # Weight for conservation loss
)
```

**Note**: FluxNet models guarantee conservation structurally, so this is not needed.

## Dual Consistency Loss (FluxNet-D Only)

FluxNet-D uses a dual approach (outflow + inflow perspectives). The Dual Consistency Loss (DCL) encourages agreement:

```python
training_config = TrainingConfig(
    dcl_weight=0.1  # Weight for dual consistency loss
)
```
