{
 "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": 1,
   "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": 6,
   "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",
    "    if aggregate_by_n:\n",
    "        def group(sample_id):\n",
    "            if sample_id < 33: return 0\n",
    "            elif sample_id < 67: return 1\n",
    "            else: return 2\n",
    "        best_of_10[\"num_nodes\"] = [group(s) for s in best_of_10.reset_index()[\"num_nodes\"].values]\n",
    "    else:\n",
    "        def group(sample_id):\n",
    "            if sample_id < 8: return 0\n",
    "            elif sample_id < 16: return 1\n",
    "            else: return 2\n",
    "        best_of_10[\"num_nodes\"] = [group(s) for s in best_of_10.reset_index()[\"sample_id\"].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\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": null,
   "id": "da27aae8",
   "metadata": {},
   "outputs": [],
   "source": [
    "!ls -lisah ../"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6bfd9398",
   "metadata": {},
   "source": [
    "### Unsat Using Random Sampling"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6e933ab4",
   "metadata": {},
   "outputs": [],
   "source": [
    "f = \"../results-UNSAT_DATASET_LESS-rnd-15473.pkl-results.pkl\"\n",
    "df_2 = aggregate_paper(pd.read_pickle(f), aggregate_by_n=True).reset_index()\n",
    "print(\"$3\\\\times 16$ requirements \" + to_latex_paper(df_2))\n",
    "print(\"\\\\midrule\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b0cdb1bf",
   "metadata": {},
   "source": [
    "### Unsat using model (5 Samples)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "0105a427",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "$3\\times 16$ requirements & Small & 0.97 & 1.00 & 1.00 & \\textbf{0.97} & 0.00 & 0.25\\\\\n",
      "& Medium & 0.97 & 1.00 & 1.00 & \\textbf{0.97} & 0.00 & 0.75\\\\\n",
      "& Large & 0.94 & 1.00 & 1.00 & \\textbf{0.94} & 0.00 & 0.75\\\\\n",
      "\\midrule\n"
     ]
    }
   ],
   "source": [
    "f = \"../results-UNSAT_DATASET_LESS-14709.pkl-results.pkl\"\n",
    "df_2 = aggregate_paper(pd.read_pickle(f), sample_id_mapping=mapping).reset_index()\n",
    "print(\"$3\\\\times 16$ requirements \" + to_latex_paper(df_2))\n",
    "print(\"\\\\midrule\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "497dca6d",
   "metadata": {},
   "source": [
    "### Unsat using model (10 samples)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "5f503f50",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "$3\\times 16$ requirements & Small & 0.86 & 1.00 & 1.00 & \\textbf{0.86} & 0.00 & 0.38\\\\\n",
      "& Medium & 0.87 & 1.00 & 1.00 & \\textbf{0.87} & 0.00 & 0.38\\\\\n",
      "& Large & 0.83 & 1.00 & 1.00 & \\textbf{0.83} & 0.00 & 0.12\\\\\n",
      "\\midrule\n"
     ]
    }
   ],
   "source": [
    "f = \"../results-UNSAT_DATASET-samples10-17480.pkl-results.pkl\"\n",
    "df_2 = aggregate_paper(pd.read_pickle(f)).reset_index()\n",
    "print(\"$3\\\\times 16$ requirements \" + to_latex_paper(df_2))\n",
    "print(\"\\\\midrule\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "a97d22ba",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.8533333333333334"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "(0.86+0.87+0.83)/3"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "ff19d7c0",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "$3\\times 16$ requirements & Small & 0.97 & 0.91 & 0.95 & \\textbf{0.95} & 0.38 & 0.88\\\\\n",
      "& Medium & 0.97 & 0.95 & 0.97 & \\textbf{0.96} & 0.38 & 0.88\\\\\n",
      "& Large & 0.93 & 0.93 & 0.97 & \\textbf{0.93} & 0.12 & 0.88\\\\\n",
      "\\midrule\n"
     ]
    }
   ],
   "source": [
    "f = \"../results-PAPER_DATASET-18885.pkl-results.pkl\"\n",
    "df_2 = aggregate_paper(pd.read_pickle(f)).reset_index()\n",
    "print(\"$3\\\\times 16$ requirements \" + to_latex_paper(df_2))\n",
    "print(\"\\\\midrule\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "2c68d25d",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "$3\\times 2$ requirements & Small & 0.97 & 0.94 & 0.97 & \\textbf{0.96} & 0.25 & 0.88\\\\\n",
      "& Medium & 0.95 & 0.98 & 1.00 & \\textbf{0.97} & 0.12 & 1.00\\\\\n",
      "& Large & 0.93 & 0.92 & 0.98 & \\textbf{0.93} & 0.00 & 0.75\\\\\n",
      "\\midrule\n"
     ]
    }
   ],
   "source": [
    "f = \"../results-PAPER_DATASET-17428.pkl-results.pkl\"\n",
    "df_2 = aggregate_paper(pd.read_pickle(f)).reset_index()\n",
    "print(\"$3\\\\times 2$ requirements \" + to_latex_paper(df_2))\n",
    "print(\"\\\\midrule\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "e8b17ff4",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "$3\\times 2$ requirements & Small & 0.96 & 0.93 & 0.97 & \\textbf{0.95} & 0.25 & 0.88\\\\\n",
      "& Medium & 0.96 & 0.97 & 1.00 & \\textbf{0.97} & 0.25 & 1.00\\\\\n",
      "& Large & 0.91 & 0.91 & 0.98 & \\textbf{0.92} & 0.00 & 0.75\\\\\n",
      "\\midrule\n"
     ]
    }
   ],
   "source": [
    "f = \"../results-PAPER_DATASET-19068.pkl-results.pkl\"\n",
    "df_2 = aggregate_paper(pd.read_pickle(f)).reset_index()\n",
    "print(\"$3\\\\times 2$ requirements \" + to_latex_paper(df_2))\n",
    "print(\"\\\\midrule\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "fb3b5bbf",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "$3\\times 16$ requirements & Small & 0.95 & 0.93 & 0.89 & \\textbf{0.91} & 0.12 & 0.50\\\\\n",
      "& Medium & 0.97 & 0.97 & 0.94 & \\textbf{0.95} & 0.25 & 0.88\\\\\n",
      "& Large & 0.92 & 0.91 & 0.94 & \\textbf{0.90} & 0.00 & 0.75\\\\\n"
     ]
    }
   ],
   "source": [
    "f = \"../results-PAPER_DATASET-topk-12558.pkl-results.pkl\"\n",
    "df_2 = aggregate_paper(pd.read_pickle(f)).reset_index()\n",
    "print(\"$3\\\\times 16$ requirements \" + to_latex_paper(df_2))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b97a2f60",
   "metadata": {},
   "source": [
    "### python3 eval_consistency.py --dataset ./dataset-ported/bgp-qlty-reqs-16 ../../models/bgp-train-long-model-epoch5160.pt --random 1 --sampling-mode topk --num-samples 20 --num-shots 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "078241a8",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "$3\\times 16$ requirements & Small & 0.97 & 0.96 & 0.99 & \\textbf{0.96} & 0.25 & 0.88\\\\\n",
      "& Medium & 0.99 & 0.98 & 0.99 & \\textbf{0.98} & 0.38 & 1.00\\\\\n",
      "& Large & 0.96 & 0.96 & 1.00 & \\textbf{0.96} & 0.25 & 0.88\\\\\n"
     ]
    }
   ],
   "source": [
    "f = \"../results-PAPER_DATASET-topk-11423.pkl-results.pkl\"\n",
    "df_2 = aggregate_paper(pd.read_pickle(f)).reset_index()\n",
    "print(\"$3\\\\times 16$ requirements \" + to_latex_paper(df_2))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "66b3831d",
   "metadata": {},
   "source": [
    "### python3 eval_consistency.py --dataset ./dataset-ported/bgp-qlty-reqs-8 ../../models/bgp-train-long-model-epoch5160.pt --random 1 --sampling-mode topk --num-samples 20 --num-shots 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "270f895f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "$3\\times 16$ requirements & Small & 0.99 & 1.00 & 0.98 & \\textbf{0.99} & 0.62 & 1.00\\\\\n",
      "& Medium & 0.99 & 0.91 & 1.00 & \\textbf{0.96} & 0.38 & 0.75\\\\\n",
      "& Large & 0.96 & 0.97 & 0.98 & \\textbf{0.94} & 0.38 & 0.62\\\\\n"
     ]
    }
   ],
   "source": [
    "f = \"../results-PAPER_DATASET-topk-req8-14551.pkl-results.pkl\"\n",
    "df_2 = aggregate_paper(pd.read_pickle(f)).reset_index()\n",
    "print(\"$3\\\\times 16$ requirements \" + to_latex_paper(df_2))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "53f7a43f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "$3\\times 16$ requirements & Small & 0.97 & 0.89 & 0.95 & \\textbf{0.94} & 0.38 & 0.75\\\\\n",
      "& Medium & 0.98 & 0.97 & 0.97 & \\textbf{0.98} & 0.38 & 1.00\\\\\n",
      "& Large & 0.92 & 0.88 & 0.95 & \\textbf{0.91} & 0.12 & 0.50\\\\\n"
     ]
    }
   ],
   "source": [
    "f = \"specification-consistency/results-eval-64-bgp-qlty-reqs-16-4shot-11562.pkl-results.pkl\"\n",
    "df_2 = aggregate_paper(pd.read_pickle(f)).reset_index()\n",
    "print(\"$3\\\\times 16$ requirements \" + to_latex_paper(df_2))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "eab89bea",
   "metadata": {},
   "source": [
    "## Figure on Specification Consistency Across Increasingly Large Topologies and Specifications"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 130,
   "id": "1b92b07c",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "$3\\times 2$ requirements & Small & 0.95 & 0.94 & 1.00 & \\textbf{0.95} & 0.62 & 0.62\\\\\n",
      "& Medium & 0.96 & 1.00 & 0.94 & \\textbf{0.96} & 0.88 & 0.88\\\\\n",
      "& Large & 0.94 & 0.94 & 1.00 & \\textbf{0.93} & 0.62 & 0.62\\\\\n",
      "\\midrule\n",
      "$3\\times 8$ requirements & Small & 0.97 & 0.98 & 0.97 & \\textbf{0.98} & 0.50 & 1.00\\\\\n",
      "& Medium & 0.99 & 0.95 & 1.00 & \\textbf{0.98} & 0.50 & 1.00\\\\\n",
      "& Large & 0.96 & 0.95 & 0.91 & \\textbf{0.94} & 0.38 & 0.88\\\\\n",
      "\\midrule\n",
      "$3\\times 16$ requirements & Small & 0.97 & 0.89 & 0.95 & \\textbf{0.94} & 0.38 & 0.75\\\\\n",
      "& Medium & 0.98 & 0.97 & 0.97 & \\textbf{0.98} & 0.38 & 1.00\\\\\n",
      "& Large & 0.92 & 0.88 & 0.95 & \\textbf{0.91} & 0.12 & 0.50\\\\\n"
     ]
    }
   ],
   "source": [
    "f = \"specification-consistency/results-eval-64-bgp-qlty-reqs-2-4shot-15649.pkl-results.pkl\"\n",
    "df_2 = aggregate_paper(pd.read_pickle(f)).reset_index()\n",
    "print(\"$3\\\\times 2$ requirements \" + to_latex_paper(df_2))\n",
    "print(\"\\\\midrule\")\n",
    "\n",
    "f = \"specification-consistency/results-eval-64-bgp-qlty-reqs-8-4shot-14793.pkl-results.pkl\"\n",
    "df_2 = aggregate_paper(pd.read_pickle(f)).reset_index()\n",
    "print(\"$3\\\\times 8$ requirements \" + to_latex_paper(df_2))\n",
    "print(\"\\\\midrule\")\n",
    "\n",
    "f = \"specification-consistency/results-eval-64-bgp-qlty-reqs-16-4shot-11562.pkl-results.pkl\"\n",
    "df_2 = aggregate_paper(pd.read_pickle(f)).reset_index()\n",
    "print(\"$3\\\\times 16$ requirements \" + to_latex_paper(df_2))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 132,
   "id": "1e95bac9",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "$3\\times 2$ requirements & Small & 0.97 & 0.94 & 1.00 & \\textbf{0.96} & 0.62 & 0.62\\\\\n",
      "& Medium & 1.00 & 1.00 & 0.94 & \\textbf{0.98} & 0.88 & 0.88\\\\\n",
      "& Large & 0.93 & 1.00 & 1.00 & \\textbf{0.96} & 0.62 & 0.62\\\\\n",
      "\\midrule\n",
      "$3\\times 8$ requirements & Small & 0.99 & 1.00 & 0.97 & \\textbf{0.99} & 0.75 & 1.00\\\\\n",
      "& Medium & 0.99 & 0.95 & 1.00 & \\textbf{0.98} & 0.62 & 1.00\\\\\n",
      "& Large & 0.98 & 0.97 & 0.95 & \\textbf{0.97} & 0.38 & 1.00\\\\\n",
      "\\midrule\n",
      "$3\\times 16$ requirements & Small & 0.99 & 0.92 & 0.95 & \\textbf{0.96} & 0.38 & 0.88\\\\\n",
      "& Medium & 0.99 & 0.96 & 0.97 & \\textbf{0.98} & 0.50 & 1.00\\\\\n",
      "& Large & 0.94 & 0.93 & 0.97 & \\textbf{0.94} & 0.12 & 0.88\\\\\n"
     ]
    }
   ],
   "source": [
    "f = \"specification-consistency/results-eval-64-bgp-qlty-reqs-2-4shot-15649.pkl-results.pkl\"\n",
    "f = \"../results-eval-bgp-2-11316.pkl-results.pkl\"\n",
    "df_2 = aggregate_paper(pd.read_pickle(f)).reset_index()\n",
    "print(\"$3\\\\times 2$ requirements \" + to_latex_paper(df_2))\n",
    "print(\"\\\\midrule\")\n",
    "\n",
    "f = \"specification-consistency/results-eval-64-bgp-qlty-reqs-8-4shot-14793.pkl-results.pkl\"\n",
    "f = \"../results-eval-bgp-8-16574.pkl-results.pkl\"\n",
    "df_2 = aggregate_paper(pd.read_pickle(f)).reset_index()\n",
    "print(\"$3\\\\times 8$ requirements \" + to_latex_paper(df_2))\n",
    "print(\"\\\\midrule\")\n",
    "\n",
    "f = \"specification-consistency/results-eval-64-bgp-qlty-reqs-16-4shot-11562.pkl-results.pkl\"\n",
    "f = \"../results-eval-bgp-16-13503.pkl-results.pkl\"\n",
    "df_2 = aggregate_paper(pd.read_pickle(f)).reset_index()\n",
    "print(\"$3\\\\times 16$ requirements \" + to_latex_paper(df_2))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 141,
   "id": "d506ba54",
   "metadata": {},
   "outputs": [],
   "source": [
    "f = \"../results-eval-bgp-16-13503.pkl-results.pkl\"\n",
    "df = pd.read_pickle(f)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 142,
   "id": "fef4bf37",
   "metadata": {},
   "outputs": [],
   "source": [
    "def collect_results_up_to(df, i):\n",
    "    res = {}\n",
    "    counts = {}\n",
    "    for row in df.iloc:\n",
    "        k = row[\"sample_id\"]\n",
    "        if not row[\"sample_id\"] 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",
    "    \n",
    "    for id in ids:\n",
    "        d = \"\"\n",
    "        if id < 8: d = \"Small\"\n",
    "        elif id < 16: d = \"Medium\"\n",
    "        else: d = \"Large\"\n",
    "        dataset_avg[d] += res[id]\n",
    "        dataset_cnt[d] += 1\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": 143,
   "id": "692a0456",
   "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": 145,
   "id": "9cd02396",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "No artists with labels found to put in legend.  Note that artists whose label start with an underscore are ignored when legend() is called with no argument.\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x7f16578f6790>"
      ]
     },
     "execution_count": 145,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "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)\n",
    "plt.plot(x, y_medium)\n",
    "plt.plot(x, y_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(\"Relative 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": "4300d642",
   "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": "markdown",
   "id": "c6fb0096",
   "metadata": {},
   "source": [
    "## Figure on Baseline/Oneshot/Multi-Shot Comparison"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "e6f4e108",
   "metadata": {},
   "outputs": [],
   "source": [
    "def to_latex_paper_baseline(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[\"overall\"],\n",
    "        ]\n",
    "        lines.append(\"& \" + \" & \".join(columns) + \"\\\\\\\\\")\n",
    "    return \"\\n\".join(lines)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "d415a08e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Random\n",
      "& Small & 0.83\\\\\n",
      "& Medium & 0.88\\\\\n",
      "& Large & 0.78\\\\\n",
      "1-Shot\n",
      "& Small & 0.92\\\\\n",
      "& Medium & 0.95\\\\\n",
      "& Large & 0.88\\\\\n",
      "4-Shot\n",
      "& Small & 0.94\\\\\n",
      "& Medium & 0.98\\\\\n",
      "& Large & 0.91\\\\\n"
     ]
    }
   ],
   "source": [
    "f = \"multishot-sampling/results-eval-64-bgp-qlty-reqs-16-random-13819.pkl-results.pkl\"\n",
    "df_2 = aggregate_paper(pd.read_pickle(f)).reset_index()\n",
    "print(\"Random\")\n",
    "print(to_latex_paper_baseline(df_2))\n",
    "\n",
    "print(\"1-Shot\")\n",
    "f = \"multishot-sampling/results-eval-64-bgp-qlty-reqs-16-oneshot-14551.pkl-results.pkl\"\n",
    "df_2 = aggregate_paper(pd.read_pickle(f)).reset_index()\n",
    "print(to_latex_paper_baseline(df_2))\n",
    "\n",
    "print(\"4-Shot\")\n",
    "f = \"specification-consistency/results-eval-64-bgp-qlty-reqs-16-4shot-11562.pkl-results.pkl\"\n",
    "df_2 = aggregate_paper(pd.read_pickle(f)).reset_index()\n",
    "print(to_latex_paper_baseline(df_2))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "56443fec",
   "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.8.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
