{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "63237658-7728-4244-a53a-6998a259613e",
   "metadata": {},
   "source": [
    "# Imports"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "outputs": [],
   "source": [
    "import sys\n",
    "base = \"/research/2025_mip/\"\n",
    "sys.path.append(base)\n",
    "sys.path.append(os.path.join(base, 'forge'))\n",
    "\n",
    "\n",
    "from forge.forge import Forge\n",
    "from forge.utils import *\n",
    "import subprocess\n",
    "\n",
    "try:\n",
    "    from gurobi_onboarder import init_gurobi\n",
    "    gurobi_venv, GUROBI_FOUND = init_gurobi.initialize_gurobi()\n",
    "except:\n",
    "    gurobi_venv = gp.Env(empty=True)"
   ],
   "metadata": {
    "collapsed": false
   },
   "id": "806c1a86-038c-485e-8760-d393f1ac485f"
  },
  {
   "cell_type": "markdown",
   "id": "ff904e07-d8af-48e4-bb80-a1de8aafd057",
   "metadata": {},
   "source": [
    "# Read Temp Files - Forge Experiments"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8fac89e0-af40-4246-89cd-04cee284d207",
   "metadata": {},
   "outputs": [],
   "source": [
    "max_time = 3700\n",
    "\n",
    "# Options: SC, CA, GISP, MVC\n",
    "p = 'SC'\n",
    "\n",
    "# Options: medium, hard, very-hard\n",
    "difficulty = 'medium' \n",
    "\n",
    "run_time_ws = []\n",
    "run_time_og = []\n",
    "node_count_ws = []\n",
    "node_count_og = []\n",
    "pg_ws = []\n",
    "pg_og = []\n",
    "\n",
    "for x in [x for x in os.listdir(os.path.join(base, 'data/temp')) if p in x and difficulty in x and 'cut' not in x]:\n",
    "    \n",
    "    with open(os.path.join(base, 'data/temp/', x), 'rb') as file:\n",
    "        run_time_ws_, run_time_og_, node_count_ws_, node_count_og_, pg_ws_, pg_og_ =  pkl.load(file)    \n",
    "        run_time_ws += run_time_ws_\n",
    "        run_time_og += run_time_og_\n",
    "        node_count_ws += node_count_ws_\n",
    "        node_count_og += node_count_og_\n",
    "        pg_ws += pg_ws_\n",
    "        pg_og += pg_og_\n",
    "\n",
    "\n",
    "pg_ws = [x for x in pg_ws if len(x) > 0]\n",
    "pg_og = [x for x in pg_og if len(x) > 0]\n",
    "cut_pg = clean_pg_dict(in_dict = pg_ws, timesteps = max_time)\n",
    "gurobi_pg = clean_pg_dict(in_dict = pg_og, timesteps = max_time)\n",
    "\n",
    "td_cut = {x : [] for x in range(max_time)}\n",
    "td_og = {x : [] for x in range(max_time)}\n",
    "\n",
    "for inst in cut_pg:\n",
    "    for t in range(max_time):\n",
    "        td_cut[t].append(0)\n",
    "    for t in inst:  \n",
    "        td_cut[int(t)][-1] = inst[t]\n",
    "\n",
    "for inst in gurobi_pg:\n",
    "    for t in range(max_time):\n",
    "        td_og[t].append(0)\n",
    "    for t in inst:  \n",
    "        td_og[int(t)][-1] = inst[t]\n",
    "\n",
    "gurobi_time = run_time_og\n",
    "cut_time = run_time_ws\n",
    "\n",
    "mean_cut = np.array([np.mean(td_cut[x]) for x in td_cut])\n",
    "std_cut = np.array([np.std(td_cut[x]) for x in td_cut]) / np.sqrt(len(td_cut[0]))\n",
    "\n",
    "mean_og = np.array([np.mean(td_og[x]) for x in td_og])\n",
    "std_og = np.array([np.std(td_og[x]) for x in td_og]) / np.sqrt(len(td_og[0]))\n",
    "\n",
    "# Set the right end-point to make the plot cleaner\n",
    "start_point = 20\n",
    "end_point = 900\n",
    "\n",
    "plt.figure(dpi = 300)\n",
    "plt.plot(list(range(start_point, end_point)), mean_cut[start_point:end_point], label = 'Warm-Start', color = (0, 1, 0, 0.8),)\n",
    "# plt.fill_between(list(range(start_point, max_time)), (mean_cut + std_cut)[start_point:], (mean_cut - std_cut)[start_point:], color = (0, 1, 0, 0.5))\n",
    "\n",
    "plt.plot(list(range(start_point, end_point)), mean_og[start_point:end_point], label = 'Gurobi', color = (1, 0, 0, 0.8),)\n",
    "# plt.fill_between(list(range(start_point, max_time)), (mean_og + std_og)[start_point:], (mean_og - std_og)[start_point:], color = (1, 0, 0, 0.5))\n",
    "\n",
    "plt.yscale('log')\n",
    "# plt.title(p + ' - ' + difficulty + ' | ' + str(len(cut_pg)) + ' Instances \\nMean Speedup : ' + str(np.round(100 * (1 - np.mean(np.array(cut_time) / np.array(gurobi_time))), 3)) + '% | Node Count Reduction : ' + str(np.round(100 * (1 - np.mean(np.array(node_count_ws) / np.array(node_count_og))), 3)) + '%\\nMedian Speedup : ' + str(np.round(100 * (1 - np.median(np.array(cut_time) / np.array(gurobi_time))), 3)) + '%')\n",
    "# plt.title('(a) ' + p + ' - ' + difficulty)\n",
    "plt.legend(fontsize = 15)\n",
    "plt.xlabel('Time (s)', fontsize = 20)\n",
    "plt.ylabel('Primal Gap (log scale)', fontsize = 20)\n",
    "plt.tick_params(axis = 'both', labelsize = 15)\n",
    "plt.show()\n",
    "print ()\n",
    "print (\"Average Primal Gap Gain: \", np.nanmean(1 - (mean_cut[start_point:end_point] / mean_og[start_point:end_point]))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2769854a-750c-4642-b50f-82a5018b2dce",
   "metadata": {
    "scrolled": true
   },
   "source": [
    "# Read Temp Files - PS-Gurobi + Forge Experiments"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3b96a93c-9439-4f88-9496-22489f133928",
   "metadata": {},
   "outputs": [],
   "source": [
    "p_type = 'CA'\n",
    "max_time = 3700\n",
    "pg_ps = []\n",
    "run_time_ps = []\n",
    "for x in [x for x in os.listdir(os.path.join('xx', 'forge_Predict-and-Search_MILP_method/res_og')) if p_type in x]:\n",
    "    \n",
    "    with open(os.path.join('xx', 'forge_Predict-and-Search_MILP_method/res_og', x), 'rb') as file:\n",
    "        run_time_ps_, pg_ps_ =  pkl.load(file)    \n",
    "        pg_ps.append(pg_ps_)\n",
    "        run_time_ps.append(run_time_ps_)\n",
    "\n",
    "pg_ps = [x for x in pg_ps if x != {}]\n",
    "run_time_ps = [x for x in run_time_ps if x > 1]\n",
    "\n",
    "ps_pg = clean_pg_dict(in_dict = pg_ps, timesteps = max_time)\n",
    "\n",
    "td_ps = {x : [] for x in range(max_time)}\n",
    "\n",
    "for inst in ps_pg:\n",
    "    for t in range(max_time):\n",
    "        td_ps[t].append(0)\n",
    "    for t in inst:  \n",
    "        td_ps[int(t)][-1] = inst[t]\n",
    "\n",
    "ps_time = run_time_ps\n",
    "\n",
    "mean_ps_og = np.array([np.mean(td_ps[x]) for x in td_ps])\n",
    "\n",
    "pg_ps = []\n",
    "run_time_ps = []\n",
    "for x in [x for x in os.listdir(os.path.join('xx', 'forge_Predict-and-Search_MILP_method/res_forge')) if p_type in x]:\n",
    "    \n",
    "    with open(os.path.join('xx', 'forge_Predict-and-Search_MILP_method/res_forge', x), 'rb') as file:\n",
    "        run_time_ps_, pg_ps_ =  pkl.load(file)    \n",
    "        pg_ps.append(pg_ps_)\n",
    "        run_time_ps.append(run_time_ps_)\n",
    "\n",
    "pg_ps = [x for x in pg_ps if x != {}]\n",
    "run_time_ps = [x for x in run_time_ps if x > 1]\n",
    "\n",
    "ps_pg = clean_pg_dict(in_dict = pg_ps, timesteps = max_time)\n",
    "\n",
    "td_ps = {x : [] for x in range(max_time)}\n",
    "\n",
    "for inst in ps_pg:\n",
    "    for t in range(max_time):\n",
    "        td_ps[t].append(0)\n",
    "    for t in inst:  \n",
    "        td_ps[int(t)][-1] = inst[t]\n",
    "\n",
    "ps_time = run_time_ps\n",
    "\n",
    "mean_ps_forge = np.array([np.mean(td_ps[x]) for x in td_ps])\n",
    "\n",
    "pg_ps = []\n",
    "run_time_ps = []\n",
    "for x in [x for x in os.listdir(os.path.join('xx', 'forge_Predict-and-Search_MILP_method/res_rand')) if p_type in x]:\n",
    "    \n",
    "    with open(os.path.join('xx', 'forge_Predict-and-Search_MILP_method/res_rand', x), 'rb') as file:\n",
    "        run_time_ps_, pg_ps_ =  pkl.load(file)    \n",
    "        pg_ps.append(pg_ps_)\n",
    "        run_time_ps.append(run_time_ps_)\n",
    "\n",
    "pg_ps = [x for x in pg_ps if x != {}]\n",
    "run_time_ps = [x for x in run_time_ps if x > 1]\n",
    "\n",
    "ps_pg = clean_pg_dict(in_dict = pg_ps, timesteps = max_time)\n",
    "\n",
    "td_ps = {x : [] for x in range(max_time)}\n",
    "\n",
    "for inst in ps_pg:\n",
    "    for t in range(max_time):\n",
    "        td_ps[t].append(0)\n",
    "    for t in inst:  \n",
    "        td_ps[int(t)][-1] = inst[t]\n",
    "\n",
    "ps_time = run_time_ps\n",
    "\n",
    "mean_ps_rand = np.array([np.mean(td_ps[x]) for x in td_ps])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "da7b6ebd-f7f4-493c-864c-83fb5b824227",
   "metadata": {},
   "outputs": [],
   "source": [
    "start_point = 20\n",
    "end_point = 3600\n",
    "\n",
    "plt.figure(dpi = 300)\n",
    "\n",
    "plt.plot(list(range(start_point, end_point)), mean_ps_og[start_point:end_point], label = 'PS Gurobi', color = (0, 1, 1, 0.8),)\n",
    "plt.plot(list(range(start_point, end_point)), mean_ps_forge[start_point:end_point], label = 'PS Gurobi + FORGE', color = (0, 0, 1, 0.8),)\n",
    "plt.plot(list(range(start_point, end_point)), mean_ps_rand[start_point:end_point], label = 'PS Gurobi + Random', color = (1, 0, 1, 0.8),)\n",
    "\n",
    "plt.yscale('log')\n",
    "# plt.title(p + ' - ' + difficulty + ' | ' + str(len(cut_pg)) + ' Instances \\nMean Speedup : ' + str(np.round(100 * (1 - np.mean(np.array(cut_time) / np.array(gurobi_time))), 3)) + '% | Node Count Reduction : ' + str(np.round(100 * (1 - np.mean(np.array(node_count_ws) / np.array(node_count_og))), 3)) + '%\\nMedian Speedup : ' + str(np.round(100 * (1 - np.median(np.array(cut_time) / np.array(gurobi_time))), 3)) + '%')\n",
    "# plt.title('(a) ' + p + ' - ' + difficulty)\n",
    "plt.legend(fontsize = 15)\n",
    "plt.xlabel('Time (s)', fontsize = 20)\n",
    "plt.ylabel('Primal Gap (log scale)', fontsize = 20)\n",
    "plt.tick_params(axis = 'both', labelsize = 15)\n",
    "\n",
    "plt.show()\n",
    "print ()\n",
    "\n",
    "print (\"Average Primal Gap Gain: \", np.nanmean(1 - (mean_ps_forge[start_point:end_point] / mean_ps_og[start_point:end_point])))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "20047fb9-7e69-48d9-be31-aa28c014f0b6",
   "metadata": {},
   "source": [
    "# Read Temp Files = Forge Predicted Cut Results"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4243ab79-c0f9-4d42-b6a3-e71d9ff23e4a",
   "metadata": {},
   "outputs": [],
   "source": [
    "res = {}\n",
    "for f in os.listdir('../../data/log'):\n",
    "    if 'cut' in f: \n",
    "        with open('../../data/log/' + f, 'r') as file: \n",
    "            d = file.read()\n",
    "            pred = [float(x.split('Cut Ratio : ')[1].split('\\n')[0]) for x in d.split(' ----') if 'Cut ' in x]\n",
    "            true = [float(x.split('True Ratio : ')[1].split('\\n')[0]) for x in d.split(' ----') if 'True ' in x]\n",
    "            true = [1 / x if x > 2 else x for x in true]\n",
    "            res[f] = {}\n",
    "            res[f]['mae'] = np.mean(np.abs(np.array(true) - np.array(pred)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "87d5c179-e7af-4453-9397-c2b45703bb5e",
   "metadata": {},
   "outputs": [],
   "source": [
    "res"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "gpu",
   "language": "python",
   "name": "gpu"
  },
  "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.11"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
