{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "5d289e4e-8334-4cbd-9b3e-0eef2ab5fc9e",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "%reload_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "7363774a-08a1-4536-bce0-19ffb9415952",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "import json\n",
    "import logging\n",
    "from IPython.display import display, HTML, Markdown, display_html\n",
    "from pathlib import Path\n",
    "from typing import Dict, List, Optional\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import networkx as nx\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "\n",
    "import egr.pandas_helpers as ph\n",
    "import nb.reports.helpers as hlp\n",
    "import nb.reports.vis_helpers as vh\n",
    "from egr.util import load_graph, load_features\n",
    "from egr.log import init_logging\n",
    "\n",
    "LOG = logging.getLogger('report')\n",
    "\n",
    "init_logging(level_name='info')\n",
    "pd.set_option('display.precision', 2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "id": "4b850c0a-816e-4da8-b0ee-29927c246684",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "RUN_ID: str = 'iclr-24'\n",
    "\n",
    "all_samples = [f'{i:04d}' for i in range(1, 6)]\n",
    "# control_rounds = ['rR', 'rL', 'rP2']\n",
    "control_rounds = ['rR']\n",
    "eegl_rounds = [f'r{i}' for i in range(3)]\n",
    "data_root: Path = Path(f'/data/results')\n",
    "exp_root: Path = data_root / RUN_ID\n",
    "fsg_args = dict(run_id=RUN_ID, tags=eegl_rounds)\n",
    "num_features: int = 10\n",
    "ncols: int = num_features"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 40,
   "id": "2ee52561-9fda-4afc-9197-72e344781174",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "def plot_adv(pattern):\n",
    "    pattern_dir: Path = data_root / 'input_data/pattern_graphs' / pattern\n",
    "    file_paths = list(file_path for file_path in sorted(pattern_dir.glob('*.json')))\n",
    "    fig, axs = plt.subplots(nrows=2, ncols=5, figsize=(15, 6))\n",
    "    for i, path in enumerate(file_paths):\n",
    "        r = i // 5\n",
    "        c = i % 5\n",
    "        G: nx.Graph = load_graph(path)\n",
    "        center = G.graph.get('__root__', nx.center(G)[0])\n",
    "        G.graph['__root__'] = center\n",
    "        G.nodes[center]['__root__'] = True\n",
    "        colors = ['grey'] * G.number_of_nodes()\n",
    "        colors[center] = 'red'\n",
    "        nx.draw(\n",
    "            G,\n",
    "            pos=nx.kamada_kawai_layout(G),\n",
    "            # pos=nx.planar_layout(G),\n",
    "            ax=axs[r, c],\n",
    "            node_size=250,\n",
    "            width=0.5,\n",
    "            node_color=colors,\n",
    "            labels={n:n for n in G.nodes()}\n",
    "        )\n",
    "        axs[r, c].axis('on')\n",
    "    plt.show()\n",
    "    \n",
    "\n",
    "def plot_pattern_features(variant, pattern, sample, save=False):\n",
    "    path = data_root / 'input_data' / variant / pattern / f'{sample}.npy'\n",
    "    X = load_features(path)\n",
    "    _, ax = plt.subplots(figsize=(5, 5))\n",
    "    ax.set_xticks([i for i in range(X.shape[1])])\n",
    "    ax.set_yticks([i for i in range(X.shape[0]) if i % 100 == 0])\n",
    "    ax.set_xlabel('Annotated Features')\n",
    "    ax.set_ylabel('Node IDs')\n",
    "    ax.imshow(X, aspect='auto', vmin=0, vmax=1, cmap=plt.get_cmap('Greys'), interpolation='none')\n",
    "    plt.show()\n",
    "    # if save:\n",
    "    #     path = exp_root / 'vis' / 'adversarial-features.pdf'\n",
    "    #     plt.savefig()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 45,
   "id": "b9e2160c-37ac-4a3d-b360-cd4fbc92eb9c",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "# VARIANT: str = 'p009-02'\n",
    "# NUM_FOLDS: int = 10\n",
    "# SAMPLE = '0002'\n",
    "PATTERN = 'P4'\n",
    "\n",
    "# def generate_latex_table(df, caption, width, fmt=None):\n",
    "#     kw = dict(caption=caption, hrules=True, position='H')\n",
    "\n",
    "#     txt = '\\\\begin{minipage}{' + str(width) + '\\linewidth}\\n'\n",
    "#     # print(begin)\n",
    "#     if fmt:\n",
    "#         # print(df.style.format('{:.2f}').to_latex())\n",
    "#         txt += df.style.format('{:.2f}').to_latex(**kw)\n",
    "#     else:\n",
    "#         # print(df.style.to_latex())\n",
    "#         txt += df.style.to_latex(**kw)\n",
    "#     txt += '\\\\end{minipage}'\n",
    "#     print(txt)\n",
    "\n",
    "def get_tex(df, fmt) -> str:\n",
    "    if fmt:\n",
    "        return df.to_latex(float_format='{:.2f}'.format)\n",
    "    return '\\\\vspace{-2cm}\\n' + df.to_latex()\n",
    "\n",
    "\n",
    "def generate_latex_table(df, caption, width, fmt=None):\n",
    "    table = get_tex(df, fmt)\n",
    "    print('\\parbox{' + f'{width}' + '\\linewidth}{\\n' + table + '}')\n",
    "\n",
    "\n",
    "def generate_report(run_id, variant, samples, num_folds, pattern=PATTERN):\n",
    "    spec = hlp.get_variant_spec(variant)\n",
    "    display(Markdown(f'# <center>Detailed Report: {spec.name}</center>'))    \n",
    "    display(Markdown(f'## <center>{spec.desc}</center>'))\n",
    "\n",
    "\n",
    "    # plot_adv(pattern)\n",
    "\n",
    "    # display(Markdown('''### Annotated features\n",
    "    # The following shows the feature matrix that were generated by annotated based on the presence/absence \n",
    "    # of adversarial patterns rooted at given nodes. The nodes of the main graph are on the Y-axis and the \n",
    "    # X-axis corresponds to the presence/absence of the feature. Black=1, White=0.'''))\n",
    "    # plot_pattern_features(variant, pattern, samples[0], save=True)\n",
    "    # all_rounds = ['rL', f'r{pattern}', 'rR', 'r0', 'r1', 'r2', 'r3']\n",
    "    all_rounds = ['rR', 'r0', 'r1', 'r2', 'r3', 'r4', 'r5']\n",
    "    \n",
    "\n",
    "    for i in range(num_folds):\n",
    "        fold = i + 1\n",
    "        # args = hlp.ReportArgs(\n",
    "        #     dirpath=exp_root,\n",
    "        #     variant=variant,\n",
    "        #     tags=all_rounds,\n",
    "        #     samples=samples,\n",
    "        #     fold=fold,\n",
    "        #     save=True,\n",
    "        #     eegl_only=True\n",
    "        # )\n",
    "\n",
    "        display(Markdown('---'))\n",
    "        display(Markdown(f'### <center>Fold: {fold}</center>'))\n",
    "        df_metrics = hlp.make_table(exp_root, samples, variant, all_rounds, fold)\n",
    "        df_labels, title_labels = hlp.make_label_stats(variant, fold, data_root)\n",
    "\n",
    "        print(df_metrics)\n",
    "        print(df_labels)\n",
    "        # print(df_metrics.style.format('{:.2f}').to_latex(caption=f'Classification results for fold: {fold:02d}', hrules=True))\n",
    "        # print(df_labels.style.format('{:.2f}').to_latex(caption=f'Label frequencies for fold: {fold:02d}', hrules=True))\n",
    "\n",
    "        # LaTeX table\n",
    "        # generate_latex_table(\n",
    "        #     df_metrics[['F1-Score']],\n",
    "        #     f'Classification results for fold: {fold:02d}',\n",
    "        #     width=0.35,\n",
    "        #     fmt='.2f'\n",
    "        # )\n",
    "        # print('\\\\hfill')\n",
    "        # generate_latex_table(\n",
    "        #     df_labels,\n",
    "        #     f'Label frequencies for fold: {fold:02d}',\n",
    "        #     width=0.65\n",
    "        # )\n",
    "        ############\n",
    "\n",
    "        # # vh.show_side_by_size((df_metrics, 'Metrics'), (df_labels, title_labels))\n",
    "        # plt.clf()\n",
    "        # hlp.plot_conf_mats(args, figsize=(14, 5), fontsize=14)\n",
    "        # # vh.fsg(run_id=run_id, variant=variant, samples=samples, tags=['r0', 'r1'], fold=fold, save=True)\n",
    "        # plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "id": "7f0591ee-88fe-4172-afe6-b3e087367b34",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "text/markdown": [
       "# <center>Detailed Report: $M_2^{''}$</center>"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/markdown": [
       "## <center>Base graph size: $300$, #Motif-A: $40$, #Motif-B: $40$</center>"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/markdown": [
       "---"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/markdown": [
       "### <center>Fold: 1</center>"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "                   Precision  Recall  F1-Score\n",
      "Random                 91.51   94.68     92.79\n",
      "(Vanilla) Round-0      59.31   63.83     60.20\n",
      "Round-1                81.93   80.85     80.20\n",
      "Round-2                77.36   78.72     77.85\n",
      "Round-3               100.00  100.00    100.00\n",
      "Round-4               100.00  100.00    100.00\n",
      "Round-5               100.00  100.00    100.00\n",
      "            0  1  2   3  4  5   6\n",
      "Frequency  25  6  6  22  5  2  28\n"
     ]
    },
    {
     "data": {
      "text/markdown": [
       "---"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/markdown": [
       "### <center>Fold: 2</center>"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "                   Precision  Recall  F1-Score\n",
      "Random                100.00  100.00    100.00\n",
      "(Vanilla) Round-0      48.15   63.83     53.39\n",
      "Round-1                67.51   67.02     61.55\n",
      "Round-2               100.00  100.00    100.00\n",
      "Round-3               100.00  100.00    100.00\n",
      "Round-4               100.00  100.00    100.00\n",
      "Round-5               100.00  100.00    100.00\n",
      "            0  1  2   3  4  5   6\n",
      "Frequency  32  3  2  26  7  3  21\n"
     ]
    },
    {
     "data": {
      "text/markdown": [
       "---"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/markdown": [
       "### <center>Fold: 3</center>"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "                   Precision  Recall  F1-Score\n",
      "Random                 95.19   94.68     94.60\n",
      "(Vanilla) Round-0      76.87   67.02     58.31\n",
      "Round-1                49.15   58.51     50.78\n",
      "Round-2                55.91   61.70     57.38\n",
      "Round-3                67.27   64.89     59.27\n",
      "Round-4                68.42   67.02     62.74\n",
      "Round-5                72.38   69.15     64.83\n",
      "            0  1  2   3  4  5   6\n",
      "Frequency  33  3  3  23  3  4  25\n"
     ]
    },
    {
     "data": {
      "text/markdown": [
       "---"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/markdown": [
       "### <center>Fold: 4</center>"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "                   Precision  Recall  F1-Score\n",
      "Random                 96.04   95.74     95.31\n",
      "(Vanilla) Round-0      62.70   64.89     56.60\n",
      "Round-1                54.29   60.64     55.97\n",
      "Round-2                62.89   68.09     64.33\n",
      "Round-3                68.41   70.21     67.95\n",
      "Round-4                59.61   61.70     56.05\n",
      "Round-5                65.58   64.89     60.32\n",
      "            0  1  2   3  4  5   6\n",
      "Frequency  39  6  3  17  2  5  22\n"
     ]
    },
    {
     "data": {
      "text/markdown": [
       "---"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/markdown": [
       "### <center>Fold: 5</center>"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "                   Precision  Recall  F1-Score\n",
      "Random                 91.04   91.49     91.10\n",
      "(Vanilla) Round-0      63.21   67.02     62.25\n",
      "Round-1                72.42   71.28     70.16\n",
      "Round-2                98.04   97.87     97.76\n",
      "Round-3                98.04   97.87     97.76\n",
      "Round-4                98.13   97.87     97.79\n",
      "Round-5                98.13   97.87     97.79\n",
      "            0  1  2   3  4  5   6\n",
      "Frequency  24  9  5  19  4  4  29\n"
     ]
    },
    {
     "data": {
      "text/markdown": [
       "---"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/markdown": [
       "### <center>Fold: 6</center>"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "                   Precision  Recall  F1-Score\n",
      "Random                 97.16   96.81     96.80\n",
      "(Vanilla) Round-0      71.24   68.09     62.73\n",
      "Round-1                84.08   77.66     75.38\n",
      "Round-2                98.98   98.94     98.87\n",
      "Round-3                98.98   98.94     98.87\n",
      "Round-4                98.98   98.94     98.87\n",
      "Round-5                98.98   98.94     98.87\n",
      "            0  1  2   3  4  5   6\n",
      "Frequency  25  3  3  29  6  4  24\n"
     ]
    },
    {
     "data": {
      "text/markdown": [
       "---"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/markdown": [
       "### <center>Fold: 7</center>"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "                   Precision  Recall  F1-Score\n",
      "Random                 88.05   86.17     85.70\n",
      "(Vanilla) Round-0      62.57   65.96     58.34\n",
      "Round-1               100.00  100.00    100.00\n",
      "Round-2                99.15   98.94     98.98\n",
      "Round-3               100.00  100.00    100.00\n",
      "Round-4               100.00  100.00    100.00\n",
      "Round-5               100.00  100.00    100.00\n",
      "            0  1  2   3  4  5   6\n",
      "Frequency  31  6  5  19  3  4  26\n"
     ]
    },
    {
     "data": {
      "text/markdown": [
       "---"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/markdown": [
       "### <center>Fold: 8</center>"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "                   Precision  Recall  F1-Score\n",
      "Random                 94.36   93.62     93.38\n",
      "(Vanilla) Round-0      65.96   68.09     63.06\n",
      "Round-1                68.91   69.15     66.94\n",
      "Round-2               100.00  100.00    100.00\n",
      "Round-3                98.98   98.94     98.87\n",
      "Round-4                98.97   98.94     98.93\n",
      "Round-5               100.00  100.00    100.00\n",
      "            0  1  2   3  4  5   6\n",
      "Frequency  27  1  4  33  1  7  21\n"
     ]
    },
    {
     "data": {
      "text/markdown": [
       "---"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/markdown": [
       "### <center>Fold: 9</center>"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "                   Precision  Recall  F1-Score\n",
      "Random                 99.15   98.94     98.92\n",
      "(Vanilla) Round-0      69.50   70.21     67.09\n",
      "Round-1                72.87   65.96     59.85\n",
      "Round-2               100.00  100.00    100.00\n",
      "Round-3                98.98   98.94     98.87\n",
      "Round-4                98.97   98.94     98.87\n",
      "Round-5               100.00  100.00    100.00\n",
      "            0  1  2   3  4  5   6\n",
      "Frequency  29  1  3  24  6  4  27\n"
     ]
    },
    {
     "data": {
      "text/markdown": [
       "---"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/markdown": [
       "### <center>Fold: 10</center>"
      ],
      "text/plain": [
       "<IPython.core.display.Markdown object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "                   Precision  Recall  F1-Score\n",
      "Random                 96.50   96.81     96.59\n",
      "(Vanilla) Round-0      54.13   61.70     54.15\n",
      "Round-1                72.02   67.02     59.77\n",
      "Round-2                98.98   98.94     98.87\n",
      "Round-3                98.98   98.94     98.87\n",
      "Round-4               100.00  100.00    100.00\n",
      "Round-5                98.98   98.94     98.87\n",
      "            0  1  2   3  4  5   6\n",
      "Frequency  35  2  6  28  3  3  17\n"
     ]
    }
   ],
   "source": [
    "generate_report(\n",
    "    RUN_ID,\n",
    "    variant='p009-02',\n",
    "    samples=['0002'],\n",
    "    num_folds=10\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8243bc55-d15f-4083-9109-c2dc4d732079",
   "metadata": {},
   "source": [
    "[<< All Reports](http://pods.evl.uic.edu/eegl/230510/main.html)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7ff3e67c-eaff-4a37-908c-34320e49b6c2",
   "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.11.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
