{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "ab12f31b-b486-4d80-a6ea-7ef47ac614a8",
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "b11186f2-ec68-4943-b13a-c4321b74857d",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/xxx/GitHub/Aligning-Flow-Div-User-Defined-Sampling/.venv/lib/python3.13/site-packages/tqdm/auto.py:21: TqdmWarning: IProgress not found. Please update jupyter and ipywidgets. See https://ipywidgets.readthedocs.io/en/stable/user_install.html\n",
      "  from .autonotebook import tqdm as notebook_tqdm\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "import pprint\n",
    "\n",
    "import seaborn as sns\n",
    "sns.set_theme(style='whitegrid', font_scale=1.3, palette=sns.color_palette(),)\n",
    "import matplotlib.pyplot as plt\n",
    "import duckdb\n",
    "import polars as pl\n",
    "import numpy as np\n",
    "import hydra\n",
    "from omegaconf import OmegaConf\n",
    "import numpy as np\n",
    "import jax\n",
    "\n",
    "import conf.conf\n",
    "import userfm.datasets\n",
    "import userfm.event_constraints\n",
    "import userfm.plots\n",
    "import userfm.utils"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "9dca96e7-4f85-40e0-aa8b-7830f85faa41",
   "metadata": {},
   "outputs": [],
   "source": [
    "os.environ['CUDA_VISIBLE_DEVICES'] = '7'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "a4e979e7-8d03-4c30-890e-aa64b18ce099",
   "metadata": {},
   "outputs": [],
   "source": [
    "duckdb.sql(\"\"\"\n",
    "attach '../runs.sqlite';\n",
    "use runs;\n",
    "\"\"\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "26e1f809-5d5a-4a6d-92e4-bfec03925519",
   "metadata": {},
   "source": [
    "# Dynamical system"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f4bd882f-8616-4253-85f4-b01d99c8532d",
   "metadata": {},
   "source": [
    "### Lorenz '63"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "97f17396-47a3-43f0-86f6-62b5deca1444",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "┌────────────┬───────────────────┬─────────────────┬──────────────────┬───┬────────────────────┬───────────┬───────┐\n",
      "│ batch_size │ batch_count_train │ batch_count_val │ batch_count_test │ … │        beta        │ rescaling │  id   │\n",
      "│   int64    │       int64       │      int64      │      int64       │   │       double       │  double   │ int64 │\n",
      "├────────────┼───────────────────┼─────────────────┼──────────────────┼───┼────────────────────┼───────────┼───────┤\n",
      "│        500 │                 0 │               0 │                0 │ … │ 2.6666666666666665 │      20.0 │     4 │\n",
      "├────────────┴───────────────────┴─────────────────┴──────────────────┴───┴────────────────────┴───────────┴───────┤\n",
      "│ 1 rows                                                                                      16 columns (7 shown) │\n",
      "└──────────────────────────────────────────────────────────────────────────────────────────────────────────────────┘\n",
      "\n"
     ]
    }
   ],
   "source": [
    "dataset_name = 'Lorenz63'\n",
    "dataset_rows = duckdb.sql(\"\"\"\n",
    "select\n",
    "    *\n",
    "from Dataset\n",
    "join DynamicalSystem on Dataset.id = DynamicalSystem.id\n",
    "join Lorenz63 on DynamicalSystem.id = Lorenz63.id\n",
    "where true\n",
    "and batch_count_train = 0\n",
    "and batch_count_val = 0\n",
    "and batch_count_test = 0\n",
    "\"\"\")\n",
    "dataset_rows.show(max_width=120)\n",
    "assert len(dataset_rows) == 1"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ebbb5bed-9af7-427d-9520-47693f267f90",
   "metadata": {},
   "source": [
    "### FitzHugh-Nagumo"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "89d6299e-8dae-4555-abac-82996d749abc",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "┌────────────┬───────────────────┬─────────────────┬──────────────────┬───┬────────┬────────────┬────────────┬───────┐\n",
      "│ batch_size │ batch_count_train │ batch_count_val │ batch_count_test │ … │   k    │ coupling12 │ coupling21 │  id   │\n",
      "│   int64    │       int64       │      int64      │      int64       │   │ double │   double   │   double   │ int64 │\n",
      "├────────────┼───────────────────┼─────────────────┼──────────────────┼───┼────────┼────────────┼────────────┼───────┤\n",
      "│        500 │                 0 │               0 │                0 │ … │  0.128 │        1.0 │        1.0 │     5 │\n",
      "├────────────┴───────────────────┴─────────────────┴──────────────────┴───┴────────┴────────────┴────────────┴───────┤\n",
      "│ 1 rows                                                                                        21 columns (8 shown) │\n",
      "└────────────────────────────────────────────────────────────────────────────────────────────────────────────────────┘\n",
      "\n"
     ]
    }
   ],
   "source": [
    "dataset_name = 'FitzHughNagumo'\n",
    "dataset_rows = duckdb.sql(\"\"\"\n",
    "select\n",
    "    *\n",
    "from Dataset\n",
    "join DynamicalSystem on Dataset.id = DynamicalSystem.id\n",
    "join FitzHughNagumo on DynamicalSystem.id = FitzHughNagumo.id\n",
    "where true\n",
    "and batch_count_train = 0\n",
    "and batch_count_val = 0\n",
    "and batch_count_test = 0\n",
    "\"\"\")\n",
    "dataset_rows.show(max_width=120)\n",
    "assert len(dataset_rows) == 1"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "88a402ac-f94a-47da-93c0-49293241e442",
   "metadata": {},
   "source": [
    "# Model"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "319ad1c0-1792-4778-9caf-80cdf4841448",
   "metadata": {},
   "source": [
    "### Probability path"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3347fe1b-edf0-458d-b4f9-3a3bf7135906",
   "metadata": {},
   "source": [
    "##### Conditional optimal transport"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3cc09696-f909-4a91-9568-cc8bfd5b9256",
   "metadata": {},
   "outputs": [],
   "source": [
    "probability_path_rows = duckdb.sql(\"\"\"\n",
    "select\n",
    "    *\n",
    "from ProbabilityPath\n",
    "join ConditionalOT on ProbabilityPath.id = ConditionalOT.id\n",
    "\"\"\")\n",
    "probability_path_rows.show()\n",
    "assert len(probability_path_rows) == 1"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f5ba7ab0-e996-4cda-80c0-dcb3e4aed8ef",
   "metadata": {},
   "source": [
    "##### Variance exploding"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "54f75614-ab9e-496d-9d69-d39ec84ba005",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "┌───────────────────┬───────┬──────────┬───────────┬───────────┬────────────────────────┬───────┐\n",
      "│  sa_inheritance   │  id   │ time_min │ sigma_min │ sigma_max │ finzi_karras_weighting │  id   │\n",
      "│      varchar      │ int64 │  double  │  double   │  double   │         int64          │ int64 │\n",
      "├───────────────────┼───────┼──────────┼───────────┼───────────┼────────────────────────┼───────┤\n",
      "│ VarianceExploding │     1 │    0.001 │     0.001 │     300.0 │                      1 │     1 │\n",
      "└───────────────────┴───────┴──────────┴───────────┴───────────┴────────────────────────┴───────┘\n",
      "\n"
     ]
    }
   ],
   "source": [
    "probability_path_rows = duckdb.sql(\"\"\"\n",
    "select\n",
    "    *\n",
    "from ProbabilityPath\n",
    "join VarianceExploding on ProbabilityPath.id = VarianceExploding.id\n",
    "\"\"\")\n",
    "probability_path_rows.show()\n",
    "assert len(probability_path_rows) == 1"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "af296184-c0c3-4d55-b3e1-facdcbdf3a49",
   "metadata": {},
   "source": [
    "### Steering"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3b0e0724-7877-421b-93d6-afd63e68314a",
   "metadata": {},
   "source": [
    "##### No steering"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "5df219a3-7601-4936-af7a-9f70df3e36f7",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "┌─────────────┬─────────────────────┬────────────────┬───────┬────────────────────┬─────────────────────────┬───────┬────────────────┐\n",
      "│ batch_count │ sampling_step_count │ sa_inheritance │  id   │ penalization_power │ symmetric_binary_reward │  id   │     solver     │\n",
      "│    int64    │        int64        │    varchar     │ int64 │       double       │          int64          │ int64 │    varchar     │\n",
      "├─────────────┼─────────────────────┼────────────────┼───────┼────────────────────┼─────────────────────────┼───────┼────────────────┤\n",
      "│          64 │                 100 │ NoSteering     │     5 │                0.0 │                       1 │     5 │ HEUN           │\n",
      "│          64 │                 100 │ NoSteering     │   309 │                0.0 │                       1 │   309 │ EULER_MARUYAMA │\n",
      "└─────────────┴─────────────────────┴────────────────┴───────┴────────────────────┴─────────────────────────┴───────┴────────────────┘\n",
      "\n"
     ]
    }
   ],
   "source": [
    "no_steering_rows = duckdb.sql(\"\"\"\n",
    "select\n",
    "    *\n",
    "from Steering\n",
    "join NoSteering on Steering.id = NoSteering.id\n",
    "where true\n",
    "and batch_count = 64\n",
    "and sampling_step_count = 100\n",
    "\"\"\")\n",
    "no_steering_rows.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "931d2d05-e30d-4d74-aa20-35aa602fa019",
   "metadata": {},
   "source": [
    "##### Source parallel tempering"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "342eed56-51ab-42d6-8f69-64ffacf31166",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "┌─────────────┬─────────────────────┬──────────────────────┬───┬──────────────┬──────────────────┬────────┐\n",
      "│ batch_count │ sampling_step_count │    sa_inheritance    │ … │ model_alt_id │ dynamical_system │  tilt  │\n",
      "│    int64    │        int64        │       varchar        │   │   varchar    │     varchar      │ double │\n",
      "├─────────────┼─────────────────────┼──────────────────────┼───┼──────────────┼──────────────────┼────────┤\n",
      "│          64 │                 100 │ SourceParallelTemp…  │ … │ o53qnpz8     │ Lorenz63         │   8.29 │\n",
      "│          64 │                 100 │ SourceParallelTemp…  │ … │ aezlb9vb     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ SourceParallelTemp…  │ … │ amh4m6kc     │ Lorenz63         │   8.29 │\n",
      "│          64 │                 100 │ SourceParallelTemp…  │ … │ rn8n5ie5     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ SourceParallelTemp…  │ … │ esvulji4     │ Lorenz63         │   8.29 │\n",
      "│          64 │                 100 │ SourceParallelTemp…  │ … │ 5e8mgjrk     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ SourceParallelTemp…  │ … │ hll6nt83     │ Lorenz63         │   8.29 │\n",
      "│          64 │                 100 │ SourceParallelTemp…  │ … │ 4zyq5rxi     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ SourceParallelTemp…  │ … │ cka3dfcm     │ Lorenz63         │   8.29 │\n",
      "│          64 │                 100 │ SourceParallelTemp…  │ … │ zfaz2cku     │ FitzHughNagumo   │  10.27 │\n",
      "│           · │                  ·  │          ·           │ · │    ·         │       ·          │    ·   │\n",
      "│           · │                  ·  │          ·           │ · │    ·         │       ·          │    ·   │\n",
      "│           · │                  ·  │          ·           │ · │    ·         │       ·          │    ·   │\n",
      "│          64 │                 100 │ SourceParallelTemp…  │ … │ zfaz2cku     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ SourceParallelTemp…  │ … │ 18r95hcg     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ SourceParallelTemp…  │ … │ dvpxg997     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ SourceParallelTemp…  │ … │ p3k9ezab     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ SourceParallelTemp…  │ … │ wzd02b2j     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ SourceParallelTemp…  │ … │ kz6qdbip     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ SourceParallelTemp…  │ … │ rfklprab     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ SourceParallelTemp…  │ … │ 12f3bglg     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ SourceParallelTemp…  │ … │ glez9y33     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ SourceParallelTemp…  │ … │ q3um6je6     │ FitzHughNagumo   │  10.27 │\n",
      "├─────────────┴─────────────────────┴──────────────────────┴───┴──────────────┴──────────────────┴────────┤\n",
      "│ 132 rows (20 shown)                                                                17 columns (6 shown) │\n",
      "└─────────────────────────────────────────────────────────────────────────────────────────────────────────┘\n",
      "\n"
     ]
    }
   ],
   "source": [
    "spt_steering_rows = duckdb.sql(\"\"\"\n",
    "select\n",
    "    *\n",
    "from Steering\n",
    "join SourceParallelTempering on Steering.id = SourceParallelTempering.id\n",
    "join BestSPTParameter on (\n",
    "    SourceParallelTempering.update_count = BestSPTParameter.update_count\n",
    "    and\n",
    "    SourceParallelTempering.chain_count = BestSPTParameter.chain_count\n",
    ")\n",
    "join SweepTrainedModel on (\n",
    "    SourceParallelTempering.tilt = SweepTrainedModel.tilt\n",
    "    and\n",
    "    BestSPTParameter.dynamical_system = SweepTrainedModel.dynamical_system\n",
    ")\n",
    "where true\n",
    "and symmetric_binary_reward\n",
    "\"\"\")\n",
    "spt_steering_rows.show(max_width=120)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7995f6f7-301a-4248-b0fa-7068c5587c89",
   "metadata": {},
   "source": [
    "##### Feynmann-Kac"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "62b13086-0a24-45d6-9f57-3e90a056e15c",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "┌─────────────┬─────────────────────┬────────────────┬───┬──────────────┬──────────────────┬────────┐\n",
      "│ batch_count │ sampling_step_count │ sa_inheritance │ … │ model_alt_id │ dynamical_system │  tilt  │\n",
      "│    int64    │        int64        │    varchar     │   │   varchar    │     varchar      │ double │\n",
      "├─────────────┼─────────────────────┼────────────────┼───┼──────────────┼──────────────────┼────────┤\n",
      "│          64 │                 100 │ FeynmannKac    │ … │ o53qnpz8     │ Lorenz63         │   8.29 │\n",
      "│          64 │                 100 │ FeynmannKac    │ … │ aezlb9vb     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ FeynmannKac    │ … │ amh4m6kc     │ Lorenz63         │   8.29 │\n",
      "│          64 │                 100 │ FeynmannKac    │ … │ rn8n5ie5     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ FeynmannKac    │ … │ esvulji4     │ Lorenz63         │   8.29 │\n",
      "│          64 │                 100 │ FeynmannKac    │ … │ 5e8mgjrk     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ FeynmannKac    │ … │ hll6nt83     │ Lorenz63         │   8.29 │\n",
      "│          64 │                 100 │ FeynmannKac    │ … │ 4zyq5rxi     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ FeynmannKac    │ … │ cka3dfcm     │ Lorenz63         │   8.29 │\n",
      "│          64 │                 100 │ FeynmannKac    │ … │ zfaz2cku     │ FitzHughNagumo   │  10.27 │\n",
      "│           · │                  ·  │      ·         │ · │    ·         │    ·             │     ·  │\n",
      "│           · │                  ·  │      ·         │ · │    ·         │    ·             │     ·  │\n",
      "│           · │                  ·  │      ·         │ · │    ·         │    ·             │     ·  │\n",
      "│          64 │                 100 │ FeynmannKac    │ … │ gwxzkawv     │ Lorenz63         │   8.29 │\n",
      "│          64 │                 100 │ FeynmannKac    │ … │ go0sty6p     │ Lorenz63         │   8.29 │\n",
      "│          64 │                 100 │ FeynmannKac    │ … │ orok93n8     │ Lorenz63         │   8.29 │\n",
      "│          64 │                 100 │ FeynmannKac    │ … │ wzd02b2j     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ FeynmannKac    │ … │ kz6qdbip     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ FeynmannKac    │ … │ rfklprab     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ FeynmannKac    │ … │ lvsc0813     │ Lorenz63         │   8.29 │\n",
      "│          64 │                 100 │ FeynmannKac    │ … │ 12f3bglg     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ FeynmannKac    │ … │ glez9y33     │ FitzHughNagumo   │  10.27 │\n",
      "│          64 │                 100 │ FeynmannKac    │ … │ q3um6je6     │ FitzHughNagumo   │  10.27 │\n",
      "├─────────────┴─────────────────────┴────────────────┴───┴──────────────┴──────────────────┴────────┤\n",
      "│ 104 rows (20 shown)                                                          21 columns (6 shown) │\n",
      "└───────────────────────────────────────────────────────────────────────────────────────────────────┘\n",
      "\n"
     ]
    }
   ],
   "source": [
    "fk_steering_rows = duckdb.sql(\"\"\"\n",
    "select\n",
    "    *\n",
    "from Steering\n",
    "join FeynmannKac on Steering.id = FeynmannKac.id\n",
    "join FeynmannKacIntermediateReward on FeynmannKac.FeynmannKacIntermediateReward = FeynmannKacIntermediateReward.id\n",
    "join FeynmannKacIntermediateRewardExpectedSample on FeynmannKacIntermediateReward.id = FeynmannKacIntermediateRewardExpectedSample.id\n",
    "join BestFKParameter on (\n",
    "    FeynmannKac.potential = BestFKParameter.potential\n",
    "    and\n",
    "    FeynmannKac.ensemble_size = BestFKParameter.ensemble_size\n",
    ")\n",
    "join SweepTrainedModel on (\n",
    "    FeynmannKac.tilt = SweepTrainedModel.tilt\n",
    "    and\n",
    "    BestFKParameter.dynamical_system = SweepTrainedModel.dynamical_system\n",
    ")\n",
    "where true\n",
    "and symmetric_binary_reward\n",
    "\"\"\")\n",
    "fk_steering_rows.show(max_width=120)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8c294185-7ec0-4469-a294-52de021b4714",
   "metadata": {},
   "source": [
    "### Model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "fb6bd849-5046-454d-bc20-76533a4ea5db",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "┌────────────────┬───────┬─────────────┬────────┬───┬──────────┬───────────┬───────────┬──────────────────────┬───────┐\n",
      "│ sa_inheritance │  id   │ epoch_count │   lr   │ … │ time_min │ sigma_min │ sigma_max │ finzi_karras_weigh…  │ id_1  │\n",
      "│    varchar     │ int64 │    int64    │ double │   │  double  │  double   │  double   │        int64         │ int64 │\n",
      "├────────────────┼───────┼─────────────┼────────┼───┼──────────┼───────────┼───────────┼──────────────────────┼───────┤\n",
      "│ FlowMatching   │     1 │        2000 │ 0.0001 │ … │    0.001 │     0.001 │     300.0 │                    1 │     1 │\n",
      "├────────────────┴───────┴─────────────┴────────┴───┴──────────┴───────────┴───────────┴──────────────────────┴───────┤\n",
      "│ 1 rows                                                                                         17 columns (9 shown) │\n",
      "└─────────────────────────────────────────────────────────────────────────────────────────────────────────────────────┘\n",
      "\n"
     ]
    }
   ],
   "source": [
    "model_rows = duckdb.sql(\"\"\"\n",
    "select\n",
    "    *\n",
    "from Model\n",
    "join FlowMatching on Model.id = FlowMatching.id\n",
    "join probability_path_rows on FlowMatching.ProbabilityPath = probability_path_rows.id\n",
    "\"\"\")\n",
    "model_rows.show(max_width=120)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "12455c64-99dc-47ca-9c8b-9349367070aa",
   "metadata": {},
   "source": [
    "# Rows"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "a000e32b-23d6-411b-a900-47ea718f0df1",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "┌──────────┬───────┬─────────────────────────┬────────────────────┬────────────┐\n",
      "│  alt_id  │  id   │        steering         │ penalization_power │ batch_size │\n",
      "│ varchar  │ int64 │         varchar         │       double       │   int64    │\n",
      "├──────────┼───────┼─────────────────────────┼────────────────────┼────────────┤\n",
      "│ 5utbygnl │   298 │ SourceParallelTempering │                0.0 │        500 │\n",
      "│ flyyddk9 │   247 │ FeynmannKac             │                0.0 │        500 │\n",
      "└──────────┴───────┴─────────────────────────┴────────────────────┴────────────┘\n",
      "\n"
     ]
    }
   ],
   "source": [
    "rows = duckdb.sql(\"\"\"\n",
    "select\n",
    "    Conf.alt_id,\n",
    "    steering_rows.*,\n",
    "    dataset_rows.batch_size,\n",
    "from Conf\n",
    "join TrainingAndEvaluation on Conf.id = TrainingAndEvaluation.id\n",
    "join dataset_rows on TrainingAndEvaluation.Dataset = dataset_rows.id\n",
    "join Trained on TrainingAndEvaluation.Model = Trained.id\n",
    "join (\n",
    "    --select id, format('{} ({})', sa_inheritance, solver) as steering, 0 as penalization_power from no_steering_rows\n",
    "    --union\n",
    "    select id, sa_inheritance as steering, penalization_power from spt_steering_rows\n",
    "    union\n",
    "    select id, sa_inheritance as steering, penalization_power from fk_steering_rows\n",
    ") as steering_rows on TrainingAndEvaluation.Steering = steering_rows.id\n",
    "join TrainingAndEvaluation as TrainingAndEvaluation_trained on Trained.Conf = TrainingAndEvaluation_trained.id\n",
    "join model_rows on TrainingAndEvaluation_trained.Model = model_rows.id\n",
    "where true\n",
    "and TrainingAndEvaluation.log_constraint_values\n",
    "and steering_rows.penalization_power = 0\n",
    "\"\"\")\n",
    "rows.show(max_width=120)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "b6495dad-4f49-4040-b724-9154270bc6c9",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "TrainingAndEvaluation(root_dir='/home/xxx/GitHub/Aligning-Flow-Div-User-Defined-Sampling',\n",
      "                      out_dir='/home/xxx/out/user-defined-events',\n",
      "                      run_subdir='runs',\n",
      "                      prediction_filename='output',\n",
      "                      device='cuda',\n",
      "                      alt_id='n9uzxfjt',\n",
      "                      rng_seed=2376999025,\n",
      "                      id=378,\n",
      "                      fit=True,\n",
      "                      predict=False,\n",
      "                      predict_split=<PredictSplit.VAL: 'val'>,\n",
      "                      log_constraint_values=False,\n",
      "                      dataset=FitzHughNagumo(device_batch_size=500,\n",
      "                                             batch_size=500,\n",
      "                                             batch_count_train=0,\n",
      "                                             batch_count_val=0,\n",
      "                                             batch_count_test=64,\n",
      "                                             id=29,\n",
      "                                             time_step=6.0,\n",
      "                                             time_step_count=310,\n",
      "                                             time_step_count_drop_first=250,\n",
      "                                             odeint_rtol=1e-06,\n",
      "                                             a1=-0.025794,\n",
      "                                             a2=-0.025794,\n",
      "                                             b1=0.0065,\n",
      "                                             b2=0.0135,\n",
      "                                             c1=0.02,\n",
      "                                             c2=0.02,\n",
      "                                             k=0.128,\n",
      "                                             coupling12=1.0,\n",
      "                                             coupling21=1.0),\n",
      "                      model=FlowMatching(id=1,\n",
      "                                         epoch_count=2000,\n",
      "                                         lr=0.0001,\n",
      "                                         lr_decay=0.995,\n",
      "                                         ema_folding_count=5,\n",
      "                                         base_channel_count=32,\n",
      "                                         use_attention=False,\n",
      "                                         probability_path=VarianceExploding(id=1,\n",
      "                                                                            time_min=0.001,\n",
      "                                                                            sigma_min=0.001,\n",
      "                                                                            sigma_max=300.0,\n",
      "                                                                            finzi_karras_weighting=True)),\n",
      "                      steering=NoSteering(batch_count=1,\n",
      "                                          sampling_step_count=10,\n",
      "                                          penalization_power=0.0,\n",
      "                                          symmetric_binary_reward=True,\n",
      "                                          id=318,\n",
      "                                          solver=<Solver.HEUN: 'heun'>))\n"
     ]
    }
   ],
   "source": [
    "with conf.conf.Session() as db:\n",
    "    with hydra.initialize(version_base=None, config_path='../conf'):\n",
    "        cfg = hydra.compose(config_name='training_and_evaluation', overrides=[\n",
    "            f'+experiment=Train{dataset_name}',\n",
    "            'dataset.batch_count_train=0',\n",
    "            'dataset.batch_count_val=0',\n",
    "            'dataset.batch_count_test=64',\n",
    "        ])\n",
    "        cfg = conf.conf.orm.instantiate_and_insert_config(db, OmegaConf.to_container(cfg, resolve=True))\n",
    "        pprint.pp(cfg)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "59eb633b-74d6-4c15-881c-e8fd7172201b",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 64/64 [00:24<00:00,  2.57it/s]\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "(32000, 60, 4)"
      ]
     },
     "execution_count": 32,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "splits = ('train', 'val', 'test')\n",
    "(\n",
    "    *key_dataset_splits,\n",
    "    key_jax_lightning,\n",
    ") = jax.random.split(\n",
    "    jax.random.key(userfm.utils.RNG_RANDBITS[cfg.rng_seed]),\n",
    "    len(splits) + 1\n",
    ")\n",
    "splits = {\n",
    "    s: userfm.datasets.get_dataset(cfg.dataset, k, batch_count)\n",
    "    for s, k in zip(splits, key_dataset_splits)\n",
    "    if (batch_count := getattr(cfg.dataset, f'batch_count_{s}')) > 0\n",
    "}\n",
    "splits['test'].trajectories.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "7aede04a-30f0-4652-852b-a12a6b81483e",
   "metadata": {},
   "outputs": [],
   "source": [
    "event_constraint = userfm.event_constraints.get_event_constraint(cfg.dataset)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "8e905c5b-15fb-445e-bfe6-22c6289bc543",
   "metadata": {},
   "outputs": [],
   "source": [
    "ground_truth = pl.DataFrame(dict(\n",
    "    alt_id='',\n",
    "    steering='GroundTruth',\n",
    "    constraint_value=np.asarray(event_constraint.constraint(splits['test']), dtype=np.float64),\n",
    "))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "33080070-a86f-4076-8446-8b43de1ee43b",
   "metadata": {},
   "outputs": [],
   "source": [
    "ground_truth_range = (\n",
    "    ground_truth\n",
    "    .filter(steering='GroundTruth')\n",
    "    .select(\n",
    "        pl.col('constraint_value').min().alias('min'),\n",
    "        pl.col('constraint_value').max().alias('max'),\n",
    "    )\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "id": "883437f7-f316-4ac1-83bf-6e7ed04e769e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "┌──────────┬───────┬──────────────────────┬───┬──────────────────────┬───────┬──────────────────────┬──────────┐\n",
      "│  alt_id  │  id   │       steering       │ … │ satisfactory_sampl…  │ step  │      file_path       │  alt_id  │\n",
      "│ varchar  │ int64 │       varchar        │   │        double        │ int64 │       varchar        │ varchar  │\n",
      "├──────────┼───────┼──────────────────────┼───┼──────────────────────┼───────┼──────────────────────┼──────────┤\n",
      "│ flyyddk9 │   247 │ FeynmannKac          │ … │                 22.0 │    63 │ /home/xxx/out…  │ flyyddk9 │\n",
      "│ 5utbygnl │   298 │ SourceParallelTemp…  │ … │                408.0 │    63 │ /home/xxx/out…  │ 5utbygnl │\n",
      "│ flyyddk9 │   247 │ FeynmannKac          │ … │                 18.0 │    62 │ /home/xxx/out…  │ flyyddk9 │\n",
      "│ 5utbygnl │   298 │ SourceParallelTemp…  │ … │                399.0 │    62 │ /home/xxx/out…  │ 5utbygnl │\n",
      "│ flyyddk9 │   247 │ FeynmannKac          │ … │                 24.0 │    61 │ /home/xxx/out…  │ flyyddk9 │\n",
      "│ 5utbygnl │   298 │ SourceParallelTemp…  │ … │                419.0 │    61 │ /home/xxx/out…  │ 5utbygnl │\n",
      "│ flyyddk9 │   247 │ FeynmannKac          │ … │                 18.0 │    60 │ /home/xxx/out…  │ flyyddk9 │\n",
      "│ 5utbygnl │   298 │ SourceParallelTemp…  │ … │                415.0 │    60 │ /home/xxx/out…  │ 5utbygnl │\n",
      "│ flyyddk9 │   247 │ FeynmannKac          │ … │                 18.0 │    59 │ /home/xxx/out…  │ flyyddk9 │\n",
      "│ 5utbygnl │   298 │ SourceParallelTemp…  │ … │                419.0 │    59 │ /home/xxx/out…  │ 5utbygnl │\n",
      "│    ·     │    ·  │      ·               │ · │                   ·  │     · │          ·           │    ·     │\n",
      "│    ·     │    ·  │      ·               │ · │                   ·  │     · │          ·           │    ·     │\n",
      "│    ·     │    ·  │      ·               │ · │                   ·  │     · │          ·           │    ·     │\n",
      "│ flyyddk9 │   247 │ FeynmannKac          │ … │                 23.0 │     4 │ /home/xxx/out…  │ flyyddk9 │\n",
      "│ 5utbygnl │   298 │ SourceParallelTemp…  │ … │                395.0 │     4 │ /home/xxx/out…  │ 5utbygnl │\n",
      "│ flyyddk9 │   247 │ FeynmannKac          │ … │                 16.0 │     3 │ /home/xxx/out…  │ flyyddk9 │\n",
      "│ 5utbygnl │   298 │ SourceParallelTemp…  │ … │                397.0 │     3 │ /home/xxx/out…  │ 5utbygnl │\n",
      "│ flyyddk9 │   247 │ FeynmannKac          │ … │                 18.0 │     2 │ /home/xxx/out…  │ flyyddk9 │\n",
      "│ 5utbygnl │   298 │ SourceParallelTemp…  │ … │                407.0 │     2 │ /home/xxx/out…  │ 5utbygnl │\n",
      "│ flyyddk9 │   247 │ FeynmannKac          │ … │                 10.0 │     1 │ /home/xxx/out…  │ flyyddk9 │\n",
      "│ 5utbygnl │   298 │ SourceParallelTemp…  │ … │                411.0 │     1 │ /home/xxx/out…  │ 5utbygnl │\n",
      "│ flyyddk9 │   247 │ FeynmannKac          │ … │                 16.0 │     0 │ /home/xxx/out…  │ flyyddk9 │\n",
      "│ 5utbygnl │   298 │ SourceParallelTemp…  │ … │                388.0 │     0 │ /home/xxx/out…  │ 5utbygnl │\n",
      "├──────────┴───────┴──────────────────────┴───┴──────────────────────┴───────┴──────────────────────┴──────────┤\n",
      "│ 128 rows (20 shown)                                                                    509 columns (7 shown) │\n",
      "└──────────────────────────────────────────────────────────────────────────────────────────────────────────────┘\n",
      "\n"
     ]
    }
   ],
   "source": [
    "metrics = (\n",
    "    pl.scan_csv([f'~/out/user-defined-events/runs/{alt_id}/metrics.csv'\n",
    "         for alt_id, in duckdb.sql(\"select alt_id from rows\").fetchall()\n",
    "    ], include_file_paths='file_path')\n",
    "    .with_columns(alt_id=pl.col('file_path').str.split('/').list.get(-2))\n",
    "    .collect()\n",
    ")\n",
    "metrics = duckdb.sql(\"\"\"\n",
    "select\n",
    "    *\n",
    "from rows\n",
    "join metrics on rows.alt_id = metrics.alt_id\n",
    "\"\"\")\n",
    "metrics.show(max_width=120)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "id": "d49cb6a2-f356-480a-9a21-9c2427f526e0",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "┌──────────┬───────┬──────────────────────┬───┬──────────┬──────────────────────┬─────────────────────┐\n",
      "│  alt_id  │  id   │       steering       │ … │ alt_id_1 │        sample        │  constraint_value   │\n",
      "│ varchar  │ int64 │       varchar        │   │ varchar  │       varchar        │       double        │\n",
      "├──────────┼───────┼──────────────────────┼───┼──────────┼──────────────────────┼─────────────────────┤\n",
      "│ flyyddk9 │   247 │ FeynmannKac          │ … │ flyyddk9 │ sample_constraint_0  │ -1.7859289646148682 │\n",
      "│ flyyddk9 │   247 │ FeynmannKac          │ … │ flyyddk9 │ sample_constraint_1  │ -1.6988221406936646 │\n",
      "│ flyyddk9 │   247 │ FeynmannKac          │ … │ flyyddk9 │ sample_constraint_10 │  -1.824821949005127 │\n",
      "│ flyyddk9 │   247 │ FeynmannKac          │ … │ flyyddk9 │ sample_constraint_…  │  -1.832219123840332 │\n",
      "│ flyyddk9 │   247 │ FeynmannKac          │ … │ flyyddk9 │ sample_constraint_…  │ -1.8791265487670898 │\n",
      "│ flyyddk9 │   247 │ FeynmannKac          │ … │ flyyddk9 │ sample_constraint_…  │ -1.8558108806610107 │\n",
      "│ flyyddk9 │   247 │ FeynmannKac          │ … │ flyyddk9 │ sample_constraint_…  │  1.6805553436279297 │\n",
      "│ flyyddk9 │   247 │ FeynmannKac          │ … │ flyyddk9 │ sample_constraint_…  │ -1.8903127908706665 │\n",
      "│ flyyddk9 │   247 │ FeynmannKac          │ … │ flyyddk9 │ sample_constraint_…  │ -1.8626151084899902 │\n",
      "│ flyyddk9 │   247 │ FeynmannKac          │ … │ flyyddk9 │ sample_constraint_…  │ -2.1630771160125732 │\n",
      "│    ·     │    ·  │      ·               │ · │    ·     │          ·           │           ·         │\n",
      "│    ·     │    ·  │      ·               │ · │    ·     │          ·           │           ·         │\n",
      "│    ·     │    ·  │      ·               │ · │    ·     │          ·           │           ·         │\n",
      "│ 5utbygnl │   298 │ SourceParallelTemp…  │ … │ 5utbygnl │ sample_constraint_90 │   1.324601173400879 │\n",
      "│ 5utbygnl │   298 │ SourceParallelTemp…  │ … │ 5utbygnl │ sample_constraint_91 │  -2.140984535217285 │\n",
      "│ 5utbygnl │   298 │ SourceParallelTemp…  │ … │ 5utbygnl │ sample_constraint_92 │  1.5334858894348145 │\n",
      "│ 5utbygnl │   298 │ SourceParallelTemp…  │ … │ 5utbygnl │ sample_constraint_93 │  1.7369461059570312 │\n",
      "│ 5utbygnl │   298 │ SourceParallelTemp…  │ … │ 5utbygnl │ sample_constraint_94 │ -1.8469997644424438 │\n",
      "│ 5utbygnl │   298 │ SourceParallelTemp…  │ … │ 5utbygnl │ sample_constraint_95 │  1.8929314613342285 │\n",
      "│ 5utbygnl │   298 │ SourceParallelTemp…  │ … │ 5utbygnl │ sample_constraint_96 │   1.624342441558838 │\n",
      "│ 5utbygnl │   298 │ SourceParallelTemp…  │ … │ 5utbygnl │ sample_constraint_97 │  -2.444632053375244 │\n",
      "│ 5utbygnl │   298 │ SourceParallelTemp…  │ … │ 5utbygnl │ sample_constraint_98 │  1.7001399993896484 │\n",
      "│ 5utbygnl │   298 │ SourceParallelTemp…  │ … │ 5utbygnl │ sample_constraint_99 │  1.6805272102355957 │\n",
      "├──────────┴───────┴──────────────────────┴───┴──────────┴──────────────────────┴─────────────────────┤\n",
      "│ ? rows (>9999 rows, 20 shown)                                                  11 columns (6 shown) │\n",
      "└─────────────────────────────────────────────────────────────────────────────────────────────────────┘\n",
      "\n"
     ]
    }
   ],
   "source": [
    "metrics_unpivot = duckdb.sql(\"\"\"\n",
    "unpivot metrics\n",
    "on columns('sample_constraint_')\n",
    "into name sample value constraint_value\n",
    "\"\"\")\n",
    "metrics_unpivot.show(max_width=120)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "id": "b765fd5f-5e13-4bfa-8c01-05cf4c315f80",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<seaborn.axisgrid.FacetGrid at 0x7d67391af250>"
      ]
     },
     "execution_count": 38,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAApMAAAJvCAYAAAAnRlqfAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjgsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvwVt1zgAAAAlwSFlzAAAPYQAAD2EBqD+naQAAfG1JREFUeJzt3Xl81PWB//H3HN+ZHEACJKCGIFcFiiesoHhVpFWxuoJH8VwQi20Vqa3Ua7u/0qrQVdfFYLte9b5Qg3dtG494VEED1BYSQAgIESUZSEgyydy/P4YZk8lMjslkJpO8no+HD5zvNZ/JXO/5nKZAIBAQAAAAEAdzqgsAAACA9EWYBAAAQNwIkwAAAIgbYRIAAABxI0wCAAAgboRJAAAAxI0wCQAAgLhZ4znJ4/HI5/MluiwAAABIIYvFIsMwunROl8LkgQMHVFNTI5fL1aU7AQAAQHqw2+3Ky8vToEGDOnV8p8PkgQMHVFVVpQEDBigvL0+GYchkMsVdUAAAAPQegUBAHo9HdXV1qqqqkqROBUpTZ5dT3L59uwzD0IgRIwiRAAAAfVQgENDu3bvl8Xg0ZsyYDo/v1AAcj8cjl8ulnJwcgiQAAEAfZjKZlJOTI5fLJY/H0+HxnQqTocE2Xe2QCQAAgPQTynydGXDdpamBqJUEAADo+7qS+ZhnEgAAAHEjTAIAACBucU1aHk1tba2cTmeiLpdwWVlZys3NTXUxgD6pqrZJ+xvdqS5GTIOzbSrIzUx1MYDYandJTkeqSxFb1lAptzDVpUAvlZAwWVtbq5UrV8rr9Sbicj3CarXquuuuI1D2UsXFxbrlllu0bNkyzZkzJyVlmDFjhiTpnXfeScn9p6uq2ibNuPs9ubz+VBclJrvVrHdu/B6BspcoKirSypUr9cQTT2jatGmpLk7q1e6SVv6b5G1OdUlis2ZI133WLwPlvffeq0cffVR/+9vfNHz48FQXp8sCgYAuuOACZWRk6JlnnumR+0hImHQ6nfJ6veHJzHsbj8ejmpoaOZ3ObofJ8ePHt7ptGIYGDBigQw89VN/97nf1gx/8QCeffLIsFku37gcdu/nmm7V69epW2zIzMzVixAh973vf049//GPl5OSkqHT9x/5Gt1xev8blZyvT1vte901un76obtT+Rne3wmTkez/S/fffr5kzZ0r6Nizdfvvtuuiii1od5/V6dcstt+jVV1/VMcccowceeECDBw9u99q1tbX605/+pHfffVe7du2Sz+fTkCFDNGLECE2ZMkUXXnihRo4cGT7+iiuu0Nq1a7V58+Y4Hy2SyukIBsm88ZKRlerStOVxSjWbg+XsZphsbm7WE088ob/85S+qrKyU2+1Wbm6uDjvsME2ePFnnnXeevvvd74aPD72XWrLb7Tr00EM1ffp0XXPNNTrkkEOiHteeqVOn6sknn+zwuK+++kqPPfaYLrvssrQMklJwIM3111+va665Rm+99ZbOOuushN9Hwpq5pWCwstvtibxkr5Sdna358+dLCn4x1NfXa+vWrXr55Zf14osv6sgjj9Tdd9+t0aNHp7ik/cO///u/q7CwUIFAQNXV1XrnnXf00EMP6S9/+YteeOGFTv+AeOyxx3q0nH1dps2ibHtCP1J6nZbv/Uideb83NTXp+uuv1/vvv69TTz1V9913nzIz2w+433zzjS655BJVVVXp8MMP13nnnafBgwdr//79+vzzz/XAAw+osLCwVZjs7S677DLNmjVLhx12WKqL0rsYWZJ9QKpL0WMaGxt12WWXqby8XMOGDdPZZ5+tvLw8HThwQJs2bdLjjz8uu93eKkyGnH766Zo0aZIkyeFw6MMPP9QzzzyjP//5z1q1apWmTp2q6667rtU55eXlevvttzVhwoTwD72QgoKCTpX5D3/4gzwej6666qo4H3Xv8L3vfU9HHHGE7r33Xp155pkJn52nb3/y95ABAwZo0aJFbbbv3btXv/vd7/TXv/5V8+fP10svvaShQ4emoIT9y/nnn6/p06eHb9900026+OKLtW3bNj311FNtPmBiSacvY6RGrPd+Z9TW1uqaa67Rhg0b9O///u+68847ZbV2/BF83333qaqqShdddJF+97vftfkS2LVrl9zu3ttfNZohQ4ZoyJAhqS4Gkuzxxx9XeXm5TjnlFP3xj39s05JZXV2tvXv3Rj33jDPOaFXL7/F4tGDBAq1Zs0Z//OMftWzZsjZdJoqLi/X2229r4sSJcb1v6+vr9frrr+ukk05Sfn5+l8/vbc477zzdfffd+vjjj1t9ZyYCo7kTaNiwYfrf//1fTZ06VXv27NH//d//tTlmx44d+tWvfqWTTz5ZRx55pE4++WT96le/0o4dO9oce/PNN2v8+PHatWuXnnzySZ177rk6+uijdcUVV0gKNmWNHz9ebrdbK1as0IwZM3T00Ufr3HPP1Z///GdJwZrTUPPbUUcdpbPOOkuvvfZam/uqr6/XQw89pCuuuEKnnHKKjjzySJ100km6/vrrVVFR0eb44uJijR8/XsXFxfrkk090xRVX6LjjjtPkyZO1cOFCbdu2LerfaOfOnbr++ut1/PHH69hjj9XcuXP13nvvdeGv3LEBAwZo9uzZkqTPP/9ckrRmzRqNHz9eRUVF+vzzz7Vw4UJNnTpV48eP1+7duyUF+0yG+k2GFBUVafz48VqzZo3eeustXXjhhTrmmGM0depU3XDDDfrmm2+ilqG2tlb33nuvfvjDH+qYY47RlClTwm/kyIFq+/fv13//93/rzDPP1FFHHaXjjz9eCxcu1IYNGxL6d0Hq7NmzR5deeqk2bNigq666Sr///e87FSQlaf369ZKCtXnRahMKCws1duxYSdLu3bs1fvx4rV27VlKwaT70380339zqvK1bt+rGG28Mv99POeUU3Xrrrfr666+jluOrr77Sf/3Xf2nGjBk68sgjdeKJJ2rx4sVR3+uh91JDQ4OWLVumGTNmaNKkSSoqKpLU+n0VEir7zTffrN27d+uGG27QtGnTdNRRR2nOnDl69913o5arvr5ed9xxh0499dTwZ9yjjz6qXbt2RX3cSJ3Qa3nu3LlRu8Tl5+eHax87YhiG5s6dK+nbz/lEe/3119XU1KRZs2ZF3e/3+/Xss8/qggsu0LHHHqtjjz1WF1xwgZ599ln5/a37kLd8fUcT+j5vqeX75LXXXtNFF12k4447rtX3VElJia688kqddNJJ4Uxx+eWX69lnn21zH6HH8eKLL3bp79AZ1EwmmMVi0c9+9jOtXbtWb7zxhm699dbwF8Dnn3+u+fPnq7GxUWeccYbGjBmj7du369VXX9Xbb7+txx57TEcddVSba95xxx367LPPdNppp+m0005r0x/zhhtuCP/aCwQCeuONN3TDDTdo4MCBevbZZ1VeXq6TTz5ZkvTGG29oyZIlOuywwzRlypTwNbZt26YVK1Zo6tSpmjFjhgYMGKBdu3bpnXfeUWlpqZ5++mkdeeSRbcr23nvv6e2339Ypp5yiuXPnatu2bSotLdU///lPvfHGG61qH3bs2KEf/ehHqq2t1amnnqqJEydq586duvbaa3XKKack5O8fElpy3mxu/Xtpw4YNeuCBBzRlyhRdcMEF2r9/f6f6+T7zzDN65513NGPGDB1//PH6/PPP9eabb6qiokKvvPKKbDZb+Nhdu3bpP/7jP1RVVaUjjzxSl1xyifx+vyorK/XYY49p7ty5ysrKCh975ZVX6quvvtK0adN0+umnq66uTm+//bYuv/xy3XfffW0CLtLLtm3btGDBAn399ddasmSJrr766i6dH+qmsXPnTk2cOLHdYwcNGqTrrrtOq1evVlVVVata+Zbnvvfee7r++uvl9/s1Y8YMjRgxQrt379bLL7+s999/X88//3yrZsB//vOfWrBggerr63XqqafqzDPPVHV1tf72t7/p/fff1xNPPNHms8vtduvKK69UXV2dTjrpJA0YMEAjRozo8PGGamELCwv17//+76qrq9Obb76pn/3sZ3r00Ud1wgknhI91uVz6j//4D23cuFGTJk3Sueeeq/r6ev3f//2fPvvssw7vC8nV8rWcSJGf84ny8ccfS5KOO+64qPuXLFmi119/XQUFBeFa07ffflu/+c1v9Nlnn+mee+5JSDkeffRRffTRRzr99NM1bdo01dfXS5KeffZZ/eY3v1F+fr5mzJihwYMHy+FwqKKiQi+//LIuueSSVtcpKCjQ8OHD9fe//12BQCChTd2EyR4wZcoUWa1WORwO7d69O9yf76abblJDQ4PuvffeVr903nzzTd1www266aab9MYbb7R5gjdu3KjVq1ersDB6x2eHw6FXX31VAwYE+9qcf/75uuSSS3TDDTdo7NixUfc9/PDDrcLk2LFj9cEHH7QZCLB582bNnTtX9957rx555JE2911SUqJHHnlEJ554YnjbPffcowcffFAvvfSSfvzjH4e3//a3v1Vtba1uvfVW/cd//Eera1x77bUd/l07q6GhITww5+ijj26178MPP9TSpUvDv2g764MPPtCLL77Y6pfjL3/5S73++usqKSlp9XwuWbJEVVVVuvHGG1s9fknat2+fsrOzw7dvuukmff3111q5cqW+//3vh7fv3btXF110kX7961/rpJNO6hd9kdNBQ0NDuHatpTFjxuicc85ps/0f//iH7r77bjU0NGj58uU6//zzu3yfZ599tsrKynTbbbfpX//6l0466SR997vfjTq4bNCgQVq0aJHWrl2rqqqqqE17+/fv14033qjs7Gw9++yzGjVqVHjf2rVrNW/ePN1+++364x//KCnYnPjzn/9cLpdLzzzzTKsv1m3btunCCy/Uf/7nf+qVV15pdT/V1dUaN26cnnrqqfCPp85Yu3atFi1a1CoI//CHP9TVV1+tRx55pFWYfPjhh7Vx40adc845uueee8KfnT/96U/DrRPoPc466yy9+uqr+t///V9VVVXptNNO06RJk5SXl9fla3k8nnDtW+TnfKKUlZVpwIABrd4jIa+//rpef/11HXnkkXryySfDr/EbbrhBl19+uV5//XWdccYZMWs1u+KTTz7R888/36Yv6QsvvCDDMPTKK6+06VK3b9++qNc66qijVFJSom3btmncuHHdLlsIzdw9wGazhX+B7d+/X5K0bt06bd++XZMnT27z4po1a5amTJmibdu2hZsBWrr66qtjBkkp+OINhUVJmjx5sgoLC3XgwIGo+0aMGNFmlOfAgQOjjigdP368pk2bpjVr1kRd7H3WrFmtgqQkXXzxxZKCtRkhX3/9tT766CONGDFCl19+eavjZ86cqalTp8Z8fB15+eWXVVRUpPvuu0+//vWvddZZZ2n79u06/PDD29zXxIkTuxwkpehNEKFfoi0f57/+9S+tX79ekyZNiloDNWTIkHAwLC8vV1lZmc4666xWQVIKdplYsGCBampqwr+OkXqNjY1auXJlm//eeOONqMe/8MILqq2t1bXXXhtXkJSkyy+/XFdffbXcbrceeughzZs3T1OnTtX3v/99/fa3v43aRaY9r7zyiurr63X99de3+ZIMtUy89957amhokCSVlpZq9+7d4a4sLY0dO1YXX3yxKioqtHXr1jb3dfPNN3cpSErB2pOf/vSnrbadcsopOuyww9o0Z7788ssym836xS9+0epH+KGHHtrqByt6hzPOOEO33HKLDMPQ008/rYULF+qkk07SaaedpltuuUX/+te/Yp779ttvq6ioSEVFRVq6dKnOPvtsrV27VkOGDNFPfvKThJfV7XarpqYmZtB96aWXJAUrFVq+xrOysrRkyRJJiWtOvvjii6MOSpKC0x5G6zITq09y6PF89dVXCSlbuBwJvRrCQs2sIZs2bZKkmHOqnXDCCSorK1N5ebkmT57cal9Hv7qivcjy8/O1a9euqPuGDRsWtY/JZ599pscff1wbNmzQ/v3724TH/fv3a9iwYa22RWv6PvTQQyVJdXV14W2hxz9lypSo0yZNnTo13M8r5LHHHgtX54fMnDmzTVNfyxqRzMxMFRYWas6cObr66qs1aNCgVsfG+ws2WveDaI/zH//4hyRp+vTpHTYhhH441NXVRa3tCoWE7du363vf+148xUaCDR8+XO+//36nj58+fbo++eQTPfTQQzruuOPa/PDqDJPJpCVLlujHP/6x3n//fX3++ef617/+pX/+8596+umn9eKLL+ree+/VGWec0anrhfribtq0KerrrqamRn6/Xzt27NCRRx4Zfp3u2rUr6vGVlZWSgq/T73znO+Htdru9w+mUopkwYULUz4hDDjmkVT/ihoYGffnllzr00EOjNp+3bHlB7zFv3jxddNFF+uijj7R+/Xpt3LhRGzZsUHFxsV5++WX913/9V5vmWUl69913w/1mbTabDjvsMF1++eVauHBhj0zZU1tbK0ltvkNCNm3aJLPZrOOPP77NvuOPP14WiyX8vdddsb63zj33XC1fvlznnHOOZs2apalTp2ry5MntDm6LrOhKFMJkD3C5XOGAEXpSQ6Eo1oiw0PYDBw602ddRE8DAgQPbbAv9Uom1L3KC+b/85S/6+c9/royMDJ188skaMWKEMjMzZTKZVFJSooqKiqgjRqO90UL33bIDcujxxxrdHu0xPvHEE6qqqmq1raCgoE2YfPTRRzs9Mi2e5hQp+t8x9IXX8nGGnr/OfLiFXiMfffSRPvroo5jH9eaVpdC+WbNmac6cObrpppv0k5/8RCtXroy7f3Bubq7OO+88nXfeeZKCr5977rlHzz//vG677TadcsoprfruxhJ63a1atard40Kvu9Dxb731VqeODxk6dGhcfbJifXlbrdZW77VQzWmszxRm0ui9srOz9YMf/EA/+MEPJAWnzHr44Ye1cuVK3XHHHZo5c2ab78poc7b2pIyMDEnB7/No6uvrlZOTE7XPvdVq1eDBg8OBtLtifW/Nnz9fgwcP1jPPPKMnn3xSjz/+uEwmk6ZNm6Zf/epXUQczNTcHJ8YPPb5EIUz2gM8++yw8iXvoF3MojFRXV0c9J7Q9WmhJ9HxQ0RQVFclms6m4uLjNfHmJGFUcelwOR/Tlwmpqatps64mVaHr6bxn6Iow1yrul0N/kl7/8pRYuXNij5ULqnHvuubLZbPrlL3+pn/3sZ+GZF7orJydHS5cu1Ycffqiqqipt2bIlaktBpNDr7vnnn9exxx7b6eP/53/+J2q/0Fh6+r0W6r4T6zMl1nb0PpmZmVq0aJE++eQTffbZZ1q3bp3OPPPMlJZp0KBBMgyjVctTSwMHDlRdXZ08Hk+bQOn1erV///5W3+ehQUKxVgqMVpEU0t576fzzz9f555+vAwcOaN26dfrrX/+q1atXa8GCBfrzn//cpvtaKOAm+scWfSYTzOfzhTuu//CHPwxvD9WmRTblhoSmyOhotGZP2blzp8aOHdsmSDqdTm3cuLHb1w81t5eVlcnn87XZH+vvkm6OOeYYSQqPlmtPqOkiWj9Z9C1nnnmm7rvvPknS9ddf32EtX2eZTKaoNQyhL65o77XQ666zPxJ76+t0wIABKiws1DfffBOe3qulsrKyFJQK3dHV/rU97YgjjtDevXvDteAtTZw4UX6/P+qsAZ9++ql8Pl+rbmahioZoU281NDR0ue9zpEGDBul73/ue7rzzTp1//vnav39/1PfA9u3bZTabdcQRR3Tr/iIRJhNo7969+vnPf65PP/1Uhx12mK655prwvilTpmj06NEqKytr80Xy1ltv6bPPPtOYMWPa9JdMloKCAu3YsaPVhLFer1fLli1LSN+KQw45RCeddJJ2796tp556qtW+kpKSPhMmjzzySB133HHauHFj1NHv+/fvDzebHH300TruuOP0zjvvxOyovWHDBjU1NfVomZEcM2bM0B//+EdZLBb94he/iDrfazSPPPKIvvjii6j7/vrXv2r79u0aNGhQqy+H0EjvPXv2tDlnzpw5GjBggFauXBn1h6LH42n1BXnGGWeooKBAzz77rD744IM2x/v9/lbzRSbT+eefL7/fr3vvvbfVj7c9e/bo8ccfT0mZENtzzz0Xc07IdevWae3atbJarZ2qMU+GadOmye/3Rx0YdMEFF0gKzl7S8jO6qakpPCVQ6Bgp+ONnzJgxWrduXav3s8/n07Jly8LNz13xySeftKm0CAQC4Vr5yNW13G63ysvLNXHixJjdSeKV0GbuaKN9e4NEl6vl9CA+n0/19fXasmWL1q9fL4/Ho6OPPlp33313q06wJpNJv//97zV//nzdcMMNeuONNzR69Ght375db7/9trKzs7V8+fKkNGlHc+WVV2rp0qWaPXt2eKmlNWvWqKamJurgmHj813/9l370ox/pzjvv1EcffaQJEyZo586dKikp0emnnx5zUuJ0c9ddd+nKK6/UXXfdpT//+c+aOnWqAoGAduzYoY8++kh//vOfw90f7rnnHl155ZW67bbb9Mwzz+ioo45Sdna2vv76a/3rX//Szp079eGHH3a45F5v0ORuWwvWG/Smcp188sl64IEH9NOf/lS/+tWv5Ha7W33hRPPqq6/qv//7vzVu3Dgdc8wxys/PV0NDgzZt2qR169bJYrHoN7/5Tav+kieccIL+8pe/6Prrr9cpp5wiu92uCRMmaMaMGRoyZIjuvfdeLV68WBdeeKGmT5+ucePGyefzac+ePfrss880ePDg8I9em82moqIiLViwQFdffbWOP/54jR8/XoZh6KuvvgoP2Gs5q0GyXH311SopKdHrr7+uyspKTZ8+XfX19Xrrrbf0b//2byopKUnZZ2pcPL20f3SCyvX+++/r//2//6fCwkJNnjxZhxxyiJqbm/XFF1/o448/lt/v169+9ateswb297//ff3pT3/SRx991GpKKinYfeXtt9/Wn//8Z51zzjnh5RpLSkpUVVWlWbNmtWqdlKQFCxbotttu0yWXXKKzzjpLdrs9PFPKhAkToi4Q0p7rrrtOWVlZOvbYY1VQUBCuKf3Xv/6lo48+us2A39B99UQXgoSEyaysLFmt1qj93noLq9WasCr00PQgUnAW/uzsbBUUFOj888/XD37wA5188slRJ1E95phj9OKLL+oPf/iDPvnkE73zzjsaPHiwzjnnHP3sZz/TmDFjElK+eFx66aUyDENPPPGEXnrpJWVnZ+uEE07QH//4R91///0JuY9Ro0Zp1apVuueee/T3v/9da9eu1fjx43X//fdr3759fSZMFhYWqri4WA8//LBKSkr01FNPyW63q6CgQFdddVWrvioFBQV6+eWX9fjjj+tvf/ubXnnlFQUCAeXn52vChAn62c9+FnXKpt5kcLZNdqtZX1Q3prooMdmtZg3O7nhwSjKccMIJevjhh7Vw4ULddtttcrvdUUevhixbtkzvvvuuPvnkE3388ceqqamRyWTSoYceqjlz5ujKK69s0z3m4osvVlVVld588009/PDD8nq9mj17driv5qmnnqqXX35ZjzzySPi9aLfbNXz4cH3/+99v0zdy0qRJevXVV/Xoo4/qvffe0wsvvCCLxaJhw4Zp6tSpKevflpGRoSeeeEL33Xef3nrrLT322GMaMWKErrnm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",
      "text/plain": [
       "<Figure size 712.125x540 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "data = duckdb.sql(\"select alt_id, steering, constraint_value from metrics_unpivot\").pl()\n",
    "data = pl.concat((data, ground_truth), how='vertical')\n",
    "data\n",
    "bins = np.histogram(np.zeros(2), bins=128, range=(ground_truth_range['min'].item(), ground_truth_range['max'].item()))[1]\n",
    "show_no_steering = False\n",
    "if show_no_steering:\n",
    "    order = ['NoSteering (EULER_MARUYAMA)', 'NoSteering (HEUN)']\n",
    "else:\n",
    "    order = ['GroundTruth', 'FeynmannKac', 'SourceParallelTempering']\n",
    "name_map = {\n",
    "    'GroundTruth': 'Dormand-Prince',\n",
    "    'NoSteering (EULER_MARUYAMA)': 'No steering (Euler-Maruyama)',\n",
    "    'NoSteering (HEUN)': 'No steering (Heun)',\n",
    "    'FeynmannKac': 'FK Steering',\n",
    "    'SourceParallelTempering': 'SPT (ours)',\n",
    "}\n",
    "common_kws = dict(\n",
    "    stat='density',\n",
    "    x='constraint_value',\n",
    "    bins=bins,\n",
    "    element='step',\n",
    ")\n",
    "include_legend = True\n",
    "plot = (\n",
    "    sns.displot(\n",
    "        data=data.filter(pl.col('steering') != 'GroundTruth'),\n",
    "        row='steering',\n",
    "        row_order=order,\n",
    "        hue='steering',\n",
    "        hue_order=order,\n",
    "        palette={\n",
    "            'GroundTruth': 'tab:grey',\n",
    "            'NoSteering (EULER_MARUYAMA)': 'tab:red',\n",
    "            'NoSteering (HEUN)': 'tab:purple',\n",
    "            'FeynmannKac': 'tab:blue',\n",
    "            'SourceParallelTempering': 'tab:orange',\n",
    "        },\n",
    "        alpha=.75,\n",
    "        common_norm=False,\n",
    "        **common_kws,\n",
    "        height=1.8,\n",
    "        # height=3,\n",
    "        aspect=2.2,\n",
    "        legend=include_legend,\n",
    "        facet_kws=dict(\n",
    "            sharex=False,\n",
    "        ),\n",
    "    )\n",
    "    .set_titles('')\n",
    "    .set(\n",
    "        xlabel='', ylabel='',\n",
    "        yscale='log' if dataset_name == 'FitzHughNagumo' else 'linear'\n",
    "    )\n",
    ")\n",
    "if not show_no_steering:\n",
    "    plot.map(\n",
    "        sns.histplot,\n",
    "        data=data.filter((pl.col('steering') == 'GroundTruth') & (pl.col('constraint_value') > 0)),\n",
    "        color='tab:grey',\n",
    "        **common_kws,\n",
    "        zorder=-1,\n",
    "    ).set_xlabels('').set_ylabels('')\n",
    "if include_legend:\n",
    "    sns.move_legend(\n",
    "        plot,\n",
    "        loc='upper center',\n",
    "        labels=[name_map[n] for n in order],\n",
    "        ncol=4,\n",
    "        title='',\n",
    "        bbox_to_anchor=(.400, 1.18),\n",
    "        frameon=True,\n",
    "        fancybox=True,\n",
    "    )\n",
    "for (row, col, hue), _ in plot.facet_data():\n",
    "    ax = plot.axes[row][col]\n",
    "    if show_no_steering or row == 0:\n",
    "        sns.histplot(\n",
    "            data=data.filter((pl.col('steering') == 'GroundTruth')),\n",
    "            color='tab:grey',\n",
    "            **common_kws,\n",
    "            ax=ax,\n",
    "            zorder=-1,\n",
    "        )\n",
    "        ax.set_xlabel('')\n",
    "        ax.set_ylabel('')\n",
    "    ax.axvline(x=0, c='r', ls=':')\n",
    "    ax.xaxis.set_tick_params(labelbottom=True)\n",
    "    ax.yaxis.set_tick_params(labelleft=True)\n",
    "    if row <= len(plot.row_names) - 2:\n",
    "        ax.set_xticklabels([])\n",
    "        # ax.xaxis.set_visible(False)\n",
    "plot.tight_layout()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "d29141cf-0727-4dca-8e52-dbbd3e1112c7",
   "metadata": {},
   "outputs": [],
   "source": [
    "if include_legend:\n",
    "    userfm.plots.save_legend(plot, 'Histogram')\n",
    "plot.savefig('Histogram.pdf', format='pdf', bbox_inches='tight', pad_inches=.06)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "id": "ac961472-204c-4285-b569-3684a93cda66",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Total Variation (all samples)\n",
      "GroundTruth\n",
      "0.9653\n",
      "FeynmannKac\n",
      "0.9733\n",
      "SourceParallelTempering\n",
      "0.6394\n"
     ]
    }
   ],
   "source": [
    "print('Total Variation (all samples)')\n",
    "ground_truth_hist = np.histogram(data.filter((pl.col('steering') == 'GroundTruth') & (pl.col('constraint_value') > 0))['constraint_value'], bins=bins, density=True)[0]\n",
    "for steering in data['steering'].unique():\n",
    "    print(steering)\n",
    "    steering_hist = np.histogram(data.filter(steering=steering)['constraint_value'], bins=bins, density=True)[0]\n",
    "    total_variation = 0.5 * np.abs(steering_hist - ground_truth_hist) * np.diff(bins)\n",
    "    print(f'{total_variation.sum():.4f}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0e5dd2cd-efdb-49e8-b2f0-5f331ce35f6f",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.13.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
