{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "104521cb",
   "metadata": {},
   "source": [
    "### Initialize libraries, functions and MiP-CRIP solver"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b36ab8ca",
   "metadata": {},
   "outputs": [],
   "source": [
    "import networkx as nx\n",
    "import pandas as pd\n",
    "import numpy as np\n",
    "import time\n",
    "from mip_crip import MiP_CRIP, eng2cut, data2graph\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3f4a59f7",
   "metadata": {},
   "source": [
    "### Sherrington-Kirkpatrick (SK) Model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ef98ff7a",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Generate SK model\n",
    "n = 1000  # number of spins\n",
    "J_mat = np.random.randn(n, n).astype(np.float32)\n",
    "J_mat = (J_mat + J_mat.T) / 2  # Make it symmetric\n",
    "np.fill_diagonal(J_mat, 0)  # Zero diagonal\n",
    "\n",
    "print(f\"SK model generated with {n} spins.\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "89d3ee2b",
   "metadata": {},
   "source": [
    "### Complete Graph ($K_n$)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "87ca27c1",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Generate Ising model from K_n graph\n",
    "n = 2000  # number of spins\n",
    "J_mat = np.random.randn(n, n).astype(np.float32)\n",
    "J_mat = J_mat + J_mat.T  # Make it symmetric\n",
    "J_mat = np.sign(J_mat)  # Ensure entries are -1 or +1\n",
    "np.fill_diagonal(J_mat, 0)  # Zero diagonal\n",
    "\n",
    "print(f\"Complete graph Ising model generated with {n} spins.\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8f898e23",
   "metadata": {},
   "source": [
    "### G-set graph"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e6380bb0",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Generate Ising model from G10 graph for MAX-CUT problem\n",
    "G, Adj_mat = data2graph('G10_graph.txt')  # Load the complex G-set graph G10\n",
    "J_mat = -Adj_mat\n",
    "n = J_mat.shape[0]\n",
    "\n",
    "print(f\"G10 graph loaded with {n} spins.\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "69a061ac",
   "metadata": {},
   "source": [
    "### Number Partitioning Problem (NPP)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c5ec4ca0",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Generate Ising model for uniform NPP\n",
    "n = 1000\n",
    "N_vec = np.random.rand(n).flatten()\n",
    "J_mat = - np.outer(N_vec, N_vec)\n",
    "np.fill_diagonal(J_mat, 0)\n",
    "\n",
    "print(f\"Uniform NPP Ising model generated with {n} spins.\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "eb1c8bcd",
   "metadata": {},
   "source": [
    "### Run MiP-CRIP"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6330f167",
   "metadata": {},
   "outputs": [],
   "source": [
    "# initialize hyperparameters for tuning\n",
    "\n",
    "rounds = 10      # tuning rounds\n",
    "tune_res = 3     # tuning grid-resolution\n",
    "\n",
    "K_outer = 200   # epochs\n",
    "T_inner = 10    # inner iterations for ADAM optimizer\n",
    "\n",
    "gamma0 = 0.1\n",
    "beta0 = 10\n",
    "step0 = 1\n",
    "\n",
    "sigma_noise = 1e-3\n",
    "\n",
    "best_energy = np.inf\n",
    "best_cut = 0\n",
    "best_sync = 0\n",
    "best_params = None\n",
    "\n",
    "n = J_mat.shape[0]\n",
    "J_sum = np.sum(J_mat)\n",
    "x0 = np.random.randn(n)\n",
    "\n",
    "Steps = np.linspace(0, step0, tune_res)[1:]\n",
    "Gamma = np.linspace(0, gamma0, tune_res)[1:]\n",
    "Beta = np.linspace(0, beta0, tune_res)[1:]\n",
    "\n",
    "for round_num in range(rounds):\n",
    "    for k, step in enumerate(Steps):\n",
    "        for i, gamma in enumerate(Gamma):\n",
    "            for j, beta in enumerate(Beta):\n",
    "\n",
    "                lambda0 = np.sqrt(gamma / (2 * beta))\n",
    "\n",
    "                for lambda_ in np.linspace(0, lambda0, tune_res)[1:]:\n",
    "                    for alpha in np.linspace(3 * beta * lambda_**2, beta * lambda_**2 + gamma, tune_res):\n",
    "\n",
    "                        start_time = time.time()\n",
    "\n",
    "                        spin_vec = MiP_CRIP(J_mat, x0, T=T_inner, K=K_outer,\n",
    "                                                alpha=alpha, beta=beta, lambda_=lambda_,\n",
    "                                                step=step,          # Adam learning rate\n",
    "                                                beta1=0.09, beta2=0.999, eps=1e-8,\n",
    "                                                sigma_noise=sigma_noise,\n",
    "                                                rng=None, return_all=False)\n",
    "                        \n",
    "                        t_elapsed = time.time() - start_time\n",
    "                        \n",
    "                        energy = -0.5 * spin_vec.T @ J_mat @ spin_vec\n",
    "                        sync = np.mean(spin_vec == np.sign(J_mat @ spin_vec))\n",
    "                        cut_value = eng2cut(energy, J_sum)\n",
    "                        print(f\"Round: {round_num+1}/{rounds} | alpha: {alpha:.4f} beta: {beta:.4f} gamma: {gamma:.4f} lambda: {lambda_:.4f} | learning rate: {step:.4f} | \"\n",
    "                              f\"energy: {energy:.2f} (best:{best_energy:.2f}) | cut: {cut_value:.2f} (best:{best_cut:.2f}) | sync: {sync:.3f} (best:{best_sync:.3f}) | runtime: {t_elapsed*1000:.1f}ms\", end='\\r')\n",
    "\n",
    "                        if energy < best_energy:\n",
    "                            best_spin = spin_vec.copy()\n",
    "                            best_energy = energy\n",
    "                            best_cut = cut_value\n",
    "                            best_sync = sync\n",
    "                            best_params = {\n",
    "                                \"alpha\": alpha,\n",
    "                                \"gamma\": gamma,\n",
    "                                \"beta\": beta,\n",
    "                                \"lambda_\": lambda_,\n",
    "                                \"step\": step,\n",
    "                                'i': i,\n",
    "                                'j': j,\n",
    "                                'k': k\n",
    "                            }\n",
    "\n",
    "    # grid refinement\n",
    "    if best_params is None:\n",
    "        continue\n",
    "\n",
    "    i, j, k = best_params[\"i\"], best_params[\"j\"], best_params[\"k\"]\n",
    "\n",
    "    # refine gamma\n",
    "    if i == 0:\n",
    "        Gamma = np.linspace(max(1e-6, Gamma[i]/2), Gamma[i], tune_res)\n",
    "    elif i == len(Gamma) - 1:\n",
    "        Gamma = np.linspace(Gamma[i], 2 * Gamma[i], tune_res)\n",
    "    else:\n",
    "        Gamma = np.linspace(Gamma[i - 1], Gamma[i + 1], tune_res)\n",
    "    \n",
    "    # refine beta\n",
    "    if j == 0:\n",
    "        Beta = np.linspace(max(1e-6, Beta[j]/2), Beta[j], tune_res)\n",
    "    elif j == len(Beta) - 1:\n",
    "        Beta = np.linspace(Beta[j], 2 * Beta[j], tune_res)\n",
    "    else:\n",
    "        Beta = np.linspace(Beta[j - 1], Beta[j + 1], tune_res)\n",
    "\n",
    "    # refine step (learning rate)\n",
    "    if k == 0:\n",
    "        Steps = np.linspace(max(1e-6, Steps[k]/2), Steps[k], tune_res)\n",
    "    elif k == len(Steps) - 1:\n",
    "        Steps = np.linspace(Steps[k], 2 * Steps[k], tune_res)\n",
    "    else:\n",
    "        Steps = np.linspace(Steps[k - 1], Steps[k + 1], tune_res)\n",
    "\n",
    "    print(\"\\n\")\n",
    "\n",
    "print(f\"Tuned parameters: alpha: {best_params['alpha']:.4f}, beta: {best_params['beta']:.4f}, gamma: {best_params['gamma']:.4f}, lambda={best_params['lambda_']:.4f}, learning rate: {best_params['step']:.4f}\")\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "90962dfa",
   "metadata": {},
   "source": [
    "### Show results"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4ede96d0",
   "metadata": {},
   "outputs": [],
   "source": [
    "print(f\"Best energy: {best_energy:.3f}, Best cut: {best_cut:.3f}, Best sync: {best_sync:.3f}\")\n",
    "\n",
    "# # uncomment for NPP discrepancy results\n",
    "# print(f\"Best Discrepency: {np.abs(best_spin @ N_vec):.5f}, Best sync: {best_sync:.3f}\")\n"
   ]
  }
 ],
 "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.9.0"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}