{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "d48825af-9f6a-4108-8198-ba7fb36cd39e",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "import geopandas as gpd\n",
    "import numpy as np\n",
    "import gurobipy as gp\n",
    "from gurobipy import GRB\n",
    "import networkx as nx\n",
    "import itertools\n",
    "import sys\n",
    "sys.path.append('..')\n",
    "from security_game.target import Target\n",
    "from security_game.green_security_game import GreenSecurityGame\n",
    "from security_game.infra_security_game import InfraSecurityGame\n",
    "\n",
    "from solvers.mip import mip\n",
    "from solvers.nash import nash\n",
    "from solvers.double_oracle import double_oracle\n",
    "from solvers.double_oracle_sf import double_oracle_sf\n",
    "from solvers.no_regret import regret_matching\n",
    "import time\n",
    "from matplotlib import pyplot as plt\n",
    "from collections import defaultdict, Counter"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6de6ec39-621b-47c6-92c9-2d00070519b3",
   "metadata": {},
   "source": [
    "# GSG"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "967b76b6-93ec-440e-b67d-5aa9582816f9",
   "metadata": {},
   "source": [
    "## NFG"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "74564a76-0afc-46a8-ab98-f247e3a13d9d",
   "metadata": {},
   "outputs": [],
   "source": [
    "df = pd.read_csv(\"lobeke.csv\")\n",
    "df.dropna(inplace=True)\n",
    "\n",
    "lat_min, lon_min = 2.0530, 15.8790\n",
    "lat_max, lon_max = 2.2837, 16.2038\n",
    "\n",
    "coordinate_rectangle = [lat_min, lat_max, lon_min, lon_max]\n",
    "\n",
    "schedule_form_kwargs = {\n",
    "    \"schedule_form\": False,\n",
    "    \"simple\": False,\n",
    "    \"attacker_penalty_factor\": 5,\n",
    "    \"defender_penalty_factor\": 5,\n",
    "}\n",
    "\n",
    "general_sum_kwargs = {\n",
    "    \"general_sum\": False,\n",
    "    \"attacker_feature_value\":  42, \n",
    "    \"defender_feature_value\": 69, \n",
    "    \"defender_step_cost\": 32.5, \n",
    "}\n",
    "\n",
    "boulou_camp = (2.2,15.9)\n",
    "# lobeke_camp = (2.25,15.75)\n",
    "kabo_djembe = (2.0532352380408088, 16.085709866529694)\n",
    "bomassa = (2.2037280296158355, 16.187056364164913)\n",
    "inner_post = (2.2,15.98)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "47cc8ed6-1759-4d8b-88d6-b8f64ddcf212",
   "metadata": {},
   "outputs": [],
   "source": [
    "gsg = GreenSecurityGame(df, coordinate_rectangle, \"centroid\", num_clusters=10, num_rows=7, num_columns=7)\n",
    "gsg.generate(num_attackers=1, \n",
    "             num_defenders=1, \n",
    "             home_base_assignments=[(kabo_djembe, bomassa, inner_post)], \n",
    "             num_timesteps=10, \n",
    "             generate_utility_matrix=True, \n",
    "             defense_time_threshold=2, \n",
    "             generate_actions=False, \n",
    "             force_return=False, \n",
    "             general_sum=False, \n",
    "             **schedule_form_kwargs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "f1a3b4e8-baa2-4526-9886-f4704665ee09",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(184647, 10, 1)"
      ]
     },
     "execution_count": 17,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "gsg.defender_actions.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "e24680e6-3053-4d91-8a3c-e4eb38fd8556",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(6, -0.41601780572368807)"
      ]
     },
     "execution_count": 18,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "start = time.time()\n",
    "nD_a, nD_d, nu = nash(gsg.utility_matrix)\n",
    "end = time.time()\n",
    "nruntime = end-start\n",
    "nsupport = sum([1 for p in nD_d if p!=0])\n",
    "nsupport,nu"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "b699950d-bd5d-44d5-81c7-70fe7b2758fe",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "starting i=1 at time 1746402172.0991766\n",
      "finished i=1 in 16.50993323326111 seconds with u=-0.7272727271\n",
      "starting i=2 at time 1746402188.6111145\n",
      "finished i=2 in 206.0364978313446 seconds with u=-0.41601772583126534\n",
      "starting i=3 at time 1746402394.6487172\n",
      "finished i=3 in 235.48542141914368 seconds with u=-0.3671328143367632\n",
      "starting i=4 at time 1746402630.1361501\n",
      "finished i=4 in 28.91105890274048 seconds with u=-0.3391608187873172\n",
      "starting i=5 at time 1746402659.0556812\n",
      "finished i=5 in 117.26184487342834 seconds with u=-0.328187365472519\n",
      "starting i=6 at time 1746402776.3185306\n",
      "finished i=6 in 125.43130922317505 seconds with u=-0.3238241838591864\n",
      "starting i=7 at time 1746402901.75185\n",
      "finished i=7 in 7.428220987319946 seconds with u=-0.32232303341713453\n"
     ]
    }
   ],
   "source": [
    "mip_us = []\n",
    "mip_supports = []\n",
    "mip_runtimes = []\n",
    "\n",
    "for i in range(1,10):\n",
    "    start = time.time()\n",
    "    print(f\"starting i={i} at time {start}\")\n",
    "    mu, mD_d = mip(gsg.utility_matrix,i)\n",
    "    end = time.time()\n",
    "    print(f\"finished i={i} in {end-start} seconds with u={mu}\")\n",
    "    msupport = sum([1 for p in mD_d if p!=0])\n",
    "    mip_us.append(mu)\n",
    "    mip_supports.append(i)\n",
    "    mip_runtimes.append(end-start)\n",
    "    if abs(mu-nu) <= 1e-8:\n",
    "        break"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "78b67aeb-aace-4349-9bb9-8401a90fa425",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>num_timesteps</th>\n",
       "      <th>num_attackers</th>\n",
       "      <th>num_defenders</th>\n",
       "      <th>num_clusters</th>\n",
       "      <th>dims</th>\n",
       "      <th>defense_time_threshold</th>\n",
       "      <th>force_return</th>\n",
       "      <th>num_defender_actions</th>\n",
       "      <th>nash_value</th>\n",
       "      <th>nash_support</th>\n",
       "      <th>nash_runtime</th>\n",
       "      <th>mip_value</th>\n",
       "      <th>mip_support</th>\n",
       "      <th>mip_runtime</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>9</td>\n",
       "      <td>1</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.322323</td>\n",
       "      <td>8</td>\n",
       "      <td>4.064473</td>\n",
       "      <td>-0.727273</td>\n",
       "      <td>1</td>\n",
       "      <td>16.509933</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>9</td>\n",
       "      <td>1</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.322323</td>\n",
       "      <td>8</td>\n",
       "      <td>4.064473</td>\n",
       "      <td>-0.416018</td>\n",
       "      <td>2</td>\n",
       "      <td>206.036498</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>9</td>\n",
       "      <td>1</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.322323</td>\n",
       "      <td>8</td>\n",
       "      <td>4.064473</td>\n",
       "      <td>-0.367133</td>\n",
       "      <td>3</td>\n",
       "      <td>235.485421</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>9</td>\n",
       "      <td>1</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.322323</td>\n",
       "      <td>8</td>\n",
       "      <td>4.064473</td>\n",
       "      <td>-0.339161</td>\n",
       "      <td>4</td>\n",
       "      <td>28.911059</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>9</td>\n",
       "      <td>1</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.322323</td>\n",
       "      <td>8</td>\n",
       "      <td>4.064473</td>\n",
       "      <td>-0.328187</td>\n",
       "      <td>5</td>\n",
       "      <td>117.261845</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>9</td>\n",
       "      <td>1</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.322323</td>\n",
       "      <td>8</td>\n",
       "      <td>4.064473</td>\n",
       "      <td>-0.323824</td>\n",
       "      <td>6</td>\n",
       "      <td>125.431309</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6</th>\n",
       "      <td>9</td>\n",
       "      <td>1</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.322323</td>\n",
       "      <td>8</td>\n",
       "      <td>4.064473</td>\n",
       "      <td>-0.322323</td>\n",
       "      <td>7</td>\n",
       "      <td>7.428221</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "   num_timesteps  num_attackers  num_defenders  num_clusters  dims  \\\n",
       "0              9              1              1            10     7   \n",
       "1              9              1              1            10     7   \n",
       "2              9              1              1            10     7   \n",
       "3              9              1              1            10     7   \n",
       "4              9              1              1            10     7   \n",
       "5              9              1              1            10     7   \n",
       "6              9              1              1            10     7   \n",
       "\n",
       "   defense_time_threshold  force_return  num_defender_actions  nash_value  \\\n",
       "0                       1         False                 41479   -0.322323   \n",
       "1                       1         False                 41479   -0.322323   \n",
       "2                       1         False                 41479   -0.322323   \n",
       "3                       1         False                 41479   -0.322323   \n",
       "4                       1         False                 41479   -0.322323   \n",
       "5                       1         False                 41479   -0.322323   \n",
       "6                       1         False                 41479   -0.322323   \n",
       "\n",
       "   nash_support  nash_runtime  mip_value  mip_support  mip_runtime  \n",
       "0             8      4.064473  -0.727273            1    16.509933  \n",
       "1             8      4.064473  -0.416018            2   206.036498  \n",
       "2             8      4.064473  -0.367133            3   235.485421  \n",
       "3             8      4.064473  -0.339161            4    28.911059  \n",
       "4             8      4.064473  -0.328187            5   117.261845  \n",
       "5             8      4.064473  -0.323824            6   125.431309  \n",
       "6             8      4.064473  -0.322323            7     7.428221  "
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df = pd.DataFrame({\n",
    "    \"num_timesteps\":[9 for i in range(len(mip_supports))],\n",
    "    \"num_attackers\":[1 for i in range(len(mip_supports))],\n",
    "    \"num_defenders\":[1 for i in range(len(mip_supports))],\n",
    "    \"num_clusters\":[10 for i in range(len(mip_supports))],\n",
    "    \"dims\":[7 for i in range(len(mip_supports))],\n",
    "    \"defense_time_threshold\":[1 for i in range(len(mip_supports))],\n",
    "    \"force_return\":[False for i in range(len(mip_supports))],\n",
    "    \"num_defender_actions\": [len(gsg.defender_actions) for i in range(len(mip_supports))],\n",
    "    \"nash_value\":[nu for i in range(len(mip_supports))],\n",
    "    \"nash_support\":[nsupport for i in range(len(mip_supports))],\n",
    "    \"nash_runtime\":[nruntime for i in range(len(mip_supports))],\n",
    "    \"mip_value\":mip_us,\n",
    "    \"mip_support\":mip_supports,\n",
    "    \"mip_runtime\":mip_runtimes,\n",
    "})\n",
    "df"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "a0cc970c-84e4-4aff-acfc-74252dc1b166",
   "metadata": {},
   "outputs": [],
   "source": [
    "df.to_csv(\"GSG_NFG_T9_1A_SPARSITY.csv\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8f1687f1-65f8-421d-9643-4d7b8649f975",
   "metadata": {},
   "source": [
    "## SF"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 86,
   "id": "169853ad-ef5c-488c-b58f-c0245bfc9225",
   "metadata": {},
   "outputs": [],
   "source": [
    "df = pd.read_csv(\"lobeke.csv\")\n",
    "df.dropna(inplace=True)\n",
    "\n",
    "lat_min, lon_min = 2.0530, 15.8790\n",
    "lat_max, lon_max = 2.2837, 16.2038\n",
    "\n",
    "coordinate_rectangle = [lat_min, lat_max, lon_min, lon_max]\n",
    "\n",
    "schedule_form_kwargs = {\n",
    "    \"schedule_form\": True,\n",
    "    \"simple\": False,\n",
    "    \"attacker_penalty_factor\": 5,\n",
    "    \"defender_penalty_factor\": 5,\n",
    "}\n",
    "\n",
    "general_sum_kwargs = {\n",
    "    \"general_sum\": False,\n",
    "    \"attacker_feature_value\":  42, \n",
    "    \"defender_feature_value\": 69, \n",
    "    \"defender_step_cost\": 32.5, \n",
    "}\n",
    "\n",
    "boulou_camp = (2.2,15.9)\n",
    "# lobeke_camp = (2.25,15.75)\n",
    "kabo_djembe = (2.0532352380408088, 16.085709866529694)\n",
    "bomassa = (2.2037280296158355, 16.187056364164913)\n",
    "inner_post = (2.2,15.98)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 87,
   "id": "f6d927c3-2833-46a6-ba2c-a150bfa94aac",
   "metadata": {},
   "outputs": [],
   "source": [
    "gsg = GreenSecurityGame(df, coordinate_rectangle, \"centroid\", num_clusters=10, num_rows=7, num_columns=7)\n",
    "gsg.generate(num_attackers=1, \n",
    "             num_defenders=2, \n",
    "             home_base_assignments=[(kabo_djembe, bomassa, inner_post),(kabo_djembe, bomassa, inner_post)], \n",
    "             num_timesteps=8, \n",
    "             generate_utility_matrix=True, \n",
    "             defense_time_threshold=2, \n",
    "             generate_actions=False, \n",
    "             force_return=True, \n",
    "             general_sum=False, \n",
    "             **schedule_form_kwargs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 88,
   "id": "e24d27d8-2059-4d08-b7a3-86ec0e050daf",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(841, 10)"
      ]
     },
     "execution_count": 88,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "gsg.schedule_form_dict[\"defender_utility_matrix\"].shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 89,
   "id": "2c5df782-671a-4fc3-b769-24e051d14ce5",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(6, -0.3465487110914404)"
      ]
     },
     "execution_count": 89,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "start = time.time()\n",
    "nD_a, nD_d, nu = nash(gsg.schedule_form_dict[\"defender_utility_matrix\"])\n",
    "end = time.time()\n",
    "nruntime = end-start\n",
    "nsupport = sum([1 for p in nD_d if p!=0])\n",
    "nsupport,nu"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 83,
   "id": "7a9e63ed-8b7c-4a36-9763-fe9664499a25",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "starting i=1 at time 1746417466.6778293\n",
      "finished i=1 in 0.5499131679534912 seconds with u=-0.4755244736946514\n",
      "starting i=2 at time 1746417467.2277424\n",
      "finished i=2 in 0.582956075668335 seconds with u=-0.36144987069523626\n",
      "starting i=3 at time 1746417467.8106985\n",
      "finished i=3 in 1.7658820152282715 seconds with u=-0.32225062576166796\n",
      "starting i=4 at time 1746417469.5776532\n",
      "finished i=4 in 1.7085793018341064 seconds with u=-0.3039164393667656\n",
      "starting i=5 at time 1746417471.2862325\n",
      "finished i=5 in 1.8729972839355469 seconds with u=-0.2954322207464402\n",
      "starting i=6 at time 1746417473.1592298\n",
      "finished i=6 in 1.427549123764038 seconds with u=-0.28959643550940306\n",
      "starting i=7 at time 1746417474.5867789\n",
      "finished i=7 in 0.29776930809020996 seconds with u=-0.2892405920467003\n"
     ]
    }
   ],
   "source": [
    "mip_us = []\n",
    "mip_supports = []\n",
    "mip_runtimes = []\n",
    "\n",
    "for i in range(1,10):\n",
    "    start = time.time()\n",
    "    print(f\"starting i={i} at time {start}\")\n",
    "    mu, mD_d = mip(gsg.schedule_form_dict[\"defender_utility_matrix\"],i)\n",
    "    end = time.time()\n",
    "    print(f\"finished i={i} in {end-start} seconds with u={mu}\")\n",
    "    msupport = sum([1 for p in mD_d if p!=0])\n",
    "    mip_us.append(mu)\n",
    "    mip_supports.append(i)\n",
    "    mip_runtimes.append(end-start)\n",
    "    if abs(mu-nu) <= 1e-12:\n",
    "        break"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 84,
   "id": "9a7eea15-b40f-46fd-831a-b2eabc3068f9",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>num_timesteps</th>\n",
       "      <th>num_attackers</th>\n",
       "      <th>num_defenders</th>\n",
       "      <th>num_clusters</th>\n",
       "      <th>dims</th>\n",
       "      <th>defense_time_threshold</th>\n",
       "      <th>force_return</th>\n",
       "      <th>num_defender_actions</th>\n",
       "      <th>nash_value</th>\n",
       "      <th>nash_support</th>\n",
       "      <th>nash_runtime</th>\n",
       "      <th>mip_value</th>\n",
       "      <th>mip_support</th>\n",
       "      <th>mip_runtime</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>10</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>2</td>\n",
       "      <td>True</td>\n",
       "      <td>2401</td>\n",
       "      <td>-0.289241</td>\n",
       "      <td>8</td>\n",
       "      <td>0.286761</td>\n",
       "      <td>-0.475524</td>\n",
       "      <td>1</td>\n",
       "      <td>0.549913</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>10</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>2</td>\n",
       "      <td>True</td>\n",
       "      <td>2401</td>\n",
       "      <td>-0.289241</td>\n",
       "      <td>8</td>\n",
       "      <td>0.286761</td>\n",
       "      <td>-0.361450</td>\n",
       "      <td>2</td>\n",
       "      <td>0.582956</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>10</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>2</td>\n",
       "      <td>True</td>\n",
       "      <td>2401</td>\n",
       "      <td>-0.289241</td>\n",
       "      <td>8</td>\n",
       "      <td>0.286761</td>\n",
       "      <td>-0.322251</td>\n",
       "      <td>3</td>\n",
       "      <td>1.765882</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>10</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>2</td>\n",
       "      <td>True</td>\n",
       "      <td>2401</td>\n",
       "      <td>-0.289241</td>\n",
       "      <td>8</td>\n",
       "      <td>0.286761</td>\n",
       "      <td>-0.303916</td>\n",
       "      <td>4</td>\n",
       "      <td>1.708579</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>10</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>2</td>\n",
       "      <td>True</td>\n",
       "      <td>2401</td>\n",
       "      <td>-0.289241</td>\n",
       "      <td>8</td>\n",
       "      <td>0.286761</td>\n",
       "      <td>-0.295432</td>\n",
       "      <td>5</td>\n",
       "      <td>1.872997</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>10</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>2</td>\n",
       "      <td>True</td>\n",
       "      <td>2401</td>\n",
       "      <td>-0.289241</td>\n",
       "      <td>8</td>\n",
       "      <td>0.286761</td>\n",
       "      <td>-0.289596</td>\n",
       "      <td>6</td>\n",
       "      <td>1.427549</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6</th>\n",
       "      <td>10</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>2</td>\n",
       "      <td>True</td>\n",
       "      <td>2401</td>\n",
       "      <td>-0.289241</td>\n",
       "      <td>8</td>\n",
       "      <td>0.286761</td>\n",
       "      <td>-0.289241</td>\n",
       "      <td>7</td>\n",
       "      <td>0.297769</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "   num_timesteps  num_attackers  num_defenders  num_clusters  dims  \\\n",
       "0             10              1              2            10     7   \n",
       "1             10              1              2            10     7   \n",
       "2             10              1              2            10     7   \n",
       "3             10              1              2            10     7   \n",
       "4             10              1              2            10     7   \n",
       "5             10              1              2            10     7   \n",
       "6             10              1              2            10     7   \n",
       "\n",
       "   defense_time_threshold  force_return  num_defender_actions  nash_value  \\\n",
       "0                       2          True                  2401   -0.289241   \n",
       "1                       2          True                  2401   -0.289241   \n",
       "2                       2          True                  2401   -0.289241   \n",
       "3                       2          True                  2401   -0.289241   \n",
       "4                       2          True                  2401   -0.289241   \n",
       "5                       2          True                  2401   -0.289241   \n",
       "6                       2          True                  2401   -0.289241   \n",
       "\n",
       "   nash_support  nash_runtime  mip_value  mip_support  mip_runtime  \n",
       "0             8      0.286761  -0.475524            1     0.549913  \n",
       "1             8      0.286761  -0.361450            2     0.582956  \n",
       "2             8      0.286761  -0.322251            3     1.765882  \n",
       "3             8      0.286761  -0.303916            4     1.708579  \n",
       "4             8      0.286761  -0.295432            5     1.872997  \n",
       "5             8      0.286761  -0.289596            6     1.427549  \n",
       "6             8      0.286761  -0.289241            7     0.297769  "
      ]
     },
     "execution_count": 84,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df = pd.DataFrame({\n",
    "    \"num_timesteps\":[10 for i in range(len(mip_supports))],\n",
    "    \"num_attackers\":[1 for i in range(len(mip_supports))],\n",
    "    \"num_defenders\":[2 for i in range(len(mip_supports))],\n",
    "    \"num_clusters\":[10 for i in range(len(mip_supports))],\n",
    "    \"dims\":[7 for i in range(len(mip_supports))],\n",
    "    \"defense_time_threshold\":[2 for i in range(len(mip_supports))],\n",
    "    \"force_return\":[True for i in range(len(mip_supports))],\n",
    "    \"num_defender_actions\": [gsg.schedule_form_dict[\"defender_utility_matrix\"].shape[0] for i in range(len(mip_supports))],\n",
    "    \"nash_value\":[nu for i in range(len(mip_supports))],\n",
    "    \"nash_support\":[nsupport for i in range(len(mip_supports))],\n",
    "    \"nash_runtime\":[nruntime for i in range(len(mip_supports))],\n",
    "    \"mip_value\":mip_us,\n",
    "    \"mip_support\":mip_supports,\n",
    "    \"mip_runtime\":mip_runtimes,\n",
    "})\n",
    "df"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 85,
   "id": "88bc7cbf-4b61-4729-a4ad-8d2a61c31ecc",
   "metadata": {},
   "outputs": [],
   "source": [
    "df.to_csv(\"GSG_SF_T10_SPARSITY_FINAL.csv\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "410f2929-4d5f-475e-bc76-2523de1bfb1e",
   "metadata": {},
   "source": [
    "# ISG"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "6b11deac-5b3c-49fe-bee2-f32297df15d2",
   "metadata": {},
   "outputs": [],
   "source": [
    "gdf = gpd.read_file(\"chinatown_infra.geojson\")\n",
    "\n",
    "# Step 1: Handle relevant columns\n",
    "infra_columns = [\n",
    "    \"id\", \"name\", \"power\", \"man_made\", \"amenity\",\n",
    "    \"generator:method\", \"generator:source\", \"geometry\"\n",
    "]\n",
    "available_columns = [col for col in infra_columns if col in gdf.columns]\n",
    "gdf = gdf[available_columns].copy()\n",
    "\n",
    "# Step 2: Extract generator type if present\n",
    "gdf[\"generator_type\"] = gdf.get(\"generator:method\")\n",
    "if \"generator_type\" not in gdf.columns or gdf[\"generator_type\"].isnull().all():\n",
    "    gdf[\"generator_type\"] = gdf.get(\"generator:source\")\n",
    "\n",
    "# Step 3: Construct unified 'type' column\n",
    "gdf[\"type\"] = gdf.get(\"power\")\n",
    "if \"amenity\" in gdf.columns:\n",
    "    gdf[\"type\"] = gdf[\"type\"].combine_first(gdf[\"amenity\"])\n",
    "if \"man_made\" in gdf.columns:\n",
    "    gdf[\"type\"] = gdf[\"type\"].combine_first(gdf[\"man_made\"])\n",
    "\n",
    "# Step 4: Refine generator classification (solar vs. other)\n",
    "gdf.loc[(gdf[\"type\"] == \"generator\") & (gdf[\"generator_type\"] == \"photovoltaic\"), \"type\"] = \"solar_generator\"\n",
    "gdf.loc[(gdf[\"type\"] == \"generator\") & (gdf[\"generator_type\"] == \"solar\"), \"type\"] = \"solar_generator\"\n",
    "\n",
    "# Step 5: Drop raw columns now that 'type' is finalized\n",
    "df_simple = gdf[[\"id\", \"name\", \"type\", \"geometry\"]].copy()\n",
    "\n",
    "# Step 6: Separate nodes and ways\n",
    "df_nodes = df_simple[df_simple[\"id\"].str.contains(\"node\")].copy()\n",
    "df_nodes[\"x\"] = df_nodes.geometry.x\n",
    "df_nodes[\"y\"] = df_nodes.geometry.y\n",
    "df_nodes = df_nodes.drop(columns=[\"geometry\"])\n",
    "\n",
    "df_ways = df_simple[df_simple[\"id\"].str.contains(\"way\")].copy()\n",
    "df_ways = df_ways.set_geometry(\"geometry\").to_crs(\"EPSG:32618\")\n",
    "df_ways[\"centroid\"] = df_ways.geometry.centroid\n",
    "df_ways = df_ways.set_geometry(\"centroid\").to_crs(\"EPSG:4326\")\n",
    "df_ways[\"x\"] = df_ways.geometry.x\n",
    "df_ways[\"y\"] = df_ways.geometry.y\n",
    "df_ways = df_ways.drop(columns=[\"geometry\", \"centroid\"])\n",
    "\n",
    "# Step 7: Combine nodes and ways\n",
    "df_combined = pd.concat([df_nodes, df_ways], ignore_index=True)\n",
    "df_combined = pd.concat([df_nodes, df_ways], ignore_index=True)\n",
    "ny_blocks_gdf =  gpd.read_file(\"tl_2020_36_tabblock20.shp\")\n",
    "INFRA_WEIGHTS = {\n",
    "    # Power Infrastructure\n",
    "    \"plant\": 1.5,\n",
    "    \"generator\": 1.35,\n",
    "    \"solar_generator\": 0.95,\n",
    "    \"substation\": 1.45,\n",
    "    \"transformer\": 1.25,\n",
    "    \"tower\": 1.1,\n",
    "    \"pole\": 0.85,\n",
    "    \"line\": 1.0,\n",
    "    \"minor_line\": 0.9,\n",
    "    \"cable\": 0.95,\n",
    "    \"switchgear\": 1.2,\n",
    "    \"busbar\": 0.8,\n",
    "    \"bay\": 0.85,\n",
    "    \"converter\": 1.05,\n",
    "    \"insulator\": 0.75,\n",
    "    \"portal\": 0.75,\n",
    "    \"connection\": 0.7,\n",
    "    \"compensator\": 1.0,\n",
    "    \"rectifier\": 0.95,\n",
    "    \"inverter\": 0.95,\n",
    "    \"storage\": 0.9,\n",
    "\n",
    "    # Healthcare\n",
    "    \"hospital\": 1.5,\n",
    "    \"clinic\": 1.35,\n",
    "\n",
    "    # Education\n",
    "    \"school\": 1.25,\n",
    "    \"university\": 1.4,\n",
    "\n",
    "    # Water & Sanitation\n",
    "    \"water_works\": 1.45,\n",
    "    \"wastewater_plant\": 1.4,\n",
    "\n",
    "    # Government & Emergency Services\n",
    "    \"fire_station\": 1.3,\n",
    "    \"police\": 1.4,\n",
    "    \"courthouse\": 1.2,\n",
    "\n",
    "    # Critical Infrastructure\n",
    "    \"bunker_silo\": 1.0,\n",
    "\n",
    "    # Communications\n",
    "    \"communications_tower\": 1.25,\n",
    "}"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d6f9ca92-a41d-466f-b471-12cf91c4c34e",
   "metadata": {},
   "source": [
    "## SF"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "5063db75-59b6-48fc-b96d-559a24228386",
   "metadata": {},
   "outputs": [],
   "source": [
    "import warnings\n",
    "warnings.filterwarnings('ignore', category=FutureWarning)\n",
    "\n",
    "schedule_form_kwargs = {\n",
    "    \"schedule_form\": True,\n",
    "    \"simple\": False,\n",
    "    \"attacker_penalty_factor\": 3,\n",
    "    \"defender_penalty_factor\": 3,\n",
    "}\n",
    "\n",
    "general_sum_kwargs = {\n",
    "    \"general_sum\": False,\n",
    "    \"attacker_feature_value\":  42, \n",
    "    \"defender_feature_value\": 69, \n",
    "    \"defender_step_cost\": 32.5, \n",
    "}\n",
    "\n",
    "# Bounding box for Hoboken, NJ\n",
    "# bbox_hoboken_small = (40.752635, 40.745600, -74.030386,-74.043903)\n",
    "bbox_hoboken_low = (40.745411, 40.735486, -74.025857,-74.041479)\n",
    "bbox_hoboken_east = (40.748337, 40.734641,-74.022961,-74.031286)\n",
    "bbox_downtown = (40.718721, 40.714078, -73.996074, -74.002651)\n",
    "bbox_downtown_large = (40.7215, 40.710, -73.9935, -74.010)\n",
    "# 40.7060, -74.0140, 40.7205, -73.9935\n",
    "college_police = (40.743293077312465, -74.02670221027175)\n",
    "police_station = (40.73768931976651, -74.02990519431108)\n",
    "traffic_police = (40.7366602084371, -74.03449866349136)\n",
    "downtown_station = (40.71232433042349, -74.00187755238431)\n",
    "fifth_ave_station = (40.71637413934789, -73.9973285259067)\n",
    "fifth_precinct = (40.71625547686622, -73.99736909131171)\n",
    "booking_station = (40.716191530904815, -74.00102237385177)\n",
    "police_plaza = (40.71236124409745, -74.00173715463521)\n",
    "troop_nyc = (40.71657885026091, -74.00641139014367)\n",
    "first_precinct = (40.720411300417446, -74.0070247584372)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 63,
   "id": "38230705-0072-4596-9a7f-5f259e775d83",
   "metadata": {},
   "outputs": [],
   "source": [
    "isg = InfraSecurityGame(df_combined, ny_blocks_gdf, INFRA_WEIGHTS, bbox=bbox_downtown_large)\n",
    "isg.generate(num_attackers=1, \n",
    "             num_defenders=3, \n",
    "             home_base_assignments=[(fifth_precinct,booking_station, troop_nyc, first_precinct, police_plaza),(fifth_precinct,booking_station, troop_nyc, first_precinct, police_plaza),(fifth_precinct,booking_station, troop_nyc, first_precinct, police_plaza)],\n",
    "             num_timesteps=9, \n",
    "             generate_utility_matrix=True, \n",
    "             generate_actions=False, \n",
    "             force_return=True, \n",
    "             defense_time_threshold=3, \n",
    "             **general_sum_kwargs, \n",
    "             **schedule_form_kwargs)\n",
    "\n",
    "#10 timesteps, 3 def, dt 2: (11, -0.4043878750324344)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 64,
   "id": "f6db6127-6da3-41af-938a-e51e5f3a8aeb",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(29791, 23)"
      ]
     },
     "execution_count": 64,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "isg.schedule_form_dict[\"defender_utility_matrix\"].shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "136926c7-d42c-4848-82f0-74f96712aace",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(11, -0.420864964164455)"
      ]
     },
     "execution_count": 30,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "#9 timesteps 3 def 3dt\n",
    "start = time.time()\n",
    "nD_a, nD_d, nu = nash(isg.schedule_form_dict[\"defender_utility_matrix\"])\n",
    "end = time.time()\n",
    "nruntime = end-start\n",
    "nsupport = sum([1 for p in nD_d if p!=0])\n",
    "nsupport,nu"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "59fc25a3-9a5c-4998-aea1-9b4f838228f0",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(9, -0.4176016505451655)"
      ]
     },
     "execution_count": 33,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "#9 timesteps 3 def 2dt\n",
    "start = time.time()\n",
    "nD_a, nD_d, nu = nash(isg.schedule_form_dict[\"defender_utility_matrix\"])\n",
    "end = time.time()\n",
    "nruntime = end-start\n",
    "nsupport = sum([1 for p in nD_d if p!=0])\n",
    "nsupport,nu"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 62,
   "id": "f45cf9da-68dd-47ac-a7f8-6b29af32b49e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(11, -0.420864964164455)"
      ]
     },
     "execution_count": 62,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "start = time.time()\n",
    "nD_a, nD_d, nu = nash(isg.schedule_form_dict[\"defender_utility_matrix\"])\n",
    "end = time.time()\n",
    "nruntime = end-start\n",
    "nsupport = sum([1 for p in nD_d if p!=0])\n",
    "nsupport,nu"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d90bda67-fcaf-4861-97c5-0beb2c38038d",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "starting i=1 at time 1745993149.0215442\n",
      "finished i=1 in 39.68994855880737 seconds with u=-0.49543616569765425\n",
      "starting i=2 at time 1745993188.7185214\n",
      "finished i=2 in 185.47285151481628 seconds with u=-0.4518894049262475\n",
      "starting i=3 at time 1745993374.1913729\n",
      "finished i=3 in 43.33791470527649 seconds with u=-0.44334275147120084\n",
      "starting i=4 at time 1745993417.5292876\n",
      "finished i=4 in 67.4903175830841 seconds with u=-0.42472217747854063\n",
      "starting i=5 at time 1745993485.0196052\n",
      "finished i=5 in 926.8983609676361 seconds with u=-0.42302295804969314\n",
      "starting i=6 at time 1745994411.9179661\n",
      "finished i=6 in 2541.8093144893646 seconds with u=-0.4213980041933654\n",
      "starting i=7 at time 1745996953.7272806\n"
     ]
    }
   ],
   "source": [
    "mip_us = []\n",
    "mip_supports = []\n",
    "mip_runtimes = []\n",
    "\n",
    "for i in range(1,10):\n",
    "    start = time.time()\n",
    "    print(f\"starting i={i} at time {start}\")\n",
    "    mu, mD_d = mip(isg.schedule_form_dict[\"defender_utility_matrix\"],i)\n",
    "    end = time.time()\n",
    "    print(f\"finished i={i} in {end-start} seconds with u={mu}\")\n",
    "    msupport = sum([1 for p in mD_d if p!=0])\n",
    "    mip_us.append(mu)\n",
    "    mip_supports.append(i)\n",
    "    mip_runtimes.append(end-start)\n",
    "    if abs(mu-nu) <= 1e-12:\n",
    "        break"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 52,
   "id": "0a8d8760-da2e-4954-91ee-be86cc52468d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>num_timesteps</th>\n",
       "      <th>num_attackers</th>\n",
       "      <th>num_defenders</th>\n",
       "      <th>num_clusters</th>\n",
       "      <th>dims</th>\n",
       "      <th>defense_time_threshold</th>\n",
       "      <th>force_return</th>\n",
       "      <th>num_defender_actions</th>\n",
       "      <th>nash_value</th>\n",
       "      <th>nash_support</th>\n",
       "      <th>nash_runtime</th>\n",
       "      <th>mip_value</th>\n",
       "      <th>mip_support</th>\n",
       "      <th>mip_runtime</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>9</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.463238</td>\n",
       "      <td>8</td>\n",
       "      <td>18.287839</td>\n",
       "      <td>-0.889447</td>\n",
       "      <td>1</td>\n",
       "      <td>33.652055</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>9</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.463238</td>\n",
       "      <td>8</td>\n",
       "      <td>18.287839</td>\n",
       "      <td>-0.597020</td>\n",
       "      <td>2</td>\n",
       "      <td>362.620123</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>9</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.463238</td>\n",
       "      <td>8</td>\n",
       "      <td>18.287839</td>\n",
       "      <td>-0.507538</td>\n",
       "      <td>3</td>\n",
       "      <td>62.383189</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>9</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.463238</td>\n",
       "      <td>8</td>\n",
       "      <td>18.287839</td>\n",
       "      <td>-0.478263</td>\n",
       "      <td>4</td>\n",
       "      <td>117.299998</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>9</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.463238</td>\n",
       "      <td>8</td>\n",
       "      <td>18.287839</td>\n",
       "      <td>-0.471666</td>\n",
       "      <td>5</td>\n",
       "      <td>127.923895</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>9</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.463238</td>\n",
       "      <td>8</td>\n",
       "      <td>18.287839</td>\n",
       "      <td>-0.465396</td>\n",
       "      <td>6</td>\n",
       "      <td>141.439785</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6</th>\n",
       "      <td>9</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.463238</td>\n",
       "      <td>8</td>\n",
       "      <td>18.287839</td>\n",
       "      <td>-0.463238</td>\n",
       "      <td>7</td>\n",
       "      <td>20.655978</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "   num_timesteps  num_attackers  num_defenders  num_clusters  dims  \\\n",
       "0              9              2              1            10     7   \n",
       "1              9              2              1            10     7   \n",
       "2              9              2              1            10     7   \n",
       "3              9              2              1            10     7   \n",
       "4              9              2              1            10     7   \n",
       "5              9              2              1            10     7   \n",
       "6              9              2              1            10     7   \n",
       "\n",
       "   defense_time_threshold  force_return  num_defender_actions  nash_value  \\\n",
       "0                       1         False                 41479   -0.463238   \n",
       "1                       1         False                 41479   -0.463238   \n",
       "2                       1         False                 41479   -0.463238   \n",
       "3                       1         False                 41479   -0.463238   \n",
       "4                       1         False                 41479   -0.463238   \n",
       "5                       1         False                 41479   -0.463238   \n",
       "6                       1         False                 41479   -0.463238   \n",
       "\n",
       "   nash_support  nash_runtime  mip_value  mip_support  mip_runtime  \n",
       "0             8     18.287839  -0.889447            1    33.652055  \n",
       "1             8     18.287839  -0.597020            2   362.620123  \n",
       "2             8     18.287839  -0.507538            3    62.383189  \n",
       "3             8     18.287839  -0.478263            4   117.299998  \n",
       "4             8     18.287839  -0.471666            5   127.923895  \n",
       "5             8     18.287839  -0.465396            6   141.439785  \n",
       "6             8     18.287839  -0.463238            7    20.655978  "
      ]
     },
     "execution_count": 52,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df = pd.DataFrame({\n",
    "    \"num_timesteps\":[9 for i in range(len(mip_supports))],\n",
    "    \"num_attackers\":[2 for i in range(len(mip_supports))],\n",
    "    \"num_defenders\":[1 for i in range(len(mip_supports))],\n",
    "    \"num_clusters\":[10 for i in range(len(mip_supports))],\n",
    "    \"dims\":[7 for i in range(len(mip_supports))],\n",
    "    \"defense_time_threshold\":[1 for i in range(len(mip_supports))],\n",
    "    \"force_return\":[False for i in range(len(mip_supports))],\n",
    "    \"num_defender_actions\": [len(gsg.defender_actions) for i in range(len(mip_supports))],\n",
    "    \"nash_value\":[nu for i in range(len(mip_supports))],\n",
    "    \"nash_support\":[nsupport for i in range(len(mip_supports))],\n",
    "    \"nash_runtime\":[nruntime for i in range(len(mip_supports))],\n",
    "    \"mip_value\":mip_us,\n",
    "    \"mip_support\":mip_supports,\n",
    "    \"mip_runtime\":mip_runtimes,\n",
    "})\n",
    "df"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 53,
   "id": "9eb8ea54-ae19-4068-a36e-893b238c1f0f",
   "metadata": {},
   "outputs": [],
   "source": [
    "df.to_csv(\"GSG_NFG_T9_SPARSITY.csv\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "95ffd1cf-da71-499a-b70b-0e524ef53388",
   "metadata": {},
   "source": [
    "## SF"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 286,
   "id": "c5207266-6d28-44d9-bf46-a8be21fed4e7",
   "metadata": {},
   "outputs": [],
   "source": [
    "df = pd.read_csv(\"lobeke.csv\")\n",
    "df.dropna(inplace=True)\n",
    "\n",
    "lat_min, lon_min = 2.0530, 15.8790\n",
    "lat_max, lon_max = 2.2837, 16.2038\n",
    "\n",
    "coordinate_rectangle = [lat_min, lat_max, lon_min, lon_max]\n",
    "\n",
    "schedule_form_kwargs = {\n",
    "    \"schedule_form\": True,\n",
    "    \"simple\": False,\n",
    "    \"attacker_penalty_factor\": 5,\n",
    "    \"defender_penalty_factor\": 5,\n",
    "}\n",
    "\n",
    "general_sum_kwargs = {\n",
    "    \"general_sum\": False,\n",
    "    \"attacker_feature_value\":  42, \n",
    "    \"defender_feature_value\": 69, \n",
    "    \"defender_step_cost\": 32.5, \n",
    "}\n",
    "\n",
    "boulou_camp = (2.2,15.9)\n",
    "# lobeke_camp = (2.25,15.75)\n",
    "kabo_djembe = (2.0532352380408088, 16.085709866529694)\n",
    "bomassa = (2.2037280296158355, 16.187056364164913)\n",
    "inner_post = (2.2,15.98)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 287,
   "id": "8d686919-686e-48dc-82fb-7cf6b84d3d33",
   "metadata": {},
   "outputs": [],
   "source": [
    "gsg = GreenSecurityGame(df, coordinate_rectangle, \"centroid\", num_clusters=10, num_rows=8, num_columns=8)\n",
    "gsg.generate(num_attackers=1, \n",
    "             num_defenders=2, \n",
    "             home_base_assignments=[(kabo_djembe, bomassa, inner_post),(kabo_djembe, bomassa, inner_post)], \n",
    "             num_timesteps=8, \n",
    "             generate_utility_matrix=True, \n",
    "             defense_time_threshold=2, \n",
    "             generate_actions=False, \n",
    "             force_return=True, \n",
    "             general_sum=False, \n",
    "             **schedule_form_kwargs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 288,
   "id": "1dc03747-4cad-4bb6-99bf-949a8733cc8d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(361, 10)"
      ]
     },
     "execution_count": 288,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "gsg.schedule_form_dict[\"defender_utility_matrix\"].shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 289,
   "id": "80542e65-b38d-4795-bf6a-c7552ebdf913",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(6, -0.27729882609690903)"
      ]
     },
     "execution_count": 289,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "start = time.time()\n",
    "nD_a, nD_d, nu = nash(gsg.schedule_form_dict[\"defender_utility_matrix\"])\n",
    "end = time.time()\n",
    "nruntime = end-start\n",
    "nsupport = sum([1 for p in nD_d if p!=0])\n",
    "nsupport,nu"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 290,
   "id": "77779925-50b1-4f79-b0b6-0a2272605188",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "starting i=1 at time 1745986556.0693774\n",
      "finished i=1 in 0.09237241744995117 seconds with u=-0.5226130653266332\n",
      "starting i=2 at time 1745986556.1617498\n",
      "finished i=2 in 0.24329137802124023 seconds with u=-0.3366140189826578\n",
      "starting i=3 at time 1745986556.4050412\n",
      "finished i=3 in 0.13461565971374512 seconds with u=-0.28245611317941893\n",
      "starting i=4 at time 1745986556.540661\n",
      "finished i=4 in 0.3160886764526367 seconds with u=-0.28007355003955064\n",
      "starting i=5 at time 1745986556.8567498\n",
      "finished i=5 in 0.07416820526123047 seconds with u=-0.2772988260969091\n"
     ]
    }
   ],
   "source": [
    "mip_us = []\n",
    "mip_supports = []\n",
    "mip_runtimes = []\n",
    "\n",
    "for i in range(1,10):\n",
    "    start = time.time()\n",
    "    print(f\"starting i={i} at time {start}\")\n",
    "    mu, mD_d = mip(gsg.schedule_form_dict[\"defender_utility_matrix\"],i)\n",
    "    end = time.time()\n",
    "    print(f\"finished i={i} in {end-start} seconds with u={mu}\")\n",
    "    msupport = sum([1 for p in mD_d if p!=0])\n",
    "    mip_us.append(mu)\n",
    "    mip_supports.append(i)\n",
    "    mip_runtimes.append(end-start)\n",
    "    if abs(mu-nu) <= 1e-12:\n",
    "        break"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 291,
   "id": "a1de506c-fb8a-499a-af4b-e16470b1b8e2",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>num_timesteps</th>\n",
       "      <th>num_attackers</th>\n",
       "      <th>num_defenders</th>\n",
       "      <th>num_clusters</th>\n",
       "      <th>dims</th>\n",
       "      <th>defense_time_threshold</th>\n",
       "      <th>force_return</th>\n",
       "      <th>num_defender_actions</th>\n",
       "      <th>nash_value</th>\n",
       "      <th>nash_support</th>\n",
       "      <th>nash_runtime</th>\n",
       "      <th>mip_value</th>\n",
       "      <th>mip_support</th>\n",
       "      <th>mip_runtime</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>8</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>10</td>\n",
       "      <td>10</td>\n",
       "      <td>2</td>\n",
       "      <td>True</td>\n",
       "      <td>361</td>\n",
       "      <td>-0.277299</td>\n",
       "      <td>6</td>\n",
       "      <td>0.061458</td>\n",
       "      <td>-0.522613</td>\n",
       "      <td>1</td>\n",
       "      <td>0.092372</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>8</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>10</td>\n",
       "      <td>10</td>\n",
       "      <td>2</td>\n",
       "      <td>True</td>\n",
       "      <td>361</td>\n",
       "      <td>-0.277299</td>\n",
       "      <td>6</td>\n",
       "      <td>0.061458</td>\n",
       "      <td>-0.336614</td>\n",
       "      <td>2</td>\n",
       "      <td>0.243291</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>8</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>10</td>\n",
       "      <td>10</td>\n",
       "      <td>2</td>\n",
       "      <td>True</td>\n",
       "      <td>361</td>\n",
       "      <td>-0.277299</td>\n",
       "      <td>6</td>\n",
       "      <td>0.061458</td>\n",
       "      <td>-0.282456</td>\n",
       "      <td>3</td>\n",
       "      <td>0.134616</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>8</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>10</td>\n",
       "      <td>10</td>\n",
       "      <td>2</td>\n",
       "      <td>True</td>\n",
       "      <td>361</td>\n",
       "      <td>-0.277299</td>\n",
       "      <td>6</td>\n",
       "      <td>0.061458</td>\n",
       "      <td>-0.280074</td>\n",
       "      <td>4</td>\n",
       "      <td>0.316089</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>8</td>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>10</td>\n",
       "      <td>10</td>\n",
       "      <td>2</td>\n",
       "      <td>True</td>\n",
       "      <td>361</td>\n",
       "      <td>-0.277299</td>\n",
       "      <td>6</td>\n",
       "      <td>0.061458</td>\n",
       "      <td>-0.277299</td>\n",
       "      <td>5</td>\n",
       "      <td>0.074168</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "   num_timesteps  num_attackers  num_defenders  num_clusters  dims  \\\n",
       "0              8              1              2            10    10   \n",
       "1              8              1              2            10    10   \n",
       "2              8              1              2            10    10   \n",
       "3              8              1              2            10    10   \n",
       "4              8              1              2            10    10   \n",
       "\n",
       "   defense_time_threshold  force_return  num_defender_actions  nash_value  \\\n",
       "0                       2          True                   361   -0.277299   \n",
       "1                       2          True                   361   -0.277299   \n",
       "2                       2          True                   361   -0.277299   \n",
       "3                       2          True                   361   -0.277299   \n",
       "4                       2          True                   361   -0.277299   \n",
       "\n",
       "   nash_support  nash_runtime  mip_value  mip_support  mip_runtime  \n",
       "0             6      0.061458  -0.522613            1     0.092372  \n",
       "1             6      0.061458  -0.336614            2     0.243291  \n",
       "2             6      0.061458  -0.282456            3     0.134616  \n",
       "3             6      0.061458  -0.280074            4     0.316089  \n",
       "4             6      0.061458  -0.277299            5     0.074168  "
      ]
     },
     "execution_count": 291,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df = pd.DataFrame({\n",
    "    \"num_timesteps\":[8 for i in range(len(mip_supports))],\n",
    "    \"num_attackers\":[1 for i in range(len(mip_supports))],\n",
    "    \"num_defenders\":[2 for i in range(len(mip_supports))],\n",
    "    \"num_clusters\":[10 for i in range(len(mip_supports))],\n",
    "    \"dims\":[10 for i in range(len(mip_supports))],\n",
    "    \"defense_time_threshold\":[2 for i in range(len(mip_supports))],\n",
    "    \"force_return\":[True for i in range(len(mip_supports))],\n",
    "    \"num_defender_actions\": [gsg.schedule_form_dict[\"defender_utility_matrix\"].shape[0] for i in range(len(mip_supports))],\n",
    "    \"nash_value\":[nu for i in range(len(mip_supports))],\n",
    "    \"nash_support\":[nsupport for i in range(len(mip_supports))],\n",
    "    \"nash_runtime\":[nruntime for i in range(len(mip_supports))],\n",
    "    \"mip_value\":mip_us,\n",
    "    \"mip_support\":mip_supports,\n",
    "    \"mip_runtime\":mip_runtimes,\n",
    "})\n",
    "df"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 292,
   "id": "da9b98ba-c463-4934-bc21-0014c92e1f60",
   "metadata": {},
   "outputs": [],
   "source": [
    "df.to_csv(\"GSG_SF_T8_SPARSITY.csv\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "id": "fd790611-809b-4b3a-9c6e-b33bfe77d24f",
   "metadata": {},
   "outputs": [],
   "source": [
    "gsg = GreenSecurityGame(df, coordinate_rectangle, \"centroid\", num_clusters=10, num_rows=7, num_columns=7)\n",
    "gsg.generate(num_attackers=2, \n",
    "             num_defenders=1, \n",
    "             home_base_assignments=[(kabo_djembe, bomassa, inner_post)], \n",
    "             num_timesteps=9, \n",
    "             generate_utility_matrix=True, \n",
    "             defense_time_threshold=1, \n",
    "             generate_actions=False, \n",
    "             force_return=False, \n",
    "             general_sum=False, \n",
    "             **schedule_form_kwargs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 49,
   "id": "f1a8bb09-2b2f-44c1-9d25-9d98652dd0c9",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(41479, 9, 1)"
      ]
     },
     "execution_count": 49,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "gsg.defender_actions.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 50,
   "id": "8f2d6be5-cc1b-476b-ad24-32938c60941d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(8, -0.46323812842864565)"
      ]
     },
     "execution_count": 50,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "start = time.time()\n",
    "nD_a, nD_d, nu = nash(gsg.utility_matrix)\n",
    "end = time.time()\n",
    "nruntime = end-start\n",
    "nsupport = sum([1 for p in nD_d if p!=0])\n",
    "nsupport,nu"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 51,
   "id": "d4e45c01-8407-4ace-97a5-633f7f21d7dc",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "starting i=1 at time 1745981272.9549503\n",
      "finished i=1 in 33.65205478668213 seconds with u=-0.8894472361809045\n",
      "starting i=2 at time 1745981306.6102126\n",
      "finished i=2 in 362.6201231479645 seconds with u=-0.5970202788449397\n",
      "starting i=3 at time 1745981669.2332556\n",
      "finished i=3 in 62.38318920135498 seconds with u=-0.5075376884422111\n",
      "starting i=4 at time 1745981731.6164448\n",
      "finished i=4 in 117.29999780654907 seconds with u=-0.4782634344465709\n",
      "starting i=5 at time 1745981848.9164426\n",
      "finished i=5 in 127.92389488220215 seconds with u=-0.4716662632569949\n",
      "starting i=6 at time 1745981976.8419363\n",
      "finished i=6 in 141.43978476524353 seconds with u=-0.4653956597949693\n",
      "starting i=7 at time 1745982118.2914765\n",
      "finished i=7 in 20.655978202819824 seconds with u=-0.46323812842864576\n"
     ]
    }
   ],
   "source": [
    "mip_us = []\n",
    "mip_supports = []\n",
    "mip_runtimes = []\n",
    "\n",
    "for i in range(1,10):\n",
    "    start = time.time()\n",
    "    print(f\"starting i={i} at time {start}\")\n",
    "    mu, mD_d = mip(gsg.utility_matrix,i)\n",
    "    end = time.time()\n",
    "    print(f\"finished i={i} in {end-start} seconds with u={mu}\")\n",
    "    msupport = sum([1 for p in mD_d if p!=0])\n",
    "    mip_us.append(mu)\n",
    "    mip_supports.append(i)\n",
    "    mip_runtimes.append(end-start)\n",
    "    if abs(mu-nu) <= 1e-12:\n",
    "        break"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 52,
   "id": "7f48d4dc-721c-403e-bb66-d555f3d85b63",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>num_timesteps</th>\n",
       "      <th>num_attackers</th>\n",
       "      <th>num_defenders</th>\n",
       "      <th>num_clusters</th>\n",
       "      <th>dims</th>\n",
       "      <th>defense_time_threshold</th>\n",
       "      <th>force_return</th>\n",
       "      <th>num_defender_actions</th>\n",
       "      <th>nash_value</th>\n",
       "      <th>nash_support</th>\n",
       "      <th>nash_runtime</th>\n",
       "      <th>mip_value</th>\n",
       "      <th>mip_support</th>\n",
       "      <th>mip_runtime</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>9</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.463238</td>\n",
       "      <td>8</td>\n",
       "      <td>18.287839</td>\n",
       "      <td>-0.889447</td>\n",
       "      <td>1</td>\n",
       "      <td>33.652055</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>9</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.463238</td>\n",
       "      <td>8</td>\n",
       "      <td>18.287839</td>\n",
       "      <td>-0.597020</td>\n",
       "      <td>2</td>\n",
       "      <td>362.620123</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>9</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.463238</td>\n",
       "      <td>8</td>\n",
       "      <td>18.287839</td>\n",
       "      <td>-0.507538</td>\n",
       "      <td>3</td>\n",
       "      <td>62.383189</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>9</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.463238</td>\n",
       "      <td>8</td>\n",
       "      <td>18.287839</td>\n",
       "      <td>-0.478263</td>\n",
       "      <td>4</td>\n",
       "      <td>117.299998</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>9</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.463238</td>\n",
       "      <td>8</td>\n",
       "      <td>18.287839</td>\n",
       "      <td>-0.471666</td>\n",
       "      <td>5</td>\n",
       "      <td>127.923895</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>9</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.463238</td>\n",
       "      <td>8</td>\n",
       "      <td>18.287839</td>\n",
       "      <td>-0.465396</td>\n",
       "      <td>6</td>\n",
       "      <td>141.439785</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6</th>\n",
       "      <td>9</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "      <td>10</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>False</td>\n",
       "      <td>41479</td>\n",
       "      <td>-0.463238</td>\n",
       "      <td>8</td>\n",
       "      <td>18.287839</td>\n",
       "      <td>-0.463238</td>\n",
       "      <td>7</td>\n",
       "      <td>20.655978</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "   num_timesteps  num_attackers  num_defenders  num_clusters  dims  \\\n",
       "0              9              2              1            10     7   \n",
       "1              9              2              1            10     7   \n",
       "2              9              2              1            10     7   \n",
       "3              9              2              1            10     7   \n",
       "4              9              2              1            10     7   \n",
       "5              9              2              1            10     7   \n",
       "6              9              2              1            10     7   \n",
       "\n",
       "   defense_time_threshold  force_return  num_defender_actions  nash_value  \\\n",
       "0                       1         False                 41479   -0.463238   \n",
       "1                       1         False                 41479   -0.463238   \n",
       "2                       1         False                 41479   -0.463238   \n",
       "3                       1         False                 41479   -0.463238   \n",
       "4                       1         False                 41479   -0.463238   \n",
       "5                       1         False                 41479   -0.463238   \n",
       "6                       1         False                 41479   -0.463238   \n",
       "\n",
       "   nash_support  nash_runtime  mip_value  mip_support  mip_runtime  \n",
       "0             8     18.287839  -0.889447            1    33.652055  \n",
       "1             8     18.287839  -0.597020            2   362.620123  \n",
       "2             8     18.287839  -0.507538            3    62.383189  \n",
       "3             8     18.287839  -0.478263            4   117.299998  \n",
       "4             8     18.287839  -0.471666            5   127.923895  \n",
       "5             8     18.287839  -0.465396            6   141.439785  \n",
       "6             8     18.287839  -0.463238            7    20.655978  "
      ]
     },
     "execution_count": 52,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df = pd.DataFrame({\n",
    "    \"num_timesteps\":[9 for i in range(len(mip_supports))],\n",
    "    \"num_attackers\":[2 for i in range(len(mip_supports))],\n",
    "    \"num_defenders\":[1 for i in range(len(mip_supports))],\n",
    "    \"num_clusters\":[10 for i in range(len(mip_supports))],\n",
    "    \"dims\":[7 for i in range(len(mip_supports))],\n",
    "    \"defense_time_threshold\":[1 for i in range(len(mip_supports))],\n",
    "    \"force_return\":[False for i in range(len(mip_supports))],\n",
    "    \"num_defender_actions\": [len(gsg.defender_actions) for i in range(len(mip_supports))],\n",
    "    \"nash_value\":[nu for i in range(len(mip_supports))],\n",
    "    \"nash_support\":[nsupport for i in range(len(mip_supports))],\n",
    "    \"nash_runtime\":[nruntime for i in range(len(mip_supports))],\n",
    "    \"mip_value\":mip_us,\n",
    "    \"mip_support\":mip_supports,\n",
    "    \"mip_runtime\":mip_runtimes,\n",
    "})\n",
    "df"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 53,
   "id": "98f4aed4-0ede-42f2-8a03-e8e110c2c453",
   "metadata": {},
   "outputs": [],
   "source": [
    "df.to_csv(\"GSG_NFG_T9_SPARSITY.csv\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "64bd38d5-e604-4363-a6cf-715e5ee88475",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "041a79ec-cb12-43b2-9212-8cafae923f5b",
   "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.12.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
