{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "8ae5768d",
   "metadata": {},
   "source": [
    "## Library"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "9d0ab9ad",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-02-01T17:21:01.982083Z",
     "start_time": "2024-02-01T17:20:52.388057Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[33mWARNING: Running pip as the 'root' user can result in broken permissions and conflicting behaviour with the system package manager. It is recommended to use a virtual environment instead: https://pip.pypa.io/warnings/venv\u001b[0m\n",
      "Requirement already satisfied: urllib3<1.27,>=1.21.1 in /opt/conda/lib/python3.8/site-packages (from requests>=2.19.0->dgl==1.0.1+cu117) (1.26.6)\n",
      "\u001b[33mWARNING: Running pip as the 'root' user can result in broken permissions and conflicting behaviour with the system package manager. It is recommended to use a virtual environment instead: https://pip.pypa.io/warnings/venv\u001b[0m\n",
      "Requirement already satisfied: networkx==3.1 in /opt/conda/lib/python3.8/site-packages (3.1)\n"
     ]
    }
   ],
   "source": [
    "! pip install dgl==1.0.1+cu117 -f https://data.dgl.ai/wheels/cu117/repo.html | tail -n 1\n",
    "! pip install networkx==3.1 | tail -n 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "103cd4dd",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-02-01T17:21:21.540922Z",
     "start_time": "2024-02-01T17:21:21.489600Z"
    }
   },
   "outputs": [],
   "source": [
    "import random\n",
    "import os\n",
    "import copy\n",
    "from collections import OrderedDict, defaultdict\n",
    "from itertools import chain, islice, combinations\n",
    "from time import time\n",
    "from tqdm import tqdm\n",
    "\n",
    "import dgl\n",
    "from dgl.nn.pytorch import GraphConv\n",
    "from dgl.nn.pytorch import SAGEConv\n",
    "\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from mpl_toolkits.mplot3d import Axes3D\n",
    "import pandas as pd\n",
    "import seaborn as sns\n",
    "import sklearn \n",
    "from sklearn.preprocessing import StandardScaler\n",
    "from sklearn.decomposition import PCA\n",
    "from networkx.algorithms.approximation.clique import maximum_independent_set as mis\n",
    "import networkx as nx\n",
    "\n",
    "from main import utils\n",
    "from main import gnn\n",
    "from main import instance\n",
    "\n",
    "device = torch.device('cuda:0' if torch.cuda.is_available() else 'cpu')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "447f97ba",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-02-01T17:21:22.568083Z",
     "start_time": "2024-02-01T17:21:22.550571Z"
    }
   },
   "outputs": [],
   "source": [
    "# fix seed\n",
    "SEED=0\n",
    "utils.fix_seed(SEED)\n",
    "torch_type=torch.float32"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d446f0b4",
   "metadata": {},
   "source": [
    "## Finding Penalty-diversified Solutions"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b5dd5d0f",
   "metadata": {},
   "source": [
    "### Step 1: Set hyperparameters"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "29c9ed6f",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-02-01T17:27:08.291332Z",
     "start_time": "2024-02-01T17:27:04.260828Z"
    }
   },
   "outputs": [],
   "source": [
    "# Set Penalties\n",
    "penalties=[0.04*i for i in range(1, 50)]\n",
    "\n",
    "# Graph parameters\n",
    "N, d, p, graph_type = 1000, 20, None, \"reg\"\n",
    "nx_graph = nx.random_regular_graph(d=d, n=N, seed=SEED)\n",
    "dgl_graph = dgl.from_networkx(nx_graph).to(device)\n",
    "Q_mats = [utils.qubo_dict_to_torch(nx_graph, \n",
    "                                   instance.gen_q_dict_mis_sym(nx_graph,penalty=penalty)\n",
    "                                  ).to(device) for penalty in penalties]\n",
    "\n",
    "# GNN Architecture\n",
    "in_feats = int(dgl_graph.number_of_nodes()**(0.8))\n",
    "hidden_size=int(in_feats)\n",
    "num_class=len(penalties)\n",
    "dropout=0.0\n",
    "model=gnn.GNNSage_dev(in_feats, \n",
    "                      hidden_size, \n",
    "                      num_class, \n",
    "                      dropout, \n",
    "                      device, \n",
    "                      agg_type='mean'\n",
    "                     ).to(device)\n",
    "embedding= nn.Embedding(dgl_graph.number_of_nodes(), \n",
    "                        in_feats\n",
    "                       ).type(torch_type).to(device)\n",
    "\n",
    "# Learning Parameters\n",
    "num_epoch=int(1e+5)\n",
    "lr=1e-4 \n",
    "weight_decay=1e-2\n",
    "tol=1e-5 \n",
    "patience=1000\n",
    "vari_param=0\n",
    "init_reg_param=-10 \n",
    "annealing_rate=1e-3\n",
    "check_interval=5000"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b21437fe",
   "metadata": {},
   "source": [
    "### Set 2: Define loss function"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "36aa5e35",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-02-01T17:21:32.844757Z",
     "start_time": "2024-02-01T17:21:32.840315Z"
    }
   },
   "outputs": [],
   "source": [
    "def loss(probs, reg_param, vari_param=0):\n",
    "    \n",
    "    # cost function\n",
    "    sum_cost=0\n",
    "    for idx in range(probs.size(1)):\n",
    "        sum_cost+=probs[:, idx].T @ Q_mats[idx] @ probs[:, idx]\n",
    "    \n",
    "    # diversity measure\n",
    "    vari_term = -1*probs.size(1)*probs.std(dim=1).sum()\n",
    "    \n",
    "    # Annealed Term\n",
    "    reg_term = torch.sum((1-(2*probs-1)**2))\n",
    "    \n",
    "    return sum_cost+reg_param*reg_term+vari_param*vari_term, sum_cost/probs.size(1), reg_term, vari_term"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8b301921",
   "metadata": {},
   "source": [
    "### Set 3: Run CTRA-PI-GNN solver"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "cabb0327",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-02-01T17:25:10.597897Z",
     "start_time": "2024-02-01T17:21:32.846606Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "【TRAIN EPOCH 0】LOSS 27151.906, COST 6148.474, REG 27412.330, VAR -16182.199, PARAM -10.000\n",
      "【TRAIN EPOCH 5000】LOSS -138172.766, COST 1061.220, REG 38034.508, VAR -4545.029, PARAM -5.000\n",
      "【TRAIN EPOCH 10000】LOSS -9024.488, COST -184.173, REG 274.687, VAR -14063.186, PARAM -0.000\n",
      "【TRAIN EPOCH 15000】LOSS -9122.794, COST -186.186, REG 0.066, VAR -14185.266, PARAM 5.000\n",
      "Early Stopping 18621\n"
     ]
    }
   ],
   "source": [
    "model, bit_strings, cost, reg_term, vari_term, runtime = gnn.fit_model_multishot(model,\n",
    "                                                                                 dgl_graph, \n",
    "                                                                                 embedding,\n",
    "                                                                                 loss,\n",
    "                                                                                 num_epoch=num_epoch,\n",
    "                                                                                 lr=lr, \n",
    "                                                                                 weight_decay=weight_decay,\n",
    "                                                                                 tol=tol, \n",
    "                                                                                 patience=patience, \n",
    "                                                                                 vari_param=0,\n",
    "                                                                                 device=device,\n",
    "                                                                                 annealing=True,\n",
    "                                                                                 init_reg_param=init_reg_param,\n",
    "                                                                                 annealing_rate=annealing_rate,\n",
    "                                                                                 check_interval=check_interval\n",
    "                                                                                )"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e3f2b570",
   "metadata": {},
   "source": [
    "### Visualize result"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "7d69faea",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-02-01T17:25:12.225218Z",
     "start_time": "2024-02-01T17:25:10.600150Z"
    }
   },
   "outputs": [],
   "source": [
    "mis_results=[]\n",
    "for shot in range(bit_strings.size(1)):\n",
    "    size_mis, _, number_violation = utils.postprocess_gnn_mis(bit_strings[:, shot], nx_graph)\n",
    "    mis_results.append([size_mis.item(), number_violation])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "9402868f",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-02-01T17:25:12.524086Z",
     "start_time": "2024-02-01T17:25:12.226530Z"
    }
   },
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 460.8x345.6 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "mis_results=np.array(mis_results)\n",
    "\n",
    "fig, ax_left = plt.subplots(figsize=(6.4, 4.8))\n",
    "ax_left.plot(np.array(penalties), mis_results[:, 0]/N, label=\"Independent Set\", color=\"tab:blue\")\n",
    "ax_right = ax_left.twinx()\n",
    "ax_right.plot(np.array(penalties), mis_results[:, 1]/N, label=\"Violation\", color=\"tab:orange\")\n",
    "ax_left.grid()\n",
    "ax_left.set_ylabel(\"Independent Set\")\n",
    "ax_right.set_ylabel(\"Violation\")\n",
    "ax_left.set_xlabel(r\"$\\lambda$\")\n",
    "\n",
    "h1, l1 = ax_right.get_legend_handles_labels()\n",
    "h2, l2 = ax_left.get_legend_handles_labels()\n",
    "ax_left.legend(h1+h2, l1+l2)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "34cbaf21",
   "metadata": {},
   "source": [
    "## Finding Heterogeneous Solutions"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "539c0f52",
   "metadata": {},
   "source": [
    "### Step1: Setting Hyperparameters"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "b6cccad1",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-02-01T17:25:12.632962Z",
     "start_time": "2024-02-01T17:25:12.525597Z"
    }
   },
   "outputs": [],
   "source": [
    "# load problem\n",
    "Q_data=np.load(\"data/maxcut/MCPg14.npz\")\n",
    "Adj= -1/2*Q_data['weight']\n",
    "nx_graph = nx.convert_matrix.from_numpy_array(Adj)\n",
    "Q_weight = torch.from_numpy(Q_data['weight'].astype(np.float32)).clone().to(device)\n",
    "Q_bias = torch.from_numpy(Q_data['bias'].astype(np.float32)).clone().to(device)\n",
    "\n",
    "# GNN Architecture\n",
    "dgl_graph = dgl.from_networkx(nx_graph).to(device)\n",
    "in_feats = int(dgl_graph.number_of_nodes()**(0.7))\n",
    "hidden_size=int(in_feats)\n",
    "dropout=0.0\n",
    "num_class=100\n",
    "\n",
    "model=gnn.GNNSage_dev(in_feats, \n",
    "                      hidden_size, \n",
    "                      num_class, \n",
    "                      dropout, \n",
    "                      device, \n",
    "                      agg_type='mean'\n",
    "                     ).to(device)\n",
    "\n",
    "embedding= nn.Embedding(dgl_graph.number_of_nodes(), \n",
    "                        in_feats\n",
    "                       ).type(torch_type).to(device)\n",
    "\n",
    "# Learning Parameters\n",
    "num_epoch=int(1e+5)\n",
    "lr=1e-4 \n",
    "weight_decay=1e-2\n",
    "tol=1e-5 \n",
    "patience=1000\n",
    "vari_param=0.4\n",
    "init_reg_param=-3\n",
    "annealing_rate=1e-3\n",
    "check_interval=5000"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c0069030",
   "metadata": {},
   "source": [
    "### Step2: Set Loss Function"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "39e171d0",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-02-01T17:25:12.637817Z",
     "start_time": "2024-02-01T17:25:12.634678Z"
    }
   },
   "outputs": [],
   "source": [
    "def loss(probs, reg_param, vari_param=0):\n",
    "    \n",
    "    # cost function\n",
    "    sum_cost= -0.5*torch.diag(probs.T @ Q_weight @ probs).sum()-(Q_bias @ probs).sum()\n",
    "    \n",
    "    # diversity measure\n",
    "    vari_term = -1*probs.size(1)*probs.std(dim=1).sum()\n",
    "    \n",
    "    # Annealed term\n",
    "    reg_term = torch.sum((1-(2*probs-1)**2))\n",
    "    \n",
    "    return sum_cost+reg_param*reg_term+vari_param*vari_term, sum_cost/probs.size(1), reg_term, vari_term"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "63143891",
   "metadata": {},
   "source": [
    "### Step3: Run CTRA-PI-GNN solver"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3cb0c142",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-02-01T17:21:04.106967Z",
     "start_time": "2024-02-01T17:20:58.889Z"
    }
   },
   "outputs": [],
   "source": [
    "model, bit_strings, cost, reg_term, vari_term, runtime = gnn.fit_model_multishot(model,\n",
    "                                                                                 dgl_graph, \n",
    "                                                                                 embedding,\n",
    "                                                                                 loss,\n",
    "                                                                                 num_epoch=num_epoch,\n",
    "                                                                                 lr=lr, \n",
    "                                                                                 weight_decay=weight_decay,\n",
    "                                                                                 tol=tol, \n",
    "                                                                                 patience=patience, \n",
    "                                                                                 vari_param=vari_param,\n",
    "                                                                                 device=device,\n",
    "                                                                                 annealing=True,\n",
    "                                                                                 init_reg_param=init_reg_param,\n",
    "                                                                                 annealing_rate=annealing_rate,\n",
    "                                                                                 check_interval=check_interval)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6b31e755",
   "metadata": {},
   "source": [
    "### Visualize results"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d6f0c690",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-02-01T17:21:04.107924Z",
     "start_time": "2024-02-01T17:20:59.550Z"
    }
   },
   "outputs": [],
   "source": [
    "maxcut_results=[]\n",
    "for shot in range(bit_strings.size(1)):\n",
    "    hamming_dis = torch.sum(bit_strings[:, 0]!=bit_strings[:, shot]).item()\n",
    "    size_max_cut, [S0, S1], cut_edges, uncut_edges = utils.postprocess_gnn_max_cut(bit_strings[:, shot], nx_graph)\n",
    "    maxcut_results.append([hamming_dis, size_max_cut])\n",
    "maxcut_results=np.array(maxcut_results)\n",
    "    \n",
    "# PCA\n",
    "data=bit_strings.detach().cpu().numpy().T\n",
    "ss = StandardScaler()\n",
    "standardized_data = ss.fit_transform(data)\n",
    "pca = PCA()\n",
    "pca.fit(standardized_data)\n",
    "feature = pca.transform(standardized_data)\n",
    "\n",
    "# CCR \n",
    "cr = pca.explained_variance_ratio_\n",
    "ccr = np.add.accumulate(cr)\n",
    "eigenvalue = pca.explained_variance_\n",
    "eigenvector = pca.components_\n",
    "\n",
    "# Vis\n",
    "fig, axes = plt.subplots(1, 3, figsize=(6.4*3, 4.8), constrained_layout=True)\n",
    "axes[0].hist(maxcut_results[:, 0], bins=20, color=\"tab:blue\", alpha=0.5)\n",
    "axes[0].grid()\n",
    "axes[0].set_xlabel(\"Hamming Distance\", fontsize=15)\n",
    "\n",
    "axes[1].plot(ccr[:10])\n",
    "axes[1].set_ylabel('CCR', fontsize=15)\n",
    "axes[1].set_xlabel('Axis', fontsize=15)\n",
    "axes[1].grid()\n",
    "\n",
    "axes[2].scatter(feature[:, 0], feature[:, 1], s=50, color=\"tab:blue\")\n",
    "axes[2].set_xlabel(\"PC1\")\n",
    "axes[2].set_ylabel(\"PC2\")\n",
    "axes[2].grid()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b53b6f06",
   "metadata": {},
   "source": [
    "## (Optional) Multi-Problem"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c5e5fab6",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-02-01T17:21:04.108620Z",
     "start_time": "2024-02-01T17:21:00.102Z"
    }
   },
   "outputs": [],
   "source": [
    "!ls data/matching"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2b1b283f",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-02-01T17:21:04.109581Z",
     "start_time": "2024-02-01T17:21:00.328Z"
    }
   },
   "outputs": [],
   "source": [
    "# load problems\n",
    "N1, N2 = 50, 50\n",
    "c_data=np.load(\"data/matching/data_c.npy\").reshape(-1, N1, N2)\n",
    "m_data=np.load(\"data/matching/data_m.npy\").reshape(-1, N1, N2)\n",
    "c_data = torch.from_numpy(c_data.astype(np.float32)).clone().to(device)\n",
    "m_data = torch.from_numpy(m_data.astype(np.float32)).clone().to(device)\n",
    "# set matching type\n",
    "p, q = 0.05, 0.05\n",
    "nx_graph = nx.complete_graph(N1*N2)\n",
    "dgl_graph = dgl.from_networkx(nx_graph).to(device)\n",
    "\n",
    "# GNN Architecture\n",
    "in_feats = 2500\n",
    "hidden_size=int(in_feats*0.65)\n",
    "dropout=0.0\n",
    "num_class=len(c_data)\n",
    "\n",
    "model=gnn.GNNSage_dev(in_feats, \n",
    "                      hidden_size, \n",
    "                      num_class, \n",
    "                      dropout, \n",
    "                      device, \n",
    "                      agg_type='mean'\n",
    "                     ).to(device)\n",
    "\n",
    "embedding= nn.Embedding(dgl_graph.number_of_nodes(), \n",
    "                        in_feats\n",
    "                       ).type(torch_type).to(device)\n",
    "\n",
    "# penalty parameterts\n",
    "con_param_1, con_param_2, con_param_3, con_param_4 = 12, 12, 2, 2"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5dddeb59",
   "metadata": {},
   "source": [
    "### Define loss function"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a5033ccb",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-02-01T17:21:04.110307Z",
     "start_time": "2024-02-01T17:21:01.894Z"
    }
   },
   "outputs": [],
   "source": [
    "def multi_loss_func(probs, \n",
    "                    reg_param, \n",
    "                    vari_param=0):\n",
    "    sum_cost=0\n",
    "    for i in range(len(c_data)):\n",
    "        probs_ = probs[:, i].view(N1, N2)\n",
    "        obj_fn = -1*torch.sum(probs_*c_data[i])\n",
    "        con_1=torch.sum(F.relu(probs_.sum(dim=1)-1))\n",
    "        con_2=torch.sum(F.relu(probs_.sum(dim=0)-1))\n",
    "        con_3=F.relu(p*torch.sum(probs_)-torch.sum(probs_*m_data[i]))\n",
    "        con_4=F.relu(q*torch.sum(probs_)-torch.sum(probs_*(1-m_data[i])))\n",
    "        cost=obj_fn+con_param_1*con_1+con_param_2*con_2+con_param_3*con_3+con_param_4*con_4 \n",
    "        sum_cost+= cost\n",
    "    vari_term = probs.size(1)*probs.std(dim=1).sum()\n",
    "    reg_term = torch.sum((1-(2*probs-1)**2))\n",
    "            \n",
    "    return sum_cost+reg_param*reg_term+vari_param*vari_term, sum_cost/probs.size(1), reg_term, vari_term\n",
    "\n",
    "\n",
    "\n",
    "model, bit_strings, cost, reg_term, vari_term, runtime = gnn.fit_model_multishot(model,\n",
    "                                                                                   dgl_graph, \n",
    "                                                                                   embedding,\n",
    "                                                                                   multi_loss_func,\n",
    "                                                                                   num_epoch=int(1e+5),\n",
    "                                                                                   lr=1e-4, \n",
    "                                                                                   weight_decay=1e-2,\n",
    "                                                                                   tol=1e-5, \n",
    "                                                                                   patience=1000, \n",
    "                                                                                   vari_param=0,\n",
    "                                                                                   device=device,\n",
    "                                                                                   annealing=True,\n",
    "                                                                                   init_reg_param=-15,\n",
    "                                                                                   annealing_rate=5e-4,\n",
    "                                                                                   check_interval=1000)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "88a89fb3",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.10"
  },
  "toc": {
   "base_numbering": 1,
   "nav_menu": {},
   "number_sections": false,
   "sideBar": true,
   "skip_h1_title": false,
   "title_cell": "Table of Contents",
   "title_sidebar": "Contents",
   "toc_cell": false,
   "toc_position": {},
   "toc_section_display": true,
   "toc_window_display": false
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
