# KIND: Blending Stationary and Transient Koopman Dynamics with Learned Uncertainty

This repository features the Python code of our Kalman-Inspired Neural Decomposition (KIND) framework, as well as Jupyter
notebooks that were used to produce results reported in the paper.
The notebooks are provided in an 'executed' state, so they should display all plots and other information upon opening.

## Requirements

The main required Python packages are listed in a conda-environment file, named environment.yaml.
If mamba is already installed, the above environment can be imported using command:
mamba env create --file environment.yaml

## Structure of KIND

The main KIND's code is located in kind.py file. There are also helping utilities located in utils_*.py files.

## Structure of Jupyter notebooks

There are three main groups of notebooks that cover the three datasets: TESLA, ETTh2 and M4 Hourly. Accordingly,
there are notebooks, named such as 01_tesla_data_sim.ipynb, or, e.g., 15_et_data_filter.ipynb, and, finally,
24_m4_h_kind_train.ipynb. The notebooks use numbers in the beginnning of their names to denote the logical
order of their execution.

## Notes on external code

### Online DMD

The OnlineDMD class used in this project is adapted from the official GitHub repository:
https://github.com/haozhg/odmd

For convenience and clarity, the class is included directly in a Jupyter notebook 06_tesla_odmd_eval.ipynb, along with inline comments referencing the original source.

### Koopa

The Koopa baseline implementation is adapted from its official GitHub repository:
https://github.com/thuml/Koopa

The code is included in the external/Koopa/ directory for ease of integration and reproducibility.

Some modifications were made to support fair comparison and evaluation in our experimental setting:

* Metric computation: Adjusted the evaluation pipeline to compute MSE and MAE in real units (i.e., unscaled), bypassing Koopa’s default use of double or single scaling during inference. See exp_main.py, line 228.

* Test slicing: Modified the code to allow testing on custom data slices, such as rising or falling slope segments, instead of using the hardcoded test set boundaries. See data_loader.py, line 54.

All changes are documented in the code via inline comments.