{
 "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": 99,
   "id": "b11186f2-ec68-4943-b13a-c4321b74857d",
   "metadata": {},
   "outputs": [],
   "source": [
    "from pathlib import Path\n",
    "\n",
    "import seaborn as sns\n",
    "sns.set_theme(style='whitegrid', font_scale=2, palette=sns.color_palette(),)\n",
    "import matplotlib.pyplot as plt\n",
    "import duckdb\n",
    "import polars as pl\n",
    "\n",
    "import userfm.plots"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "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": 9,
   "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": 175,
   "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": 5,
   "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": "931d2d05-e30d-4d74-aa20-35aa602fa019",
   "metadata": {},
   "source": [
    "##### Source parallel tempering"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 50,
   "id": "342eed56-51ab-42d6-8f69-64ffacf31166",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "┌─────────────┬─────────────────────┬────────────────┬───────┬───┬──────────────────────┬───────┬───────┐\n",
      "│ batch_count │ sampling_step_count │ sa_inheritance │  id   │ … │    sa_inheritance    │  id   │  id   │\n",
      "│    int64    │        int64        │    varchar     │ int64 │   │       varchar        │ int64 │ int64 │\n",
      "├─────────────┼─────────────────────┼────────────────┼───────┼───┼──────────────────────┼───────┼───────┤\n",
      "│          64 │                 100 │ FeynmannKac    │   310 │ … │ FeynmannKacInterme…  │     1 │     1 │\n",
      "│          64 │                 100 │ FeynmannKac    │   311 │ … │ FeynmannKacInterme…  │     1 │     1 │\n",
      "│          64 │                 100 │ FeynmannKac    │   312 │ … │ FeynmannKacInterme…  │     1 │     1 │\n",
      "│          64 │                 100 │ FeynmannKac    │   313 │ … │ FeynmannKacInterme…  │     1 │     1 │\n",
      "├─────────────┴─────────────────────┴────────────────┴───────┴───┴──────────────────────┴───────┴───────┤\n",
      "│ 4 rows                                                                           14 columns (7 shown) │\n",
      "└───────────────────────────────────────────────────────────────────────────────────────────────────────┘\n",
      "\n"
     ]
    }
   ],
   "source": [
    "spt_steering_rows = duckdb.sql(f\"\"\"\n",
    "select\n",
    "    *\n",
    "from Steering\n",
    "join SourceParallelTempering on Steering.id = SourceParallelTempering.id\n",
    "where (tilt, update_count, chain_count) in (\n",
    "    select\n",
    "        SweepTilt.tilt, update_count, chain_count\n",
    "    from SweepTrainedModel\n",
    "    natural join BestSPTParameter\n",
    "    cross join SweepTilt\n",
    "    where penalization_power = 0\n",
    "    and dynamical_system = {dataset_name!r}\n",
    "    order by dynamical_system desc, SweepTilt.tilt\n",
    ")\n",
    "and symmetric_binary_reward;\n",
    "\"\"\")\n",
    "steering_rows.show(max_width=120)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7995f6f7-301a-4248-b0fa-7068c5587c89",
   "metadata": {},
   "source": [
    "##### Feynmann-Kac"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 51,
   "id": "62b13086-0a24-45d6-9f57-3e90a056e15c",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "┌─────────────┬─────────────────────┬────────────────┬───────┬───┬──────────────────────┬───────┬───────┐\n",
      "│ batch_count │ sampling_step_count │ sa_inheritance │  id   │ … │    sa_inheritance    │  id   │  id   │\n",
      "│    int64    │        int64        │    varchar     │ int64 │   │       varchar        │ int64 │ int64 │\n",
      "├─────────────┼─────────────────────┼────────────────┼───────┼───┼──────────────────────┼───────┼───────┤\n",
      "│          64 │                 100 │ FeynmannKac    │   310 │ … │ FeynmannKacInterme…  │     1 │     1 │\n",
      "│          64 │                 100 │ FeynmannKac    │   311 │ … │ FeynmannKacInterme…  │     1 │     1 │\n",
      "│          64 │                 100 │ FeynmannKac    │   312 │ … │ FeynmannKacInterme…  │     1 │     1 │\n",
      "│          64 │                 100 │ FeynmannKac    │   313 │ … │ FeynmannKacInterme…  │     1 │     1 │\n",
      "├─────────────┴─────────────────────┴────────────────┴───────┴───┴──────────────────────┴───────┴───────┤\n",
      "│ 4 rows                                                                           14 columns (7 shown) │\n",
      "└───────────────────────────────────────────────────────────────────────────────────────────────────────┘\n",
      "\n"
     ]
    }
   ],
   "source": [
    "fk_steering_rows = duckdb.sql(f\"\"\"\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",
    "where (tilt, potential, ensemble_size) in (\n",
    "    select\n",
    "        SweepTilt.tilt, potential, ensemble_size,\n",
    "    from SweepTrainedModel\n",
    "    natural join BestFKParameter\n",
    "    cross join SweepTilt\n",
    "    where penalization_power = 0\n",
    "    and dynamical_system = {dataset_name!r}\n",
    "    order by dynamical_system desc, SweepTilt.tilt\n",
    ")\n",
    "and symmetric_binary_reward;\n",
    "\"\"\")\n",
    "steering_rows.show(max_width=120)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8c294185-7ec0-4469-a294-52de021b4714",
   "metadata": {},
   "source": [
    "### Model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 52,
   "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": 71,
   "id": "a000e32b-23d6-411b-a900-47ea718f0df1",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "┌──────────┬───────┬────────┬─────────────────────────┬─────────────┬────────────┐\n",
      "│  alt_id  │  id   │  tilt  │        steering         │ batch_count │ batch_size │\n",
      "│ varchar  │ int64 │ double │         varchar         │    int64    │   int64    │\n",
      "├──────────┼───────┼────────┼─────────────────────────┼─────────────┼────────────┤\n",
      "│ 95bkwfod │   313 │   40.0 │ FeynmannKac             │          64 │        500 │\n",
      "│ 0y3y6670 │   312 │   20.0 │ FeynmannKac             │          64 │        500 │\n",
      "│ nh82zcqn │   311 │   10.0 │ FeynmannKac             │          64 │        500 │\n",
      "│ dyhcek5w │   310 │    5.0 │ FeynmannKac             │          64 │        500 │\n",
      "│ 5no87fr1 │   302 │   40.0 │ SourceParallelTempering │          64 │        500 │\n",
      "│ ufou3hgo │   301 │   20.0 │ SourceParallelTempering │          64 │        500 │\n",
      "│ pq8dhnab │   300 │   10.0 │ SourceParallelTempering │          64 │        500 │\n",
      "│ qpqsdxqq │   299 │    5.0 │ SourceParallelTempering │          64 │        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, tilt, sa_inheritance as steering, batch_count from spt_steering_rows\n",
    "    union\n",
    "    select id, tilt, sa_inheritance as steering, batch_count 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 not TrainingAndEvaluation.log_constraint_values\n",
    "\"\"\")\n",
    "rows.show(max_width=120)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 89,
   "id": "883437f7-f316-4ac1-83bf-6e7ed04e769e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "┌──────────┬───────┬────────┬──────────────────────┬───┬───────┬──────────────────────┬──────────┐\n",
      "│  alt_id  │  id   │  tilt  │       steering       │ … │ step  │      file_path       │  alt_id  │\n",
      "│ varchar  │ int64 │ double │       varchar        │   │ int64 │       varchar        │ varchar  │\n",
      "├──────────┼───────┼────────┼──────────────────────┼───┼───────┼──────────────────────┼──────────┤\n",
      "│ 95bkwfod │   313 │   40.0 │ FeynmannKac          │ … │    63 │ /home/ttransue/out…  │ 95bkwfod │\n",
      "│ 0y3y6670 │   312 │   20.0 │ FeynmannKac          │ … │    63 │ /home/ttransue/out…  │ 0y3y6670 │\n",
      "│ nh82zcqn │   311 │   10.0 │ FeynmannKac          │ … │    63 │ /home/ttransue/out…  │ nh82zcqn │\n",
      "│ dyhcek5w │   310 │    5.0 │ FeynmannKac          │ … │    63 │ /home/ttransue/out…  │ dyhcek5w │\n",
      "│ 5no87fr1 │   302 │   40.0 │ SourceParallelTemp…  │ … │    63 │ /home/ttransue/out…  │ 5no87fr1 │\n",
      "│ ufou3hgo │   301 │   20.0 │ SourceParallelTemp…  │ … │    63 │ /home/ttransue/out…  │ ufou3hgo │\n",
      "│ pq8dhnab │   300 │   10.0 │ SourceParallelTemp…  │ … │    63 │ /home/ttransue/out…  │ pq8dhnab │\n",
      "│ qpqsdxqq │   299 │    5.0 │ SourceParallelTemp…  │ … │    63 │ /home/ttransue/out…  │ qpqsdxqq │\n",
      "│ 95bkwfod │   313 │   40.0 │ FeynmannKac          │ … │    62 │ /home/ttransue/out…  │ 95bkwfod │\n",
      "│ 0y3y6670 │   312 │   20.0 │ FeynmannKac          │ … │    62 │ /home/ttransue/out…  │ 0y3y6670 │\n",
      "│    ·     │    ·  │     ·  │      ·               │ · │     · │          ·           │    ·     │\n",
      "│    ·     │    ·  │     ·  │      ·               │ · │     · │          ·           │    ·     │\n",
      "│    ·     │    ·  │     ·  │      ·               │ · │     · │          ·           │    ·     │\n",
      "│ pq8dhnab │   300 │   10.0 │ SourceParallelTemp…  │ … │     1 │ /home/ttransue/out…  │ pq8dhnab │\n",
      "│ qpqsdxqq │   299 │    5.0 │ SourceParallelTemp…  │ … │     1 │ /home/ttransue/out…  │ qpqsdxqq │\n",
      "│ 95bkwfod │   313 │   40.0 │ FeynmannKac          │ … │     0 │ /home/ttransue/out…  │ 95bkwfod │\n",
      "│ 0y3y6670 │   312 │   20.0 │ FeynmannKac          │ … │     0 │ /home/ttransue/out…  │ 0y3y6670 │\n",
      "│ nh82zcqn │   311 │   10.0 │ FeynmannKac          │ … │     0 │ /home/ttransue/out…  │ nh82zcqn │\n",
      "│ dyhcek5w │   310 │    5.0 │ FeynmannKac          │ … │     0 │ /home/ttransue/out…  │ dyhcek5w │\n",
      "│ 5no87fr1 │   302 │   40.0 │ SourceParallelTemp…  │ … │     0 │ /home/ttransue/out…  │ 5no87fr1 │\n",
      "│ ufou3hgo │   301 │   20.0 │ SourceParallelTemp…  │ … │     0 │ /home/ttransue/out…  │ ufou3hgo │\n",
      "│ pq8dhnab │   300 │   10.0 │ SourceParallelTemp…  │ … │     0 │ /home/ttransue/out…  │ pq8dhnab │\n",
      "│ qpqsdxqq │   299 │    5.0 │ SourceParallelTemp…  │ … │     0 │ /home/ttransue/out…  │ qpqsdxqq │\n",
      "├──────────┴───────┴────────┴──────────────────────┴───┴───────┴──────────────────────┴──────────┤\n",
      "│ 512 rows (20 shown)                                                       10 columns (7 shown) │\n",
      "└────────────────────────────────────────────────────────────────────────────────────────────────┘\n",
      "\n"
     ]
    }
   ],
   "source": [
    "metrics = (\n",
    "    pl.scan_csv([\n",
    "        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": 105,
   "id": "341c09c1-b5a9-4aa4-a27d-c8c3f0584849",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "┌──────────┬────────┬─────────────────────────┬───────────────────────────┬──────────────┐\n",
      "│  alt_id  │  tilt  │        steering         │ satisfactory_sample_total │ success_rate │\n",
      "│ varchar  │ double │         varchar         │          double           │    double    │\n",
      "├──────────┼────────┼─────────────────────────┼───────────────────────────┼──────────────┤\n",
      "│ qpqsdxqq │    5.0 │ SourceParallelTempering │                   30812.0 │     0.962875 │\n",
      "│ dyhcek5w │    5.0 │ FeynmannKac             │                   21665.0 │   0.67703125 │\n",
      "│ nh82zcqn │   10.0 │ FeynmannKac             │                   21668.0 │     0.677125 │\n",
      "│ pq8dhnab │   10.0 │ SourceParallelTempering │                   31705.0 │   0.99078125 │\n",
      "│ ufou3hgo │   20.0 │ SourceParallelTempering │                   31746.0 │    0.9920625 │\n",
      "│ 0y3y6670 │   20.0 │ FeynmannKac             │                   21668.0 │     0.677125 │\n",
      "│ 95bkwfod │   40.0 │ FeynmannKac             │                   21668.0 │     0.677125 │\n",
      "│ 5no87fr1 │   40.0 │ SourceParallelTempering │                   31746.0 │    0.9920625 │\n",
      "└──────────┴────────┴─────────────────────────┴───────────────────────────┴──────────────┘\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 660x300 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "data = duckdb.sql(\"\"\"\n",
    "select\n",
    "    alt_id,\n",
    "    tilt,\n",
    "    steering,\n",
    "    sum(satisfactory_sample_count) as satisfactory_sample_total,\n",
    "    satisfactory_sample_total / (batch_count * batch_size) as success_rate,\n",
    "from metrics\n",
    "group by alt_id, tilt, steering, batch_count, batch_size,\n",
    "order by tilt\n",
    "\"\"\")\n",
    "data.show()\n",
    "data = data.pl()\n",
    "order = ['FeynmannKac', 'SourceParallelTempering']\n",
    "include_legend = False\n",
    "plot = (\n",
    "    sns.relplot(\n",
    "        kind='line',\n",
    "        markers=True,\n",
    "        markersize=16,\n",
    "        linewidth=4,\n",
    "        height=3,\n",
    "        aspect=2.2,\n",
    "        legend=include_legend,\n",
    "        data=data,\n",
    "        x='tilt',\n",
    "        y='success_rate',\n",
    "        hue='steering',\n",
    "        hue_order=order,\n",
    "        style='steering',\n",
    "        style_order=order,\n",
    "    )\n",
    "    .set(\n",
    "        xscale='log',\n",
    "        xticks=data['tilt'].unique(),\n",
    "        xticklabels=data['tilt'].unique(),\n",
    "        xlabel='Tilt',\n",
    "        ylabel='Success rate',\n",
    "        ylim=(.6, 1.1),\n",
    "        # yticks=[0, .5, 1],\n",
    "    )\n",
    ")\n",
    "if include_legend:\n",
    "    sns.move_legend(\n",
    "        plot,\n",
    "        loc='upper center',\n",
    "        labels=['FK', 'SPT'],\n",
    "        ncol=4,\n",
    "        title='',\n",
    "        bbox_to_anchor=(.400, 1.18),\n",
    "        frameon=True,\n",
    "        fancybox=True,\n",
    "    )\n",
    "ax = plot.axes[0, 0]\n",
    "ax.minorticks_off()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 106,
   "id": "d29141cf-0727-4dca-8e52-dbbd3e1112c7",
   "metadata": {},
   "outputs": [],
   "source": [
    "if include_legend:\n",
    "    userfm.plots.save_legend(plot, 'AblateTilt')\n",
    "plot.savefig('AblateTilt.pdf', format='pdf', bbox_inches='tight', pad_inches=.06)"
   ]
  }
 ],
 "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
}
