{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "bfac73b4",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Set HuggingFace cache to /tmp to avoid NFS locking issues\n",
    "import os\n",
    "os.environ['HF_DATASETS_CACHE'] = '/tmp/hf_datasets_cache'\n",
    "os.environ['HF_HOME'] = '/tmp/hf_home'\n",
    "\n",
    "from datasets import load_dataset\n",
    "from transformers import AutoModel\n",
    "import torch\n",
    "from collections import defaultdict\n",
    "from fim.trainers.utils import get_accel_type\n",
    "from fim.models.mjp import FIMMJP\n",
    "import pandas as pd\n",
    "\n",
    "device = get_accel_type()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f51abdcc",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Loading the Discrete Flashing Ratchet (DFR) dataset from Huggingface\n",
    "data = load_dataset(\"FIM4Science/mjp\", download_mode=\"force_redownload\", trust_remote_code=True, name=\"default\")\n",
    "data.set_format(\"torch\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "77ae260b",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Loading the FIMMJP model from Huggingface\n",
    "fimmjp = FIMMJP.from_pretrained(\"FIM4Science/fim-mjp\", trust_remote_code=True)\n",
    "\n",
    "fimmjp = fimmjp.to(device)\n",
    "fimmjp.eval()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0bae13ec",
   "metadata": {},
   "outputs": [],
   "source": [
    "# copy data to device\n",
    "batch = {k: v.to(device) for k, v in data[\"train\"][:1].items()}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "252fb656",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Prepare a batch\n",
    "n_paths_eval = [1, 30, 100, 300, 500, 1000, 5000]\n",
    "total_n_paths = batch[\"observation_grid\"].shape[2]\n",
    "statistics = total_n_paths // 300 "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "eaf32478",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "os.environ[\"CUDA_LAUNCH_BLOCKING\"] = \"1\"\n",
    "os.environ[\"TORCH_SHOW_CPP_STACKTRACES\"] = \"1\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "1996e8fe",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<style type=\"text/css\">\n",
       "</style>\n",
       "<table id=\"T_36b51\">\n",
       "  <caption>Per-graph-class Hellinger (aggregated over Train/Val-kept datasets only; THRESHOLD=0.15)</caption>\n",
       "  <thead>\n",
       "    <tr>\n",
       "      <th class=\"blank level0\" >&nbsp;</th>\n",
       "      <th id=\"T_36b51_level0_col0\" class=\"col_heading level0 col0\" >graph_class</th>\n",
       "      <th id=\"T_36b51_level0_col1\" class=\"col_heading level0 col1\" >N_files</th>\n",
       "      <th id=\"T_36b51_level0_col2\" class=\"col_heading level0 col2\" >TrainVal_mean</th>\n",
       "      <th id=\"T_36b51_level0_col3\" class=\"col_heading level0 col3\" >TrainVal_std</th>\n",
       "      <th id=\"T_36b51_level0_col4\" class=\"col_heading level0 col4\" >TrainVal_N</th>\n",
       "      <th id=\"T_36b51_level0_col5\" class=\"col_heading level0 col5\" >OpenFIM_mean</th>\n",
       "      <th id=\"T_36b51_level0_col6\" class=\"col_heading level0 col6\" >OpenFIM_std</th>\n",
       "      <th id=\"T_36b51_level0_col7\" class=\"col_heading level0 col7\" >OpenFIM_N</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th id=\"T_36b51_level0_row0\" class=\"row_heading level0 row0\" >0</th>\n",
       "      <td id=\"T_36b51_row0_col0\" class=\"data row0 col0\" >apollonian</td>\n",
       "      <td id=\"T_36b51_row0_col1\" class=\"data row0 col1\" >12</td>\n",
       "      <td id=\"T_36b51_row0_col2\" class=\"data row0 col2\" >0.067145</td>\n",
       "      <td id=\"T_36b51_row0_col3\" class=\"data row0 col3\" >0.020099</td>\n",
       "      <td id=\"T_36b51_row0_col4\" class=\"data row0 col4\" >12</td>\n",
       "      <td id=\"T_36b51_row0_col5\" class=\"data row0 col5\" >0.138392</td>\n",
       "      <td id=\"T_36b51_row0_col6\" class=\"data row0 col6\" >0.031277</td>\n",
       "      <td id=\"T_36b51_row0_col7\" class=\"data row0 col7\" >12</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th id=\"T_36b51_level0_row1\" class=\"row_heading level0 row1\" >1</th>\n",
       "      <td id=\"T_36b51_row1_col0\" class=\"data row1 col0\" >complete</td>\n",
       "      <td id=\"T_36b51_row1_col1\" class=\"data row1 col1\" >11</td>\n",
       "      <td id=\"T_36b51_row1_col2\" class=\"data row1 col2\" >0.076529</td>\n",
       "      <td id=\"T_36b51_row1_col3\" class=\"data row1 col3\" >0.020482</td>\n",
       "      <td id=\"T_36b51_row1_col4\" class=\"data row1 col4\" >11</td>\n",
       "      <td id=\"T_36b51_row1_col5\" class=\"data row1 col5\" >0.128438</td>\n",
       "      <td id=\"T_36b51_row1_col6\" class=\"data row1 col6\" >0.015218</td>\n",
       "      <td id=\"T_36b51_row1_col7\" class=\"data row1 col7\" >11</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th id=\"T_36b51_level0_row2\" class=\"row_heading level0 row2\" >2</th>\n",
       "      <td id=\"T_36b51_row2_col0\" class=\"data row2 col0\" >d-regular</td>\n",
       "      <td id=\"T_36b51_row2_col1\" class=\"data row2 col1\" >8</td>\n",
       "      <td id=\"T_36b51_row2_col2\" class=\"data row2 col2\" >0.062852</td>\n",
       "      <td id=\"T_36b51_row2_col3\" class=\"data row2 col3\" >0.017691</td>\n",
       "      <td id=\"T_36b51_row2_col4\" class=\"data row2 col4\" >8</td>\n",
       "      <td id=\"T_36b51_row2_col5\" class=\"data row2 col5\" >0.160092</td>\n",
       "      <td id=\"T_36b51_row2_col6\" class=\"data row2 col6\" >0.053489</td>\n",
       "      <td id=\"T_36b51_row2_col7\" class=\"data row2 col7\" >8</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th id=\"T_36b51_level0_row3\" class=\"row_heading level0 row3\" >3</th>\n",
       "      <td id=\"T_36b51_row3_col0\" class=\"data row3 col0\" >delaunay</td>\n",
       "      <td id=\"T_36b51_row3_col1\" class=\"data row3 col1\" >10</td>\n",
       "      <td id=\"T_36b51_row3_col2\" class=\"data row3 col2\" >0.075483</td>\n",
       "      <td id=\"T_36b51_row3_col3\" class=\"data row3 col3\" >0.023750</td>\n",
       "      <td id=\"T_36b51_row3_col4\" class=\"data row3 col4\" >10</td>\n",
       "      <td id=\"T_36b51_row3_col5\" class=\"data row3 col5\" >0.146120</td>\n",
       "      <td id=\"T_36b51_row3_col6\" class=\"data row3 col6\" >0.038363</td>\n",
       "      <td id=\"T_36b51_row3_col7\" class=\"data row3 col7\" >10</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th id=\"T_36b51_level0_row4\" class=\"row_heading level0 row4\" >4</th>\n",
       "      <td id=\"T_36b51_row4_col0\" class=\"data row4 col0\" >erdos-renyi</td>\n",
       "      <td id=\"T_36b51_row4_col1\" class=\"data row4 col1\" >15</td>\n",
       "      <td id=\"T_36b51_row4_col2\" class=\"data row4 col2\" >0.043034</td>\n",
       "      <td id=\"T_36b51_row4_col3\" class=\"data row4 col3\" >0.018799</td>\n",
       "      <td id=\"T_36b51_row4_col4\" class=\"data row4 col4\" >15</td>\n",
       "      <td id=\"T_36b51_row4_col5\" class=\"data row4 col5\" >0.179191</td>\n",
       "      <td id=\"T_36b51_row4_col6\" class=\"data row4 col6\" >0.024847</td>\n",
       "      <td id=\"T_36b51_row4_col7\" class=\"data row4 col7\" >6</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th id=\"T_36b51_level0_row5\" class=\"row_heading level0 row5\" >5</th>\n",
       "      <td id=\"T_36b51_row5_col0\" class=\"data row5 col0\" >euclid-mst</td>\n",
       "      <td id=\"T_36b51_row5_col1\" class=\"data row5 col1\" >8</td>\n",
       "      <td id=\"T_36b51_row5_col2\" class=\"data row5 col2\" >0.077969</td>\n",
       "      <td id=\"T_36b51_row5_col3\" class=\"data row5 col3\" >0.026071</td>\n",
       "      <td id=\"T_36b51_row5_col4\" class=\"data row5 col4\" >8</td>\n",
       "      <td id=\"T_36b51_row5_col5\" class=\"data row5 col5\" >0.188115</td>\n",
       "      <td id=\"T_36b51_row5_col6\" class=\"data row5 col6\" >0.056790</td>\n",
       "      <td id=\"T_36b51_row5_col7\" class=\"data row5 col7\" >8</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th id=\"T_36b51_level0_row6\" class=\"row_heading level0 row6\" >6</th>\n",
       "      <td id=\"T_36b51_row6_col0\" class=\"data row6 col0\" >grid</td>\n",
       "      <td id=\"T_36b51_row6_col1\" class=\"data row6 col1\" >11</td>\n",
       "      <td id=\"T_36b51_row6_col2\" class=\"data row6 col2\" >0.073523</td>\n",
       "      <td id=\"T_36b51_row6_col3\" class=\"data row6 col3\" >0.025638</td>\n",
       "      <td id=\"T_36b51_row6_col4\" class=\"data row6 col4\" >11</td>\n",
       "      <td id=\"T_36b51_row6_col5\" class=\"data row6 col5\" >0.129476</td>\n",
       "      <td id=\"T_36b51_row6_col6\" class=\"data row6 col6\" >0.023320</td>\n",
       "      <td id=\"T_36b51_row6_col7\" class=\"data row6 col7\" >11</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th id=\"T_36b51_level0_row7\" class=\"row_heading level0 row7\" >7</th>\n",
       "      <td id=\"T_36b51_row7_col0\" class=\"data row7 col0\" >k-partite</td>\n",
       "      <td id=\"T_36b51_row7_col1\" class=\"data row7 col1\" >12</td>\n",
       "      <td id=\"T_36b51_row7_col2\" class=\"data row7 col2\" >0.065086</td>\n",
       "      <td id=\"T_36b51_row7_col3\" class=\"data row7 col3\" >0.012550</td>\n",
       "      <td id=\"T_36b51_row7_col4\" class=\"data row7 col4\" >12</td>\n",
       "      <td id=\"T_36b51_row7_col5\" class=\"data row7 col5\" >0.139336</td>\n",
       "      <td id=\"T_36b51_row7_col6\" class=\"data row7 col6\" >0.037164</td>\n",
       "      <td id=\"T_36b51_row7_col7\" class=\"data row7 col7\" >12</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th id=\"T_36b51_level0_row8\" class=\"row_heading level0 row8\" >8</th>\n",
       "      <td id=\"T_36b51_row8_col0\" class=\"data row8 col0\" >sbm</td>\n",
       "      <td id=\"T_36b51_row8_col1\" class=\"data row8 col1\" >14</td>\n",
       "      <td id=\"T_36b51_row8_col2\" class=\"data row8 col2\" >0.016917</td>\n",
       "      <td id=\"T_36b51_row8_col3\" class=\"data row8 col3\" >0.022451</td>\n",
       "      <td id=\"T_36b51_row8_col4\" class=\"data row8 col4\" >14</td>\n",
       "      <td id=\"T_36b51_row8_col5\" class=\"data row8 col5\" >0.119431</td>\n",
       "      <td id=\"T_36b51_row8_col6\" class=\"data row8 col6\" >0.140599</td>\n",
       "      <td id=\"T_36b51_row8_col7\" class=\"data row8 col7\" >12</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th id=\"T_36b51_level0_row9\" class=\"row_heading level0 row9\" >9</th>\n",
       "      <td id=\"T_36b51_row9_col0\" class=\"data row9 col0\" >torus</td>\n",
       "      <td id=\"T_36b51_row9_col1\" class=\"data row9 col1\" >12</td>\n",
       "      <td id=\"T_36b51_row9_col2\" class=\"data row9 col2\" >0.079918</td>\n",
       "      <td id=\"T_36b51_row9_col3\" class=\"data row9 col3\" >0.031215</td>\n",
       "      <td id=\"T_36b51_row9_col4\" class=\"data row9 col4\" >12</td>\n",
       "      <td id=\"T_36b51_row9_col5\" class=\"data row9 col5\" >0.130439</td>\n",
       "      <td id=\"T_36b51_row9_col6\" class=\"data row9 col6\" >0.025994</td>\n",
       "      <td id=\"T_36b51_row9_col7\" class=\"data row9 col7\" >12</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th id=\"T_36b51_level0_row10\" class=\"row_heading level0 row10\" >10</th>\n",
       "      <td id=\"T_36b51_row10_col0\" class=\"data row10 col0\" >watts-strogatz</td>\n",
       "      <td id=\"T_36b51_row10_col1\" class=\"data row10 col1\" >13</td>\n",
       "      <td id=\"T_36b51_row10_col2\" class=\"data row10 col2\" >0.093951</td>\n",
       "      <td id=\"T_36b51_row10_col3\" class=\"data row10 col3\" >0.034819</td>\n",
       "      <td id=\"T_36b51_row10_col4\" class=\"data row10 col4\" >13</td>\n",
       "      <td id=\"T_36b51_row10_col5\" class=\"data row10 col5\" >0.134997</td>\n",
       "      <td id=\"T_36b51_row10_col6\" class=\"data row10 col6\" >0.025755</td>\n",
       "      <td id=\"T_36b51_row10_col7\" class=\"data row10 col7\" >13</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n"
      ],
      "text/plain": [
       "<pandas.io.formats.style.Styler at 0x14890fd70d70>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "<style type=\"text/css\">\n",
       "</style>\n",
       "<table id=\"T_be0b5\">\n",
       "  <caption>Overall across Train/Val-kept datasets (OpenFIM used when available)</caption>\n",
       "  <thead>\n",
       "    <tr>\n",
       "      <th class=\"blank level0\" >&nbsp;</th>\n",
       "      <th id=\"T_be0b5_level0_col0\" class=\"col_heading level0 col0\" >graph_class</th>\n",
       "      <th id=\"T_be0b5_level0_col1\" class=\"col_heading level0 col1\" >N_files</th>\n",
       "      <th id=\"T_be0b5_level0_col2\" class=\"col_heading level0 col2\" >TrainVal_mean</th>\n",
       "      <th id=\"T_be0b5_level0_col3\" class=\"col_heading level0 col3\" >TrainVal_std</th>\n",
       "      <th id=\"T_be0b5_level0_col4\" class=\"col_heading level0 col4\" >TrainVal_N</th>\n",
       "      <th id=\"T_be0b5_level0_col5\" class=\"col_heading level0 col5\" >OpenFIM_mean</th>\n",
       "      <th id=\"T_be0b5_level0_col6\" class=\"col_heading level0 col6\" >OpenFIM_std</th>\n",
       "      <th id=\"T_be0b5_level0_col7\" class=\"col_heading level0 col7\" >OpenFIM_N</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th id=\"T_be0b5_level0_row0\" class=\"row_heading level0 row0\" >0</th>\n",
       "      <td id=\"T_be0b5_row0_col0\" class=\"data row0 col0\" ><ALL (TV-kept)></td>\n",
       "      <td id=\"T_be0b5_row0_col1\" class=\"data row0 col1\" >126</td>\n",
       "      <td id=\"T_be0b5_row0_col2\" class=\"data row0 col2\" >0.064932</td>\n",
       "      <td id=\"T_be0b5_row0_col3\" class=\"data row0 col3\" >0.031458</td>\n",
       "      <td id=\"T_be0b5_row0_col4\" class=\"data row0 col4\" >126</td>\n",
       "      <td id=\"T_be0b5_row0_col5\" class=\"data row0 col5\" >0.141263</td>\n",
       "      <td id=\"T_be0b5_row0_col6\" class=\"data row0 col6\" >0.056651</td>\n",
       "      <td id=\"T_be0b5_row0_col7\" class=\"data row0 col7\" >115</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n"
      ],
      "text/plain": [
       "<pandas.io.formats.style.Styler at 0x148a99e188f0>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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ZkXjf505c0azZs3SPffco8jISEVHR+vxxx93ew+o3bt3a/DgwYqMjNQdd9yhhIQEXblyxeHt2L43v/rqK73++uvq0aOHbr31Vm3ZssUo99lnnykuLk4dO3bUbbfdph49emjKlCn6+eefy23z9OnTeuGFF9SuXTu1bdtWzz33nE6ePKk777zT2FdGRoZRvrLxEh0dc82Rc5IjsdcmR9vNkb97Rc6fP6/f/e53GjBggLp06aJWrVopKipK999/v15//XXjvW1V2ViUlZ3Trp5/4sQJjR07Vu3bt1ebNm00atQoZWVl2e3jk08+UXx8vPr06aN27dqpZcuWuv322/Xss8/a/d2nTp1qbDsxMdFuGwcPHjSW3XPPPTX6W1jl5+dr7ty5+tWvfqWoqChFRUXp4YcfVlJSkt0xcvXfIjExUX379lXLli21ZMkSSVJRUZFef/113XHHHYqMjNTjjz+ub7/9tsr9nzx5UlOnTlXfvn3VqlUrRUdHa8CAAVq4cGG58/mhQ4c0duxYdevWTS1btlTbtm111113ady4ccY5tCZs32NZWVl64YUX1LFjR0VGRho9TYuKivTee+9pwIABio6OVuvWrdWvXz/NmzdPBQUFdttz5Jq4d+9eo+wDDzxQrm4PP/ywsXzXrl2SajYGbE3bsbJtVTVussVi0erVq/XUU08Z771u3bopPj5eqampDv29bTlzvFTm6vrv3LlTAwcOVGRkpPr27Wu8f3fs2KFf//rXioyMVPfu3TV//vxy21q+fLl+/etfG+/Hzp07a8CAAZo6darx/o2IiLDrWfLmm296ZIx9Z+3evVtDhgxRZGSkunbtqtmzZ+vHH38sd167+pyWnp6u4cOHq127durevbukqq9NtXVtrsvmz59v/H169OihY8eOVVn+6nPtokWLdO+996pVq1bq27ev/va3v9mVd/Wz9BdffKFHH31UkZGRGj9+fI2O7Yra/IknntCTTz5prLd9+/YKr5MWi0XLli3TkCFD1L59eyOuhIQE5eTkOPEX9j4XLlzQjBkz1LdvX6M9evTooWeeeUYLFy6scJ2ff/5Z/+///T916NBB0dHRGjlypH766Se7MrbHxrJly7RgwQL16NFDUVFRio2N1ZEjR1RaWmrMb926tQYOHKidO3dei7C9Xk2OUVfu7a5+L0mq9LN+Zc9lKSgo0BtvvKF+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794aM2aMTp06pddff135+fk6cuSIEhMTqxzr5cqVKxo/fryRBIyPj9dDDz2kkydPas6cObpw4YL++Mc/ql+/fmrZsqXWrFmjBQsWaPPmzZKkCRMm6P7775f03/G/JkyYYCRfH3vsMT311FPKzc3Vn//8Z2VkZOiDDz5Qv3791K9fP2ebpkIzZ840bvDatm2rl19+WXl5eUpISHBpu8eOHdPTTz+t/v37KycnRy1bttSmTZuMG50mTZrolVdeUcuWLbVx40YlJycrOztbEydONN4H7777rnGBbtiwoaZOnarw8HDNnTvX7oO2Ozh7Tqos9trmSLs5+nevTFhYmF555RVFRkYqLCxMgYGBunDhgt555x3t2bNHBw4c0GeffaZf//rXLseXn5+vVq1a6U9/+pOys7M1e/ZsFRcX6+uvv9aPP/6oNm3aSJIeeugh3XXXXbrpppvUoEEDlZSUaN++fZozZ46ksp4Dv/71r9W+fXt1795du3bt0sGDB5Wammokvj777DNJZR+UHUnAvvrqq0bS7q677tJvfvMbmc1mvfXWW9q/f782bNigjz76SMOHD7db79ixYxowYICefPJJFRUVqWnTpvq///s/I5lmMpk0fvx49ejRQ6tXr9ann35abt8Wi0UvvfSSkXzt3r27nn/+eZ09e1Zz5sxRbm6uvvnmG73zzjuaNGmStm3bZiRfH3vsMcXFxam4uFg///yztm3bpoYNG9Y4bltZWVn6/e9/r86dO+vIkSMKCgrSn//8ZyP52qlTJ40fP14hISF67733tG3bNu3evVtvvfWWXn311XLbq8k18dlnn9WUKVMkSStXrjQSsLZJCUd6HjjbjjWxaNEi41rk5+en4cOHq1+/frp06ZK2bNni9JcVjh4vjsjKylK/fv0UHx9v1wbz5s1TVFSU/vCHP2jLli1avHixEaO1bazvpYCAAM2ePVtt2rTR+fPndeTIEX355ZdGEnvNmjVatmyZ0bN26NChio2NlSSvTx7ZnnvbtGmjKVOmKCcnR3/605+qXC8vL0+NGjXSH//4R0VGRtr1tqluP7V5ba4rUlJSNGvWLEll4+avWrVKLVq0cGgbR44c0YQJE9SzZ0+tX7/e+FXU22+/bXxZ4cpn6ePHj6t9+/YaP368QkNDlZ+fr3Hjxjl1bCckJGj79u2aPn26JKljx4764x//KOm/z39ITk42znMtW7bUpEmT1LRpUy1btkz//Oc/dezYMU2fPl3vvvuuQ38nb3Lu3Dkj4dOhQwe9/PLLql+/vrKysrR79+4K3zc5OTlq1qyZ3n//fV2+fFlz5szR6dOnlZ2drT//+c+aN29euXVOnDihKVOmqF27dvrDH/6g06dPG19KDxs2TP3791dCQoLS09OVk5OjNWvWGMm161V1x6ir93ZXv5ecYb0GRUREaPr06WrUqJHOnDmj1NRUu18nOqp+/fq6/fbbtXPnTp09e1bHjh3TbbfdVmEy1rpckqKjo9WkSROH7imquz4WFhZWe66wnidCQ0P12muv6eabb1Z2drYOHz6szz//XCaTqdqYa/petP76eOzYsTpz5owk6b333tNNN90kqeKx9zMyMvTCCy+oT58++vnnn3XjjTc6fPz8+c9/NpKvzZo109SpU+Xv71/ltXj06NG6+eab1bhxY9WvX1+XLl3SP//5T61atUr5+flauHChXn/9dWOc7sGDBxvr2ibQb7rpJruxvKWyX2xOmjTJ+JWXM50a3IUELCrUuXNn7dixQ/PmzdPRo0d16dIloyejVPat7969e/XQQw/ZrTd69GijJ9Pdd9+tnj17Kjs7WxcvXtSWLVv06KOPavXq1Ub5SZMmKT4+XlLZt1ndu3fXlStX9NNPP2n37t3q0aOHQ/W2WCx237T98Y9/1IABAySVfTC/7777JJX9vO/UqVPlvoVt37698fPKzp07Gzc6V/fEqUhAQIB69uypDz74QD/88IPOnz+vkpISuzJ79uypcN0mTZro/fffV7169fTAAw9o+fLlKigoUHZ2tvLz852+Gc/NzdXGjRuN1/Pnz7drM+sg4a60SbNmzTRv3jz5+/vr8uXLxlMipbIbxNtuu0033XSTcdNb2d+yXr16ev/9941eDYWFhZo9e7aksl6WVSVg//3vfxs/pbQOEyDJ+NlqSkqKSktLtXLlSk2dOlU9e/ZUkyZNjPVbtWplN67fgQMHjJ5bjRs31siRI2UymXTDDTdo8ODBxreYy5cvr9UE7Pnz5+1+sjp//nwjKVVSUlLum+KjR4+W66HbsGHDCsdYHDhwoN544w27ebY/r3zyySfVtm1bSdLjjz+uzz77TKdPn9aePXuMBJttr7FJkybp2WeflVT2d7a+t9zF2XOSVHHstcnRdrM+KVWq2d+9Mk2bNlXr1q310Ucf6cCBA8rNzTV60Vvt2bOnVhKwUlkC3ppc++KLL4wPZj/99JNRzz59+mjBggVKTEzUzz//XO7nuT/++KMuXryoG264Qc8884zxs/QVK1aoU6dOysjIML5Au/POO9W8efMa1S0nJ8c41wUEBOj55583enQOGzZMU6dONfZzdeKua9euev/99+3mWctLZUnl3//+95Kke++9Vzt37iw3TMSBAwd08OBBSVJgYKDef/9948Ou2Ww2kpurVq3SpEmT7HruREREqHXr1oqIiDCSgrb27t2rwsJCu3mVjVk4ffp04/x97733ymKxaNWqVcbycePGGV9SjBgxwugluGLFigoTsDW5Jg4aNEivvfaaLl68qDVr1mj69OkKDAy0S8AOGTKk3LYr4ko71oTte+83v/mNkVCSZFw3nGG9yZRqdrwUFhZq79695bbTrl27cr266tWrp7/+9a8KCwtTeHi43Rczf/rTn3TXXXepZ8+eRgLWtm2s2woICFDr1q3VoUMH4xpr+zPRnj172vXkjIiIqBNj3Z47d87u3LtgwQLjHFVcXFzhMW1r/vz5xtAZVQ134ug53td8+eWX2rdvnywWi2699VatXLnSODefPHmy3BewwcHBuv3228ttZ+DAgcZ74/7779fevXt14MABlZaW6rPPPtO4ceNc+iwdEhKi5cuX232+k+TUsd2+fXtduHDBeB0aGlpuPdvzyfDhw40vd4cPH64NGzaouLhY69evN655dVGDBg3k5+en0tJSNWnSRLfddptatWqloKAgPfXUU5Wu99577xm/4mnQoIFGjx4tqSwRVVpaWu4XDI899pheeuklSdKWLVuMn49HRETojTfekMlkUnp6uvGFR03uy3xddceo7a8fHL23q+y95KjQ0FAVFBQoNDRUt912m6KiolS/fn3FxMRo5syZLm27d+/eRkJy586daty4sQ4fPqywsDCFhITo5MmT+vnnn+1+/dO7d29Jjt1T1OT6WN25wnofHxISottuu03t27c3zgm214/s7GwdPXq0XKw9e/as8XsxPDxc4eHhdmO43n777VV+YTZmzJhy10tHcgPdunWzyzckJCQY+ZeGDRsa94tXu+eee/TXv/5V3333nbKzs8sN1WU911f0mffqv7Ftmfz8fE2dOtVIvo4cOdLul3aeRgIWFZowYYLWrVtX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",
      "text/plain": [
       "<Figure size 1395x530 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# --- CONFIG ---\n",
    "\n",
    "# Updated paths for publication-ready repository\n",
    "OPENFIM_ROOT = \"../data/benchmarks/openfim_baseline\"\n",
    "TRAINVAL_ROOT = \"../data/benchmarks/neural_model\"\n",
    "\n",
    "import os, glob, json\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "from IPython.display import display\n",
    "\n",
    "import matplotlib as mpl\n",
    "mpl.rcParams.update({\n",
    "    \"font.size\": 13.0,\n",
    "    \"axes.labelsize\": 14.0,\n",
    "    \"xtick.labelsize\": 12.5,\n",
    "    \"ytick.labelsize\": 12.5,\n",
    "    \"legend.fontsize\": 12.5,\n",
    "\n",
    "    \"font.weight\": \"semibold\",         # bold-ish default\n",
    "    \"axes.labelweight\": \"semibold\",\n",
    "    \"axes.titleweight\": \"semibold\",\n",
    "\n",
    "    \"text.color\": \"0.10\",              # dark gray for better print contrast\n",
    "    \"axes.labelcolor\": \"0.10\",\n",
    "    \"xtick.color\": \"0.10\",\n",
    "    \"ytick.color\": \"0.10\",\n",
    "    \"axes.edgecolor\": \"0.10\",\n",
    "\n",
    "    \"grid.color\": \"0.80\",\n",
    "    \"grid.linestyle\": \":\",\n",
    "    \"grid.linewidth\": 0.9,\n",
    "\n",
    "    # keep text selectable in exported PDFs\n",
    "    \"pdf.fonttype\": 42,\n",
    "    \"ps.fonttype\": 42,\n",
    "    \"savefig.transparent\": True,\n",
    "})\n",
    "\n",
    "\n",
    "def _latest(path_glob):\n",
    "    files = glob.glob(path_glob)\n",
    "    if not files:\n",
    "        return None\n",
    "    files.sort(key=lambda p: os.path.getmtime(p))\n",
    "    return files[-1]\n",
    "\n",
    "def read_json(path):\n",
    "    with open(path, \"r\") as f:\n",
    "        return json.load(f)\n",
    "\n",
    "def collect_openfim(openfim_root):\n",
    "    rows = []\n",
    "    for d in sorted(glob.glob(os.path.join(openfim_root, \"*\"))):\n",
    "        if not os.path.isdir(d): \n",
    "            continue\n",
    "        ds = os.path.basename(os.path.normpath(d))\n",
    "        j = _latest(os.path.join(d, f\"openfim_results_{ds}.json\")) or _latest(os.path.join(d, \"*results*.json\"))\n",
    "        if not j:\n",
    "            continue\n",
    "        data = read_json(j)\n",
    "        H_mean = data.get(\"H_mean\", None)\n",
    "        H_std  = data.get(\"H_std\", None)\n",
    "        if H_mean is None:\n",
    "            continue\n",
    "        rows.append({\n",
    "            \"dataset\": ds,\n",
    "            \"H_openfim_mean\": float(H_mean),\n",
    "            \"H_openfim_time_std\": float(H_std) if H_std is not None else np.nan,\n",
    "            \"openfim_json\": j,\n",
    "        })\n",
    "    return pd.DataFrame(rows)\n",
    "\n",
    "def collect_trainval(trainval_root):\n",
    "    rows = []\n",
    "    for d in sorted(glob.glob(os.path.join(trainval_root, \"*\"))):\n",
    "        if not os.path.isdir(d):\n",
    "            continue\n",
    "        ds = os.path.basename(os.path.normpath(d))\n",
    "        jsonl = os.path.join(d, \"metrics.jsonl\")\n",
    "        best_epoch = None\n",
    "        H_mean = None\n",
    "        src = None\n",
    "\n",
    "        if os.path.isfile(jsonl):\n",
    "            records = []\n",
    "            with open(jsonl, \"r\") as f:\n",
    "                for line in f:\n",
    "                    line = line.strip()\n",
    "                    if not line:\n",
    "                        continue\n",
    "                    try:\n",
    "                        records.append(json.loads(line))\n",
    "                    except Exception:\n",
    "                        pass\n",
    "            if records:\n",
    "                df = pd.DataFrame(records)\n",
    "                if \"val_loss\" in df.columns:\n",
    "                    idx = df[\"val_loss\"].astype(float).idxmin()\n",
    "                    row = df.loc[idx]\n",
    "                    best_epoch = int(row[\"epoch\"])\n",
    "                    if \"H_mean\" in df.columns:\n",
    "                        H_mean = float(row[\"H_mean\"])\n",
    "                        src = \"metrics.jsonl\"\n",
    "\n",
    "        if H_mean is None:\n",
    "            eval_files = glob.glob(os.path.join(d, \"epoch_*_eval.json\"))\n",
    "            if eval_files:\n",
    "                cand = []\n",
    "                for p in eval_files:\n",
    "                    try:\n",
    "                        ev = read_json(p)\n",
    "                        val_loss = float(ev.get(\"val_loss\", np.inf))\n",
    "                        epoch = int(os.path.basename(p).split(\"_\")[1].split(\".\")[0])\n",
    "                        cand.append((val_loss, epoch, p))\n",
    "                    except Exception:\n",
    "                        pass\n",
    "                if cand:\n",
    "                    cand.sort(key=lambda x: (x[0], x[1]))\n",
    "                    _, best_epoch, _ = cand[0]\n",
    "                    fore_path = os.path.join(d, f\"epoch_{best_epoch:03d}_forecast.json\")\n",
    "                    if os.path.isfile(fore_path):\n",
    "                        fm = read_json(fore_path)\n",
    "                        try:\n",
    "                            H_mean = float(fm[\"Hellinger\"][\"mean\"])\n",
    "                            src = f\"epoch_{best_epoch:03d}_forecast.json\"\n",
    "                        except Exception:\n",
    "                            pass\n",
    "\n",
    "        if H_mean is not None:\n",
    "            rows.append({\n",
    "                \"dataset\": ds,\n",
    "                \"H_trainval_mean\": H_mean,\n",
    "                \"val_best_epoch\": best_epoch,\n",
    "                \"trainval_source\": src,\n",
    "                \"trainval_dir\": d,\n",
    "            })\n",
    "    return pd.DataFrame(rows)\n",
    "\n",
    "THRESHOLD = 0.15\n",
    "\n",
    "def by_class_summary(openfim_root, trainval_root, threshold=THRESHOLD):\n",
    "    openfim_root = os.path.expanduser(openfim_root)\n",
    "    trainval_root = os.path.expanduser(trainval_root)\n",
    "    df_fim = collect_openfim(openfim_root)                 # may be empty or partial\n",
    "    df_tv  = collect_trainval(trainval_root)               # must exist\n",
    "\n",
    "    if df_tv.empty:\n",
    "        raise RuntimeError(\"No Train/Val results found; cannot apply threshold filtering.\")\n",
    "\n",
    "    # --- class parser (keep your current behavior) ---\n",
    "    def graph_class_from_name(name: str) -> str:\n",
    "        return name.split(\"_\", 1)[0] if \"_\" in name else name\n",
    "\n",
    "    # --- Train/Val: mark classes and apply threshold filter (ONLY here) ---\n",
    "    df_tv[\"graph_class\"] = df_tv[\"dataset\"].map(graph_class_from_name)\n",
    "    df_tv[\"skip\"] = df_tv[\"H_trainval_mean\"] >= float(threshold)\n",
    "\n",
    "    # summary (Train/Val only) - computed but not displayed\n",
    "    skip_summary = (\n",
    "        df_tv.groupby(\"graph_class\")[\"skip\"]\n",
    "            .agg(total=\"count\", skipped=\"sum\")\n",
    "            .assign(kept=lambda x: x[\"total\"] - x[\"skipped\"])\n",
    "            .reset_index()\n",
    "            .sort_values(\"graph_class\")\n",
    "    )\n",
    "    total_skipped = int(df_tv[\"skip\"].sum())\n",
    "    total_tv = int(len(df_tv))\n",
    "\n",
    "    # Keep ONLY Train/Val under threshold (this determines the comparison set)\n",
    "    df_tv_kept = df_tv.loc[~df_tv[\"skip\"], [\"dataset\", \"H_trainval_mean\", \"graph_class\"]].copy()\n",
    "    if df_tv_kept.empty:\n",
    "        print(\"No Train/Val datasets under threshold; nothing to aggregate.\")\n",
    "        return skip_summary, pd.DataFrame()\n",
    "\n",
    "    # --- Bring OpenFIM values (NO filtering by OpenFIM; left-join) ---\n",
    "    if not df_fim.empty:\n",
    "        df_sel = pd.merge(\n",
    "            df_tv_kept,\n",
    "            df_fim[[\"dataset\", \"H_openfim_mean\"]],\n",
    "            on=\"dataset\",\n",
    "            how=\"left\"\n",
    "        )\n",
    "    else:\n",
    "        df_sel = df_tv_kept.copy()\n",
    "        df_sel[\"H_openfim_mean\"] = np.nan\n",
    "\n",
    "    # --- Per-class aggregation on the Train/Val-kept set ---\n",
    "    def _mean_std_count(a):\n",
    "        arr = pd.to_numeric(a, errors=\"coerce\").to_numpy(dtype=float)\n",
    "        mask = np.isfinite(arr)\n",
    "        n = int(mask.sum())\n",
    "        if n == 0:\n",
    "            return np.nan, np.nan, 0\n",
    "        mean = float(np.nanmean(arr[mask]))\n",
    "        std  = float(np.nanstd(arr[mask], ddof=1)) if n > 1 else np.nan\n",
    "        return mean, std, n\n",
    "\n",
    "    rows = []\n",
    "    for g, gdf in df_sel.groupby(\"graph_class\"):\n",
    "        tv_m, tv_s, tv_n   = _mean_std_count(gdf[\"H_trainval_mean\"])\n",
    "        fim_m, fim_s, fim_n = _mean_std_count(gdf[\"H_openfim_mean\"])\n",
    "        rows.append({\n",
    "            \"graph_class\": g,\n",
    "            \"N_files\": int(len(gdf)),          # number of Train/Val-kept files in this class\n",
    "            \"TrainVal_mean\": tv_m,\n",
    "            \"TrainVal_std\": tv_s,\n",
    "            \"TrainVal_N\": tv_n,                # number actually used (should equal N_files)\n",
    "            \"OpenFIM_mean\": fim_m,             # may be NaN if no OpenFIM for those datasets\n",
    "            \"OpenFIM_std\": fim_s,\n",
    "            \"OpenFIM_N\": fim_n\n",
    "        })\n",
    "    class_df = pd.DataFrame(rows).sort_values(\"graph_class\").reset_index(drop=True)\n",
    "\n",
    "    # --- Overall row across the Train/Val-kept selection ---\n",
    "    tv_all_m, tv_all_s, tv_all_n   = _mean_std_count(df_sel[\"H_trainval_mean\"])\n",
    "    fim_all_m, fim_all_s, fim_all_n = _mean_std_count(df_sel[\"H_openfim_mean\"])\n",
    "    overall = pd.DataFrame([{\n",
    "        \"graph_class\": \"<ALL (TV-kept)>\",\n",
    "        \"N_files\": int(len(df_sel)),\n",
    "        \"TrainVal_mean\": tv_all_m,\n",
    "        \"TrainVal_std\":  tv_all_s,\n",
    "        \"TrainVal_N\":    tv_all_n,\n",
    "        \"OpenFIM_mean\":  fim_all_m,\n",
    "        \"OpenFIM_std\":   fim_all_s,\n",
    "        \"OpenFIM_N\":     fim_all_n,\n",
    "    }])\n",
    "\n",
    "    # --- Display tables ---\n",
    "    fmt = {\n",
    "        \"OpenFIM_mean\": \"{:.6f}\", \"OpenFIM_std\": \"{:.6f}\",\n",
    "        \"TrainVal_mean\": \"{:.6f}\", \"TrainVal_std\": \"{:.6f}\",\n",
    "    }\n",
    "    display(class_df.style.format(fmt).set_caption(\n",
    "        f\"Per-graph-class Hellinger (aggregated over Train/Val-kept datasets only; THRESHOLD={threshold})\"\n",
    "    ))\n",
    "    display(overall.style.format(fmt).set_caption(\n",
    "        \"Overall across Train/Val-kept datasets (OpenFIM used when available)\"\n",
    "    ))\n",
    "\n",
    "    if len(class_df):\n",
    "        classes = class_df[\"graph_class\"].tolist()\n",
    "        x = np.arange(len(classes), dtype=float)\n",
    "\n",
    "        # geometry\n",
    "        group_gap = 0.35\n",
    "        bar_width = 0.30\n",
    "        inner_gap = 0.10\n",
    "        offsets = np.array([-(bar_width/2 + inner_gap/2),\n",
    "                            +(bar_width/2 + inner_gap/2)], dtype=float)\n",
    "\n",
    "        # colors: slightly darker, still friendly\n",
    "        def pastelize(rgb, mix=0.50):\n",
    "            return tuple((1 - mix) * np.array(rgb) + mix * np.ones(3))\n",
    "        color_fim  = pastelize((0.85, 0.20, 0.20), mix=0.50)  # red-ish\n",
    "        color_ours = pastelize((0.20, 0.35, 0.85), mix=0.50)  # blue-ish\n",
    "        colors = [color_fim, color_ours]\n",
    "        edgecolor = \"0.15\"\n",
    "        lw_bar = 1.1\n",
    "\n",
    "        # figure size\n",
    "        fig_w = max(9.5, 0.95 * len(classes) + 3.5)\n",
    "        fig_h = 5.3\n",
    "        fig, ax = plt.subplots(figsize=(fig_w, fig_h))\n",
    "\n",
    "        # centers for each class\n",
    "        x_positions = x * (2*bar_width + inner_gap + group_gap)\n",
    "\n",
    "        # data\n",
    "        means = [class_df[\"OpenFIM_mean\"].to_numpy(),\n",
    "                 class_df[\"TrainVal_mean\"].to_numpy()]\n",
    "        stds  = [np.nan_to_num(class_df[\"OpenFIM_std\"].to_numpy(), nan=0.0),\n",
    "                 np.nan_to_num(class_df[\"TrainVal_std\"].to_numpy(), nan=0.0)]\n",
    "        labels = [\"OpenFIM\", \"Ours\"]\n",
    "\n",
    "        # better errorbar styling for bars\n",
    "        err_kw = dict(elinewidth=1.3, ecolor=\"0.25\", capthick=1.2)\n",
    "\n",
    "        # plot\n",
    "        for b in range(2):\n",
    "            xpos = x_positions + offsets[b]\n",
    "            ax.bar(\n",
    "                xpos, means[b], bar_width,\n",
    "                yerr=stds[b], capsize=5.5,\n",
    "                error_kw=err_kw,\n",
    "                label=labels[b],\n",
    "                color=colors[b], edgecolor=edgecolor, linewidth=lw_bar\n",
    "            )\n",
    "\n",
    "        # labels & ticks\n",
    "        ax.set_ylabel(\"Hellinger\")  # lateral label (kept short)\n",
    "        ax.set_xticks(x_positions)\n",
    "        ax.set_xticklabels(classes, rotation=0, ha=\"center\")\n",
    "\n",
    "        # tick appearance: a bit larger/clearer\n",
    "        ax.tick_params(axis=\"both\", which=\"major\", width=1.1, length=5, direction=\"out\")  # thicker ticks\n",
    "        # spines darker & slightly thicker\n",
    "        for sp in ax.spines.values():\n",
    "            sp.set_linewidth(1.1)\n",
    "            sp.set_color(\"0.15\")\n",
    "\n",
    "        # grid & headroom\n",
    "        ax.grid(axis=\"y\")\n",
    "        ax.set_axisbelow(True)\n",
    "        y_candidates = []\n",
    "        for m, s in zip(means, stds):\n",
    "            y_candidates.append(np.nan_to_num(m, nan=0.0) + s)\n",
    "        y_max = float(np.nanmax(np.concatenate(y_candidates))) if len(y_candidates) else 1.0\n",
    "        if np.isfinite(y_max):\n",
    "            ax.set_ylim(0.0, max(0.05, y_max) * 1.18)\n",
    "\n",
    "        # legend: clean, slightly larger handles\n",
    "        leg = ax.legend(frameon=False, ncols=2, loc=\"upper right\", handlelength=1.6, handletextpad=0.6)\n",
    "        for txt in leg.get_texts():\n",
    "            txt.set_color(\"0.10\")\n",
    "\n",
    "        # no title (per your preference)\n",
    "        plt.tight_layout()\n",
    "\n",
    "        # Export vector (PDF/SVG) + show\n",
    "        fig.savefig(\"hellinger_plot.pdf\", bbox_inches=\"tight\")  # vector; DPI irrelevant\n",
    "        fig.savefig(\"hellinger_plot.svg\", bbox_inches=\"tight\")\n",
    "        plt.show()\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "    return skip_summary, (class_df, overall)\n",
    "\n",
    "\n",
    "\n",
    "# ---- Run it\n",
    "_ = by_class_summary(OPENFIM_ROOT, TRAINVAL_ROOT, threshold=THRESHOLD)\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.12.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
