{
 "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": 128,
   "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": 117,
   "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": 129,
   "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": 130,
   "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": 131,
   "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": 132,
   "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": 133,
   "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": 134,
   "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/xxx/out…  │ 95bkwfod │\n",
      "│ 0y3y6670 │   312 │   20.0 │ FeynmannKac          │ … │    63 │ /home/xxx/out…  │ 0y3y6670 │\n",
      "│ nh82zcqn │   311 │   10.0 │ FeynmannKac          │ … │    63 │ /home/xxx/out…  │ nh82zcqn │\n",
      "│ dyhcek5w │   310 │    5.0 │ FeynmannKac          │ … │    63 │ /home/xxx/out…  │ dyhcek5w │\n",
      "│ 5no87fr1 │   302 │   40.0 │ SourceParallelTemp…  │ … │    63 │ /home/xxx/out…  │ 5no87fr1 │\n",
      "│ ufou3hgo │   301 │   20.0 │ SourceParallelTemp…  │ … │    63 │ /home/xxx/out…  │ ufou3hgo │\n",
      "│ pq8dhnab │   300 │   10.0 │ SourceParallelTemp…  │ … │    63 │ /home/xxx/out…  │ pq8dhnab │\n",
      "│ qpqsdxqq │   299 │    5.0 │ SourceParallelTemp…  │ … │    63 │ /home/xxx/out…  │ qpqsdxqq │\n",
      "│ 95bkwfod │   313 │   40.0 │ FeynmannKac          │ … │    62 │ /home/xxx/out…  │ 95bkwfod │\n",
      "│ 0y3y6670 │   312 │   20.0 │ FeynmannKac          │ … │    62 │ /home/xxx/out…  │ 0y3y6670 │\n",
      "│    ·     │    ·  │     ·  │      ·               │ · │     · │          ·           │    ·     │\n",
      "│    ·     │    ·  │     ·  │      ·               │ · │     · │          ·           │    ·     │\n",
      "│    ·     │    ·  │     ·  │      ·               │ · │     · │          ·           │    ·     │\n",
      "│ pq8dhnab │   300 │   10.0 │ SourceParallelTemp…  │ … │     1 │ /home/xxx/out…  │ pq8dhnab │\n",
      "│ qpqsdxqq │   299 │    5.0 │ SourceParallelTemp…  │ … │     1 │ /home/xxx/out…  │ qpqsdxqq │\n",
      "│ 95bkwfod │   313 │   40.0 │ FeynmannKac          │ … │     0 │ /home/xxx/out…  │ 95bkwfod │\n",
      "│ 0y3y6670 │   312 │   20.0 │ FeynmannKac          │ … │     0 │ /home/xxx/out…  │ 0y3y6670 │\n",
      "│ nh82zcqn │   311 │   10.0 │ FeynmannKac          │ … │     0 │ /home/xxx/out…  │ nh82zcqn │\n",
      "│ dyhcek5w │   310 │    5.0 │ FeynmannKac          │ … │     0 │ /home/xxx/out…  │ dyhcek5w │\n",
      "│ 5no87fr1 │   302 │   40.0 │ SourceParallelTemp…  │ … │     0 │ /home/xxx/out…  │ 5no87fr1 │\n",
      "│ ufou3hgo │   301 │   20.0 │ SourceParallelTemp…  │ … │     0 │ /home/xxx/out…  │ ufou3hgo │\n",
      "│ pq8dhnab │   300 │   10.0 │ SourceParallelTemp…  │ … │     0 │ /home/xxx/out…  │ pq8dhnab │\n",
      "│ qpqsdxqq │   299 │    5.0 │ SourceParallelTemp…  │ … │     0 │ /home/xxx/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": 135,
   "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",
      "│ 0y3y6670 │   20.0 │ FeynmannKac             │                   21668.0 │     0.677125 │\n",
      "│ ufou3hgo │   20.0 │ SourceParallelTempering │                   31746.0 │    0.9920625 │\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 520x400 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=4,\n",
    "        aspect=1.3,\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=(.0, 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": 137,
   "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)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "06ac9fba-bb82-41fd-9aad-f06b71473a7a",
   "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
}
