{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "c01b7039",
   "metadata": {},
   "source": [
    "# Table Aggregation\n",
    "\n",
    "In this notebook we aggregate the raw data into tables as used in our figures."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "3142bea7",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "announced-sample",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "\n",
    "def aggregate(df):\n",
    "    best_of_10 = df.groupby([\"sample_id\"]).max()\n",
    "    best_of_10[\"num_full_match\"] = (best_of_10[\"consistency\"] == 1.0)\n",
    "    res = best_of_10.groupby([\"num_nodes\", \"num_networks\"]).mean()\n",
    "    res[\"num_full_match\"] = best_of_10.groupby([\"num_nodes\",\"num_networks\"]).sum()[\"num_full_match\"]\n",
    "    return res\n",
    "\n",
    "def aggregate_3pred(df, consistency_column=\"overall\"):\n",
    "    best_of_10 = df.groupby([\"sample_id\"]).max()\n",
    "    best_of_10[\"num_full_match\"] = (best_of_10[\"overall\"] == 1.0)\n",
    "    best_of_10[\"num_close_match\"] = (best_of_10[\"overall\"] > 0.95)\n",
    "    best_of_10[\"num_close_match90\"] = (best_of_10[\"overall\"] > 0.9)\n",
    "    best_of_10[\"num_full_fwd\"] = (best_of_10[\"fwd\"] == 1.0)\n",
    "    best_of_10[\"num_reachable\"] = (best_of_10[\"reachable\"] == 1.0) # np.logical_and((best_of_10[\"fwd\"] == 1.0).values, (best_of_10[\"reachable\"] == 1.0).values)\n",
    "    best_of_10[\"num_trafficIsolation\"] = (best_of_10[\"trafficIsolation\"] == 1.0) # np.logical_and((best_of_10[\"fwd\"] == 1.0).values, (best_of_10[\"reachable\"] == 1.0).values)\n",
    "    res = best_of_10.groupby([\"num_nodes\", \"num_networks\"]).mean()\n",
    "    res[\"num_full_match\"] = best_of_10.groupby([\"num_nodes\",\"num_networks\"]).sum()[\"num_full_match\"]\n",
    "    res[\"num_full_fwd\"] = best_of_10.groupby([\"num_nodes\",\"num_networks\"]).sum()[\"num_full_fwd\"]\n",
    "    res[\"num_reachable\"] = best_of_10.groupby([\"num_nodes\",\"num_networks\"]).sum()[\"num_reachable\"]\n",
    "    res[\"num_trafficIsolation\"] = best_of_10.groupby([\"num_nodes\",\"num_networks\"]).sum()[\"num_trafficIsolation\"]\n",
    "    res[\"num_close_match\"] = best_of_10.groupby([\"num_nodes\",\"num_networks\"]).sum()[\"num_close_match\"]\n",
    "    res[\"num_close_match90\"] = best_of_10.groupby([\"num_nodes\",\"num_networks\"]).sum()[\"num_close_match90\"]\n",
    "    return res"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "c6b2e41e",
   "metadata": {},
   "outputs": [],
   "source": [
    "def aggregate_paper(df, consistency_column=\"overall\", aggregate_by_n=False):\n",
    "    best_of_10 = df.groupby([\"sample_id\"]).max()\n",
    "    best_of_10[\"num_full_match\"] = (best_of_10[\"overall\"] == 1.0)\n",
    "    best_of_10[\"num_close_match\"] = (best_of_10[\"overall\"] > 0.95)\n",
    "    best_of_10[\"num_close_match90\"] = (best_of_10[\"overall\"] > 0.9)\n",
    "    best_of_10[\"num_full_fwd\"] = (best_of_10[\"fwd\"] == 1.0)\n",
    "    best_of_10[\"num_reachable\"] = (best_of_10[\"reachable\"] == 1.0) # np.logical_and((best_of_10[\"fwd\"] == 1.0).values, (best_of_10[\"reachable\"] == 1.0).values)\n",
    "    best_of_10[\"num_trafficIsolation\"] = (best_of_10[\"trafficIsolation\"] == 1.0) # np.logical_and((best_of_10[\"fwd\"] == 1.0).values, (best_of_10[\"reachable\"] == 1.0).values)\n",
    "    \n",
    "    num_nodes_values = list(sorted(best_of_10[\"num_nodes\"].values))\n",
    "    if len(num_nodes_values) / 3 != int(len(num_nodes_values) / 3):\n",
    "        print(\"warning: provided samples cannot be evenly distributed into 3 datasets small/medium/large\")\n",
    "    first_boundary = int(len(num_nodes_values) / 3)\n",
    "    second_boundary = int(len(num_nodes_values) * 2 / 3)\n",
    "    # print(\"first_boundary:\", num_nodes_values[first_boundary - 1])\n",
    "    # print(\"second_boundary:\", num_nodes_values[second_boundary - 1])\n",
    "    # print(\"third_boundary:\", num_nodes_values[-1])\n",
    "    # print(num_nodes_values)\n",
    "\n",
    "    best_of_10[\"group\"] = pd.cut(best_of_10[\"num_nodes\"], [0,18,39,153])\n",
    "    assert all([v == 8 for v in best_of_10.groupby(\"group\").count()[\"overall\"].values]), \"The provided samples cannot distributed into three datasets of equal size 8 with num_nodes boundaries [0, 18, 39, 153]: Instead got num_nodes values {}\".format(num_nodes_values)\n",
    "    res = best_of_10.groupby(\"group\").mean()\n",
    "\n",
    "    res[\"num_full_match\"] = best_of_10.groupby(\"group\").sum()[\"num_full_match\"]\n",
    "    res[\"num_full_fwd\"] = best_of_10.groupby(\"group\").sum()[\"num_full_fwd\"]\n",
    "    res[\"num_reachable\"] = best_of_10.groupby(\"group\").sum()[\"num_reachable\"]\n",
    "    res[\"num_trafficIsolation\"] = best_of_10.groupby(\"group\").sum()[\"num_trafficIsolation\"]\n",
    "    res[\"num_close_match\"] = best_of_10.groupby(\"group\").sum()[\"num_close_match\"]\n",
    "    res[\"num_close_match90\"] = best_of_10.groupby(\"group\").sum()[\"num_close_match90\"]\n",
    "    return res\n",
    "    \n",
    "def to_latex_paper(df):\n",
    "    #print(to_latex(df))\n",
    "    lines = []\n",
    "    for row in df.iloc:\n",
    "        def group_name(g): return \"Small\" if g == 0 else (\"Medium\" if g == 1 else \"Large\")\n",
    "        group = group_name(row[\"num_nodes\"])\n",
    "        columns = [\n",
    "            group,\n",
    "            \"%.2f\" % row[\"fwd\"],\n",
    "            \"%.2f\" % row[\"reachable\"],\n",
    "            \"%.2f\" % row[\"trafficIsolation\"],\n",
    "            \"\\\\textbf{%.2f}\" % row[\"overall\"],\n",
    "            \"%.2f\" % (row[\"num_full_match\"] / 8),\n",
    "            \"%.2f\" % (row[\"num_close_match90\"] / 8),\n",
    "        ]\n",
    "        lines.append(\"& \" + \" & \".join(columns) + \"\\\\\\\\\")\n",
    "    return \"\\n\".join(lines)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "e3ed54ea",
   "metadata": {},
   "outputs": [],
   "source": [
    "f = \"results-eval-64-bgp-reqs-16-4shot-manysamples-10235.pkl-results.pkl\"\n",
    "df = pd.read_pickle(f).reset_index()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "70a8d903",
   "metadata": {},
   "outputs": [],
   "source": [
    "def collect_results_up_to(df, i):\n",
    "    res = {}\n",
    "    counts = {}\n",
    "    num_samples = 0\n",
    "    \n",
    "    for row in df.iloc:\n",
    "        num_samples += 1\n",
    "        k = (row[\"sample_id\"], row[\"num_nodes\"])\n",
    "        if not k in counts: \n",
    "            counts[k] = 0\n",
    "            res[k] = 0\n",
    "        if counts[k] > i - 1:\n",
    "            continue\n",
    "        counts[k] += 1\n",
    "        res[k] = max(res[k], row[\"overall\"])\n",
    "    ids = sorted(list(res.keys()))\n",
    "\n",
    "    dataset_avg = {\n",
    "        \"Small\": 0,\n",
    "        \"Medium\": 0,\n",
    "        \"Large\": 0\n",
    "    }\n",
    "    dataset_cnt = {\n",
    "        \"Small\": 0,\n",
    "        \"Medium\": 0,\n",
    "        \"Large\": 0\n",
    "    }\n",
    "    \n",
    "    buckets = [0,18,39,153]\n",
    "    \n",
    "    for id,num_nodes in ids:\n",
    "        d = \"\"\n",
    "        if num_nodes <= buckets[1]:\n",
    "            d = \"Small\"\n",
    "        elif num_nodes <= buckets[2]:\n",
    "            d = \"Medium\"\n",
    "        else:\n",
    "            d = \"Large\"\n",
    "        dataset_avg[d] += res[(id,num_nodes)]\n",
    "        dataset_cnt[d] += 1\n",
    "    \n",
    "    num_samples_per_dataset = list(dataset_cnt.values())[0]\n",
    "    assert all([v == num_samples_per_dataset for v in dataset_cnt.values()]), \"Each dataset should have the same number of samples, not {}\".format(dataset_cnt)\n",
    "\n",
    "    return [v / 8 for d,v in dataset_avg.items()]\n",
    "\n",
    "x = []\n",
    "y_small = []\n",
    "y_medium = []\n",
    "y_large = []\n",
    "\n",
    "for i in range(1, 21):\n",
    "    s,m,l = collect_results_up_to(df, i)\n",
    "    x.append(i)\n",
    "    y_small.append(s)\n",
    "    y_medium.append(m)\n",
    "    y_large.append(l)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "7317bf4e",
   "metadata": {},
   "outputs": [],
   "source": [
    "y_small = np.array(y_small)\n",
    "y_medium = np.array(y_medium)\n",
    "y_large = np.array(y_large)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "8785dc36",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x7f9e113b2cd0>"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "# import tikzplotlib\n",
    "\n",
    "plt.plot(x, y_small, label=\"Small\")\n",
    "plt.plot(x, y_medium, label=\"Medium\")\n",
    "plt.plot(x, y_large, label=\"Large\")\n",
    "\n",
    "# plt.plot(x, (y_small - y_small.min()) / (y_small.max() - y_small.min()), label=\"Small\")\n",
    "# plt.plot(x, (y_medium - y_medium.min()) / (y_medium.max() - y_medium.min()), label=\"Medium\")\n",
    "# plt.plot(x, (y_large - y_large.min()) / (y_large.max() - y_large.min()), label=\"Large\")\n",
    "\n",
    "plt.grid(True,which=\"both\", linestyle='--')\n",
    "#plt.ylim(0.0,1.1)\n",
    "plt.ylabel(\"Best Specification Consistency \")\n",
    "plt.xlabel(\"Number of Samples\")\n",
    "plt.xticks(range(0,21))\n",
    "plt.legend()\n",
    "# tikzplotlib.save(\"test.tex\")\n",
    "# !cat test.tex"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "71b371e6",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "sample_id\n",
       "0     0.918033\n",
       "1     1.000000\n",
       "2     1.000000\n",
       "3     0.951613\n",
       "4     0.921875\n",
       "5     0.869565\n",
       "6     1.000000\n",
       "7     0.890625\n",
       "8     0.945455\n",
       "9     0.980769\n",
       "10    0.933333\n",
       "11    1.000000\n",
       "12    0.958333\n",
       "13    0.984615\n",
       "14    1.000000\n",
       "15    1.000000\n",
       "16    0.945946\n",
       "17    0.868852\n",
       "18    0.887324\n",
       "19    0.814286\n",
       "20    0.983871\n",
       "21    0.928571\n",
       "22    1.000000\n",
       "23    0.844444\n",
       "Name: overall, dtype: float64"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df.groupby([\"sample_id\"]).max()[\"overall\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "56443fec",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "interpreter": {
   "hash": "9c2aca3d0dfc99fb2efc8287a7808a7621f9ab3042aa12ae3d168b7497f5397d"
  },
  "kernelspec": {
   "display_name": "Python 3.9.7 ('nsynth-sub')",
   "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.9.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
