{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "a6f294e0",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "import torch.nn.init as init\n",
    "\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import networkx as nx\n",
    "\n",
    "from pyqtorch.core.circuit import QuantumCircuit\n",
    "from pyqtorch.ansatz import AlternateLayerAnsatz\n",
    "from pyqtorch.embedding import SingleLayerEncoding\n",
    "from pyqtorch.core.operation import Z, RX"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "134a399c",
   "metadata": {},
   "source": [
    "## Fit a target function"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "2612ff0c",
   "metadata": {},
   "outputs": [],
   "source": [
    "def target_function(x, degree=3):\n",
    "    result = 0\n",
    "    for i in range(degree):\n",
    "        result += torch.cos(i*x) + torch.sin(i*x)\n",
    "    return .05 * result"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "0ecc511f",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x12c23ec10>]"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "x = torch.tensor(np.linspace(0, 10, 100))\n",
    "target_y = target_function(x, 5)\n",
    "plt.plot(x.numpy(), target_y.numpy())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "96c09bf3",
   "metadata": {},
   "outputs": [],
   "source": [
    "class Model(QuantumCircuit):\n",
    "    def __init__(self, n_qubits, n_layers):\n",
    "        super().__init__(n_qubits)\n",
    "        self.ansatz1 = AlternateLayerAnsatz(n_qubits, n_layers)\n",
    "        self.embedding = SingleLayerEncoding(n_qubits)\n",
    "        self.ansatz2 = AlternateLayerAnsatz(n_qubits, n_layers)\n",
    "        \n",
    "        \n",
    "    def forward(self, x):\n",
    "        batch_size = len(x)\n",
    "        state = self.init_state(batch_size)\n",
    "        \n",
    "        state = self.ansatz1(state)\n",
    "        state = self.embedding(state, x)\n",
    "        state = self.ansatz2(state)\n",
    "        \n",
    "        new_state = Z(state, [0], self.n_qubits)\n",
    "        \n",
    "        state = state.reshape((2**self.n_qubits, batch_size))\n",
    "        new_state = new_state.reshape((2**self.n_qubits, batch_size))\n",
    "        \n",
    "        return torch.real(torch.sum(torch.conj(state) * new_state, axis=0))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "d059e027",
   "metadata": {},
   "outputs": [],
   "source": [
    "n_qubits = 5\n",
    "n_layers = 3\n",
    "\n",
    "model = Model(n_qubits, n_layers)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "7eb3eafc",
   "metadata": {},
   "outputs": [],
   "source": [
    "with torch.no_grad():\n",
    "    y = model(x)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "ce29120e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([10])"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model(x[0:10]).shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "0b0bee1b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x12cdc5a00>]"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.plot(x.numpy(), y.numpy())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "8e4076b0",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 1 | Loss 0.029158090249084775\n",
      "Epoch 2 | Loss 0.01963727190047296\n",
      "Epoch 3 | Loss 0.012699240540944257\n",
      "Epoch 4 | Loss 0.008170924953350516\n",
      "Epoch 5 | Loss 0.005638678858704369\n",
      "Epoch 6 | Loss 0.00459087256581169\n",
      "Epoch 7 | Loss 0.004460629277062068\n",
      "Epoch 8 | Loss 0.004717230025665686\n",
      "Epoch 9 | Loss 0.0049786459483149225\n",
      "Epoch 10 | Loss 0.005036890321681123\n",
      "Epoch 11 | Loss 0.004827945440385019\n",
      "Epoch 12 | Loss 0.004382906171768449\n",
      "Epoch 13 | Loss 0.0037836222958799017\n",
      "Epoch 14 | Loss 0.0031279136944066368\n",
      "Epoch 15 | Loss 0.002505794537350578\n",
      "Epoch 16 | Loss 0.001985397355561911\n",
      "Epoch 17 | Loss 0.0016057230962374155\n",
      "Epoch 18 | Loss 0.0013748641774728166\n",
      "Epoch 19 | Loss 0.0012733792317645026\n",
      "Epoch 20 | Loss 0.001262303500662218\n",
      "Epoch 21 | Loss 0.0012944270136993508\n",
      "Epoch 22 | Loss 0.0013261049103359306\n",
      "Epoch 23 | Loss 0.0013260070341819202\n",
      "Epoch 24 | Loss 0.0012787993765996008\n",
      "Epoch 25 | Loss 0.0011840192220660924\n",
      "Epoch 26 | Loss 0.0010519213263825945\n",
      "Epoch 27 | Loss 0.0008987099834127907\n",
      "Epoch 28 | Loss 0.0007422645291149017\n",
      "Epoch 29 | Loss 0.0005989263349232255\n",
      "Epoch 30 | Loss 0.00048121968957965544\n",
      "Epoch 31 | Loss 0.0003962643129121823\n",
      "Epoch 32 | Loss 0.0003449883725964584\n",
      "Epoch 33 | Loss 0.00032232722729010684\n",
      "Epoch 34 | Loss 0.0003186390838323831\n",
      "Epoch 35 | Loss 0.00032212998559065617\n",
      "Epoch 36 | Loss 0.0003216665072806675\n",
      "Epoch 37 | Loss 0.00030935439853587343\n",
      "Epoch 38 | Loss 0.00028211030943607536\n",
      "Epoch 39 | Loss 0.0002418468743802618\n",
      "Epoch 40 | Loss 0.00019424008492743437\n",
      "Epoch 41 | Loss 0.0001465862225032511\n",
      "Epoch 42 | Loss 0.00010550229936319978\n",
      "Epoch 43 | Loss 7.522442357263626e-05\n",
      "Epoch 44 | Loss 5.6969666601762006e-05\n",
      "Epoch 45 | Loss 4.928490072305954e-05\n",
      "Epoch 46 | Loss 4.9026379467998475e-05\n",
      "Epoch 47 | Loss 5.248892450155978e-05\n",
      "Epoch 48 | Loss 5.632781939302459e-05\n",
      "Epoch 49 | Loss 5.814894205564367e-05\n",
      "Epoch 50 | Loss 5.674280010399324e-05\n",
      "Epoch 51 | Loss 5.205206679850265e-05\n",
      "Epoch 52 | Loss 4.492967074475458e-05\n",
      "Epoch 53 | Loss 3.678654750033269e-05\n",
      "Epoch 54 | Loss 2.9224459610522438e-05\n",
      "Epoch 55 | Loss 2.366307216217135e-05\n",
      "Epoch 56 | Loss 2.1003654380740067e-05\n",
      "Epoch 57 | Loss 2.1377348773144658e-05\n",
      "Epoch 58 | Loss 2.4069553517008815e-05\n",
      "Epoch 59 | Loss 2.7719450099656477e-05\n",
      "Epoch 60 | Loss 3.0763994620994334e-05\n",
      "Epoch 61 | Loss 3.195681536126707e-05\n",
      "Epoch 62 | Loss 3.076320818384408e-05\n",
      "Epoch 63 | Loss 2.745952842173007e-05\n",
      "Epoch 64 | Loss 2.292756616317949e-05\n",
      "Epoch 65 | Loss 1.8290627630207584e-05\n",
      "Epoch 66 | Loss 1.4526047678888764e-05\n",
      "Epoch 67 | Loss 1.2181634471913177e-05\n",
      "Epoch 68 | Loss 1.1283926456815322e-05\n",
      "Epoch 69 | Loss 1.1432561478703622e-05\n",
      "Epoch 70 | Loss 1.202779477984287e-05\n",
      "Epoch 71 | Loss 1.2518924363353579e-05\n",
      "Epoch 72 | Loss 1.2561685099044294e-05\n",
      "Epoch 73 | Loss 1.2061609484841753e-05\n",
      "Epoch 74 | Loss 1.111926966866214e-05\n",
      "Epoch 75 | Loss 9.942039233973366e-06\n",
      "Epoch 76 | Loss 8.766621645975841e-06\n",
      "Epoch 77 | Loss 7.789470636074813e-06\n",
      "Epoch 78 | Loss 7.125331659841794e-06\n",
      "Epoch 79 | Loss 6.786588641885452e-06\n",
      "Epoch 80 | Loss 6.694885351039274e-06\n",
      "Epoch 81 | Loss 6.710063268675386e-06\n",
      "Epoch 82 | Loss 6.677834034111654e-06\n",
      "Epoch 83 | Loss 6.477424281957403e-06\n",
      "Epoch 84 | Loss 6.064203494650899e-06\n",
      "Epoch 85 | Loss 5.486539924333125e-06\n",
      "Epoch 86 | Loss 4.864954799581828e-06\n",
      "Epoch 87 | Loss 4.342093579466845e-06\n",
      "Epoch 88 | Loss 4.02397217406979e-06\n",
      "Epoch 89 | Loss 3.942806181219968e-06\n",
      "Epoch 90 | Loss 4.053185238392467e-06\n",
      "Epoch 91 | Loss 4.258851257869863e-06\n",
      "Epoch 92 | Loss 4.453378442890067e-06\n",
      "Epoch 93 | Loss 4.557929883152337e-06\n",
      "Epoch 94 | Loss 4.542245184506449e-06\n",
      "Epoch 95 | Loss 4.422814711307702e-06\n",
      "Epoch 96 | Loss 4.245385889831442e-06\n",
      "Epoch 97 | Loss 4.060946401851772e-06\n",
      "Epoch 98 | Loss 3.9101786457294264e-06\n",
      "Epoch 99 | Loss 3.8142903793320013e-06\n",
      "Epoch 100 | Loss 3.774407029240378e-06\n",
      "Epoch 101 | Loss 3.7743435277035537e-06\n",
      "Epoch 102 | Loss 3.788110010139829e-06\n",
      "Epoch 103 | Loss 3.7891175613310776e-06\n",
      "Epoch 104 | Loss 3.7592067138580097e-06\n",
      "Epoch 105 | Loss 3.694031458558933e-06\n",
      "Epoch 106 | Loss 3.603504338195578e-06\n",
      "Epoch 107 | Loss 3.5067616904371577e-06\n",
      "Epoch 108 | Loss 3.4240403871023277e-06\n",
      "Epoch 109 | Loss 3.3675183389195726e-06\n",
      "Epoch 110 | Loss 3.337152083863462e-06\n",
      "Epoch 111 | Loss 3.322537200178122e-06\n",
      "Epoch 112 | Loss 3.3094165349318117e-06\n",
      "Epoch 113 | Loss 3.286552548869575e-06\n",
      "Epoch 114 | Loss 3.249909063399753e-06\n",
      "Epoch 115 | Loss 3.202399841595215e-06\n",
      "Epoch 116 | Loss 3.1514042035862253e-06\n",
      "Epoch 117 | Loss 3.1041642093686422e-06\n",
      "Epoch 118 | Loss 3.065231885907157e-06\n",
      "Epoch 119 | Loss 3.035133774872802e-06\n",
      "Epoch 120 | Loss 3.0115506220252195e-06\n",
      "Epoch 121 | Loss 2.9906909252619983e-06\n",
      "Epoch 122 | Loss 2.9688828522488453e-06\n",
      "Epoch 123 | Loss 2.9434554876851877e-06\n",
      "Epoch 124 | Loss 2.9136992487783555e-06\n",
      "Epoch 125 | Loss 2.8807478468976545e-06\n",
      "Epoch 126 | Loss 2.847273626448904e-06\n",
      "Epoch 127 | Loss 2.8162777887403406e-06\n",
      "Epoch 128 | Loss 2.7896726175146918e-06\n",
      "Epoch 129 | Loss 2.7677613086729764e-06\n",
      "Epoch 130 | Loss 2.7488212035823396e-06\n",
      "Epoch 131 | Loss 2.7302683971814475e-06\n",
      "Epoch 132 | Loss 2.7097620435598807e-06\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "Input \u001b[0;32mIn [12]\u001b[0m, in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[1;32m      4\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m epoch \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mrange\u001b[39m(epochs):\n\u001b[1;32m      5\u001b[0m     optimizer\u001b[38;5;241m.\u001b[39mzero_grad()\n\u001b[0;32m----> 6\u001b[0m     y_pred \u001b[38;5;241m=\u001b[39m \u001b[43mmodel\u001b[49m\u001b[43m(\u001b[49m\u001b[43mx\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m      7\u001b[0m     loss \u001b[38;5;241m=\u001b[39m F\u001b[38;5;241m.\u001b[39mmse_loss(target_y, y_pred)\n\u001b[1;32m      8\u001b[0m     loss\u001b[38;5;241m.\u001b[39mbackward()\n",
      "File \u001b[0;32m~/Desktop/QUANTUM_COMPUTING/qml-research-postdoc/lib/python3.8/site-packages/torch/nn/modules/module.py:1102\u001b[0m, in \u001b[0;36mModule._call_impl\u001b[0;34m(self, *input, **kwargs)\u001b[0m\n\u001b[1;32m   1098\u001b[0m \u001b[38;5;66;03m# If we don't have any hooks, we want to skip the rest of the logic in\u001b[39;00m\n\u001b[1;32m   1099\u001b[0m \u001b[38;5;66;03m# this function, and just call forward.\u001b[39;00m\n\u001b[1;32m   1100\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m (\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_backward_hooks\n\u001b[1;32m   1101\u001b[0m         \u001b[38;5;129;01mor\u001b[39;00m _global_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_forward_pre_hooks):\n\u001b[0;32m-> 1102\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mforward_call\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;28;43minput\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1103\u001b[0m \u001b[38;5;66;03m# Do not call functions when jit is used\u001b[39;00m\n\u001b[1;32m   1104\u001b[0m full_backward_hooks, non_full_backward_hooks \u001b[38;5;241m=\u001b[39m [], []\n",
      "Input \u001b[0;32mIn [7]\u001b[0m, in \u001b[0;36mModel.forward\u001b[0;34m(self, x)\u001b[0m\n\u001b[1;32m     13\u001b[0m state \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mansatz1(state)\n\u001b[1;32m     14\u001b[0m state \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39membedding(state, x)\n\u001b[0;32m---> 15\u001b[0m state \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mansatz2\u001b[49m\u001b[43m(\u001b[49m\u001b[43mstate\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     17\u001b[0m new_state \u001b[38;5;241m=\u001b[39m Z(state, [\u001b[38;5;241m0\u001b[39m], \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mn_qubits)\n\u001b[1;32m     19\u001b[0m state \u001b[38;5;241m=\u001b[39m state\u001b[38;5;241m.\u001b[39mreshape((\u001b[38;5;241m2\u001b[39m\u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39m\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mn_qubits, batch_size))\n",
      "File \u001b[0;32m~/Desktop/QUANTUM_COMPUTING/qml-research-postdoc/lib/python3.8/site-packages/torch/nn/modules/module.py:1102\u001b[0m, in \u001b[0;36mModule._call_impl\u001b[0;34m(self, *input, **kwargs)\u001b[0m\n\u001b[1;32m   1098\u001b[0m \u001b[38;5;66;03m# If we don't have any hooks, we want to skip the rest of the logic in\u001b[39;00m\n\u001b[1;32m   1099\u001b[0m \u001b[38;5;66;03m# this function, and just call forward.\u001b[39;00m\n\u001b[1;32m   1100\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m (\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_backward_hooks\n\u001b[1;32m   1101\u001b[0m         \u001b[38;5;129;01mor\u001b[39;00m _global_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_forward_pre_hooks):\n\u001b[0;32m-> 1102\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mforward_call\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;28;43minput\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1103\u001b[0m \u001b[38;5;66;03m# Do not call functions when jit is used\u001b[39;00m\n\u001b[1;32m   1104\u001b[0m full_backward_hooks, non_full_backward_hooks \u001b[38;5;241m=\u001b[39m [], []\n",
      "File \u001b[0;32m~/Desktop/QUANTUM_COMPUTING/qml-research-postdoc/qml-research/PyQ/pyqtorch/ansatz.py:101\u001b[0m, in \u001b[0;36mAlternateLayerAnsatz.forward\u001b[0;34m(self, state)\u001b[0m\n\u001b[1;32m     99\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21mforward\u001b[39m(\u001b[38;5;28mself\u001b[39m, state):\n\u001b[1;32m    100\u001b[0m     \u001b[38;5;28;01mfor\u001b[39;00m i, layer \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28menumerate\u001b[39m(\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mlayers):\n\u001b[0;32m--> 101\u001b[0m         state \u001b[38;5;241m=\u001b[39m \u001b[43mlayer\u001b[49m\u001b[43m(\u001b[49m\u001b[43mstate\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    102\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m state\n",
      "File \u001b[0;32m~/Desktop/QUANTUM_COMPUTING/qml-research-postdoc/lib/python3.8/site-packages/torch/nn/modules/module.py:1102\u001b[0m, in \u001b[0;36mModule._call_impl\u001b[0;34m(self, *input, **kwargs)\u001b[0m\n\u001b[1;32m   1098\u001b[0m \u001b[38;5;66;03m# If we don't have any hooks, we want to skip the rest of the logic in\u001b[39;00m\n\u001b[1;32m   1099\u001b[0m \u001b[38;5;66;03m# this function, and just call forward.\u001b[39;00m\n\u001b[1;32m   1100\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m (\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_backward_hooks\n\u001b[1;32m   1101\u001b[0m         \u001b[38;5;129;01mor\u001b[39;00m _global_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_forward_pre_hooks):\n\u001b[0;32m-> 1102\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mforward_call\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;28;43minput\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1103\u001b[0m \u001b[38;5;66;03m# Do not call functions when jit is used\u001b[39;00m\n\u001b[1;32m   1104\u001b[0m full_backward_hooks, non_full_backward_hooks \u001b[38;5;241m=\u001b[39m [], []\n",
      "File \u001b[0;32m~/Desktop/QUANTUM_COMPUTING/qml-research-postdoc/qml-research/PyQ/pyqtorch/ansatz.py:85\u001b[0m, in \u001b[0;36mOneLayerEntanglingAnsatz.forward\u001b[0;34m(self, state)\u001b[0m\n\u001b[1;32m     84\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21mforward\u001b[39m(\u001b[38;5;28mself\u001b[39m, state):\n\u001b[0;32m---> 85\u001b[0m     state \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mparam_layer\u001b[49m\u001b[43m(\u001b[49m\u001b[43mstate\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     86\u001b[0m     \u001b[38;5;28;01mfor\u001b[39;00m i \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mrange\u001b[39m(\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mn_qubits):\n\u001b[1;32m     87\u001b[0m         state \u001b[38;5;241m=\u001b[39m CNOT(state, [i \u001b[38;5;241m%\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mn_qubits, (i\u001b[38;5;241m+\u001b[39m\u001b[38;5;241m1\u001b[39m) \u001b[38;5;241m%\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mn_qubits],\n\u001b[1;32m     88\u001b[0m                             \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mn_qubits)\n",
      "File \u001b[0;32m~/Desktop/QUANTUM_COMPUTING/qml-research-postdoc/lib/python3.8/site-packages/torch/nn/modules/module.py:1102\u001b[0m, in \u001b[0;36mModule._call_impl\u001b[0;34m(self, *input, **kwargs)\u001b[0m\n\u001b[1;32m   1098\u001b[0m \u001b[38;5;66;03m# If we don't have any hooks, we want to skip the rest of the logic in\u001b[39;00m\n\u001b[1;32m   1099\u001b[0m \u001b[38;5;66;03m# this function, and just call forward.\u001b[39;00m\n\u001b[1;32m   1100\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m (\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_backward_hooks\n\u001b[1;32m   1101\u001b[0m         \u001b[38;5;129;01mor\u001b[39;00m _global_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_forward_pre_hooks):\n\u001b[0;32m-> 1102\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mforward_call\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;28;43minput\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1103\u001b[0m \u001b[38;5;66;03m# Do not call functions when jit is used\u001b[39;00m\n\u001b[1;32m   1104\u001b[0m full_backward_hooks, non_full_backward_hooks \u001b[38;5;241m=\u001b[39m [], []\n",
      "File \u001b[0;32m~/Desktop/QUANTUM_COMPUTING/qml-research-postdoc/qml-research/PyQ/pyqtorch/ansatz.py:41\u001b[0m, in \u001b[0;36mOneLayerRotation.forward\u001b[0;34m(self, state)\u001b[0m\n\u001b[1;32m     39\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39marbitrary:\n\u001b[1;32m     40\u001b[0m     \u001b[38;5;28;01mfor\u001b[39;00m i, t \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28menumerate\u001b[39m(\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mtheta):\n\u001b[0;32m---> 41\u001b[0m         state \u001b[38;5;241m=\u001b[39m \u001b[43mU\u001b[49m\u001b[43m(\u001b[49m\u001b[43mt\u001b[49m\u001b[43m[\u001b[49m\u001b[38;5;241;43m0\u001b[39;49m\u001b[43m]\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mt\u001b[49m\u001b[43m[\u001b[49m\u001b[38;5;241;43m1\u001b[39;49m\u001b[43m]\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mt\u001b[49m\u001b[43m[\u001b[49m\u001b[38;5;241;43m2\u001b[39;49m\u001b[43m]\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mstate\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m[\u001b[49m\u001b[43mi\u001b[49m\u001b[43m]\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mn_qubits\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     42\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m state\n",
      "File \u001b[0;32m~/Desktop/QUANTUM_COMPUTING/qml-research-postdoc/qml-research/PyQ/pyqtorch/core/operation.py:59\u001b[0m, in \u001b[0;36mU\u001b[0;34m(phi, theta, omega, state, qubits, N_qubits)\u001b[0m\n\u001b[1;32m     50\u001b[0m t_minus \u001b[38;5;241m=\u001b[39m torch\u001b[38;5;241m.\u001b[39mexp(\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39mj \u001b[38;5;241m*\u001b[39m (phi \u001b[38;5;241m-\u001b[39m omega) \u001b[38;5;241m/\u001b[39m \u001b[38;5;241m2\u001b[39m)\n\u001b[1;32m     51\u001b[0m mat \u001b[38;5;241m=\u001b[39m torch\u001b[38;5;241m.\u001b[39mtensor([[\u001b[38;5;241m1\u001b[39m, \u001b[38;5;241m0\u001b[39m], [\u001b[38;5;241m0\u001b[39m, \u001b[38;5;241m0\u001b[39m]], dtype\u001b[38;5;241m=\u001b[39mtorch\u001b[38;5;241m.\u001b[39mcdouble)\u001b[38;5;241m.\u001b[39mto(dev) \\\n\u001b[1;32m     52\u001b[0m     \u001b[38;5;241m*\u001b[39m torch\u001b[38;5;241m.\u001b[39mcos(theta\u001b[38;5;241m/\u001b[39m\u001b[38;5;241m2\u001b[39m) \u001b[38;5;241m*\u001b[39m t_plus \u001b[38;5;241m-\u001b[39m\\\n\u001b[1;32m     53\u001b[0m     torch\u001b[38;5;241m.\u001b[39mtensor([[\u001b[38;5;241m0\u001b[39m, \u001b[38;5;241m1\u001b[39m], [\u001b[38;5;241m0\u001b[39m, \u001b[38;5;241m0\u001b[39m]], dtype\u001b[38;5;241m=\u001b[39mtorch\u001b[38;5;241m.\u001b[39mcdouble)\u001b[38;5;241m.\u001b[39mto(dev) \\\n\u001b[0;32m   (...)\u001b[0m\n\u001b[1;32m     57\u001b[0m     torch\u001b[38;5;241m.\u001b[39mtensor([[\u001b[38;5;241m0\u001b[39m, \u001b[38;5;241m0\u001b[39m], [\u001b[38;5;241m0\u001b[39m, \u001b[38;5;241m1\u001b[39m]], dtype\u001b[38;5;241m=\u001b[39mtorch\u001b[38;5;241m.\u001b[39mcdouble)\u001b[38;5;241m.\u001b[39mto(dev) \\\n\u001b[1;32m     58\u001b[0m     \u001b[38;5;241m*\u001b[39m torch\u001b[38;5;241m.\u001b[39mcos(theta\u001b[38;5;241m/\u001b[39m\u001b[38;5;241m2\u001b[39m) \u001b[38;5;241m*\u001b[39m torch\u001b[38;5;241m.\u001b[39mconj(t_plus)\n\u001b[0;32m---> 59\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43m_apply_gate\u001b[49m\u001b[43m(\u001b[49m\u001b[43mstate\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mmat\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mqubits\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mN_qubits\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m~/Desktop/QUANTUM_COMPUTING/qml-research-postdoc/qml-research/PyQ/pyqtorch/core/utils.py:46\u001b[0m, in \u001b[0;36m_apply_gate\u001b[0;34m(state, mat, qubits, N_qubits)\u001b[0m\n\u001b[1;32m     41\u001b[0m axes \u001b[38;5;241m=\u001b[39m (mat_dims, state_dims)\n\u001b[1;32m     43\u001b[0m state \u001b[38;5;241m=\u001b[39m torch\u001b[38;5;241m.\u001b[39mtensordot(mat, state, dims\u001b[38;5;241m=\u001b[39maxes)\n\u001b[1;32m     44\u001b[0m inv_perm \u001b[38;5;241m=\u001b[39m torch\u001b[38;5;241m.\u001b[39margsort(\n\u001b[1;32m     45\u001b[0m     torch\u001b[38;5;241m.\u001b[39mtensor(\n\u001b[0;32m---> 46\u001b[0m         state_dims \u001b[38;5;241m+\u001b[39m [j \u001b[38;5;28;01mfor\u001b[39;00m j \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mrange\u001b[39m(N_qubits\u001b[38;5;241m+\u001b[39m\u001b[38;5;241m1\u001b[39m) \u001b[38;5;28;01mif\u001b[39;00m j \u001b[38;5;129;01mnot\u001b[39;00m \u001b[38;5;129;01min\u001b[39;00m state_dims]\n\u001b[1;32m     47\u001b[0m         )\n\u001b[1;32m     48\u001b[0m     )\n\u001b[1;32m     49\u001b[0m state \u001b[38;5;241m=\u001b[39m torch\u001b[38;5;241m.\u001b[39mpermute(state, \u001b[38;5;28mtuple\u001b[39m(inv_perm))\n\u001b[1;32m     50\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m state\n",
      "File \u001b[0;32m~/Desktop/QUANTUM_COMPUTING/qml-research-postdoc/qml-research/PyQ/pyqtorch/core/utils.py:46\u001b[0m, in \u001b[0;36m<listcomp>\u001b[0;34m(.0)\u001b[0m\n\u001b[1;32m     41\u001b[0m axes \u001b[38;5;241m=\u001b[39m (mat_dims, state_dims)\n\u001b[1;32m     43\u001b[0m state \u001b[38;5;241m=\u001b[39m torch\u001b[38;5;241m.\u001b[39mtensordot(mat, state, dims\u001b[38;5;241m=\u001b[39maxes)\n\u001b[1;32m     44\u001b[0m inv_perm \u001b[38;5;241m=\u001b[39m torch\u001b[38;5;241m.\u001b[39margsort(\n\u001b[1;32m     45\u001b[0m     torch\u001b[38;5;241m.\u001b[39mtensor(\n\u001b[0;32m---> 46\u001b[0m         state_dims \u001b[38;5;241m+\u001b[39m [j \u001b[38;5;28;01mfor\u001b[39;00m j \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mrange\u001b[39m(N_qubits\u001b[38;5;241m+\u001b[39m\u001b[38;5;241m1\u001b[39m) \u001b[38;5;28;01mif\u001b[39;00m j \u001b[38;5;129;01mnot\u001b[39;00m \u001b[38;5;129;01min\u001b[39;00m state_dims]\n\u001b[1;32m     47\u001b[0m         )\n\u001b[1;32m     48\u001b[0m     )\n\u001b[1;32m     49\u001b[0m state \u001b[38;5;241m=\u001b[39m torch\u001b[38;5;241m.\u001b[39mpermute(state, \u001b[38;5;28mtuple\u001b[39m(inv_perm))\n\u001b[1;32m     50\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m state\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": [
    "optimizer = torch.optim.Adam(model.parameters(), lr=.01)\n",
    "epochs = 200\n",
    "\n",
    "for epoch in range(epochs):\n",
    "    optimizer.zero_grad()\n",
    "    y_pred = model(x)\n",
    "    loss = F.mse_loss(target_y, y_pred)\n",
    "    loss.backward()\n",
    "    optimizer.step()\n",
    "    print(f\"Epoch {epoch+1} | Loss {loss}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3dd9cb34",
   "metadata": {},
   "outputs": [],
   "source": [
    "with torch.no_grad():\n",
    "    y = model(x)\n",
    "\n",
    "plt.plot(x.numpy(), target_y.numpy())    \n",
    "plt.plot(x.numpy(), y.numpy())"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "eb4585dc",
   "metadata": {},
   "source": [
    "## Solve MIS for QAOA"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 119,
   "id": "3c4e17c3",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "n_nodes = 10\n",
    "\n",
    "graph = nx.gnp_random_graph(n_nodes, .25, seed=42)\n",
    "nx.draw(graph, with_labels=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 120,
   "id": "a1b223ab",
   "metadata": {},
   "outputs": [],
   "source": [
    "from pyqtorch.matrices import generate_ising_from_graph, sum_N\n",
    "ising_matrix = generate_ising_from_graph(graph, type_ising='N')\n",
    "ising_cost = 1.2*ising_matrix - sum_N(n_nodes)\n",
    "\n",
    "ising_matrix = ising_matrix.reshape([2] * n_nodes + [1])\n",
    "ising_cost = ising_cost.reshape([2] * n_nodes + [1])\n",
    "\n",
    "\n",
    "class MIS(QuantumCircuit):\n",
    "    def __init__(self, n_qubits, n_layers):\n",
    "        super().__init__(n_qubits)\n",
    "        self.beta = nn.Parameter(torch.empty(n_layers,))\n",
    "        self.gamma = nn.Parameter(torch.empty(n_layers,))\n",
    "        self.reset_parameters()\n",
    "        \n",
    "    def reset_parameters(self):\n",
    "        init.uniform_(self.beta, -2 * np.pi, 2 * np.pi)\n",
    "        init.uniform_(self.gamma, -2 * np.pi, 2 * np.pi)\n",
    "        \n",
    "    def forward(self, return_cost=False):\n",
    "        state = self.uniform_state()\n",
    "        for b, g in zip(self.beta, self.gamma):\n",
    "            state = state * torch.exp(-1j * g * ising_matrix)\n",
    "            for i in range(self.n_qubits):\n",
    "                state = RX(b, state, [i], self.n_qubits)\n",
    "        if return_cost:\n",
    "            return torch.real(torch.sum(torch.abs(state)**2 * ising_cost))\n",
    "        else:\n",
    "            state = state.reshape((2**self.n_qubits,))\n",
    "            return torch.abs(state)**2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 121,
   "id": "c84f30d0",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 1 | Loss -0.5732532441966612\n",
      "Epoch 2 | Loss -0.6331619277058087\n",
      "Epoch 3 | Loss -0.6861459060709989\n",
      "Epoch 4 | Loss -0.7387500437209391\n",
      "Epoch 5 | Loss -0.7908989563587623\n",
      "Epoch 6 | Loss -0.837905298263808\n",
      "Epoch 7 | Loss -0.871331795290423\n",
      "Epoch 8 | Loss -0.8930770787756188\n",
      "Epoch 9 | Loss -0.9130871085937311\n",
      "Epoch 10 | Loss -0.9358536689128184\n",
      "Epoch 11 | Loss -0.9618579564057501\n",
      "Epoch 12 | Loss -0.9904167554884878\n",
      "Epoch 13 | Loss -1.0202172139050867\n",
      "Epoch 14 | Loss -1.0499487912478633\n",
      "Epoch 15 | Loss -1.07848446622836\n",
      "Epoch 16 | Loss -1.1069801480141952\n",
      "Epoch 17 | Loss -1.1382739034007834\n",
      "Epoch 18 | Loss -1.1748206428748973\n",
      "Epoch 19 | Loss -1.216481697010161\n",
      "Epoch 20 | Loss -1.2625000406431954\n",
      "Epoch 21 | Loss -1.3131403397317842\n",
      "Epoch 22 | Loss -1.3682629604322365\n",
      "Epoch 23 | Loss -1.4265116790764234\n",
      "Epoch 24 | Loss -1.4869205071502807\n",
      "Epoch 25 | Loss -1.5482089175740015\n",
      "Epoch 26 | Loss -1.6079342020331695\n",
      "Epoch 27 | Loss -1.6654217311512285\n",
      "Epoch 28 | Loss -1.7228935868765234\n",
      "Epoch 29 | Loss -1.7819442738690865\n",
      "Epoch 30 | Loss -1.8422240827066538\n",
      "Epoch 31 | Loss -1.903854197244454\n",
      "Epoch 32 | Loss -1.9670357776619865\n",
      "Epoch 33 | Loss -2.029420527504279\n",
      "Epoch 34 | Loss -2.0874651616825233\n",
      "Epoch 35 | Loss -2.139010658271825\n",
      "Epoch 36 | Loss -2.1816697566566168\n",
      "Epoch 37 | Loss -2.214839323532247\n",
      "Epoch 38 | Loss -2.2437345465331444\n",
      "Epoch 39 | Loss -2.2732522417925742\n",
      "Epoch 40 | Loss -2.3045199110029624\n",
      "Epoch 41 | Loss -2.335991307905369\n",
      "Epoch 42 | Loss -2.3654059594192622\n",
      "Epoch 43 | Loss -2.3924014581768587\n",
      "Epoch 44 | Loss -2.4178039979182033\n",
      "Epoch 45 | Loss -2.4418094445998335\n",
      "Epoch 46 | Loss -2.4645377556003805\n",
      "Epoch 47 | Loss -2.486862781704482\n",
      "Epoch 48 | Loss -2.509494217150208\n",
      "Epoch 49 | Loss -2.532223971192445\n",
      "Epoch 50 | Loss -2.5543286802827607\n",
      "Epoch 51 | Loss -2.5755040978622556\n",
      "Epoch 52 | Loss -2.5959247767488085\n",
      "Epoch 53 | Loss -2.6168223780367934\n",
      "Epoch 54 | Loss -2.6401507597998046\n",
      "Epoch 55 | Loss -2.6671632040200386\n",
      "Epoch 56 | Loss -2.698024620202698\n",
      "Epoch 57 | Loss -2.7317653153233685\n",
      "Epoch 58 | Loss -2.7663450069026734\n",
      "Epoch 59 | Loss -2.7995436635264492\n",
      "Epoch 60 | Loss -2.830108574737726\n",
      "Epoch 61 | Loss -2.858060787317405\n",
      "Epoch 62 | Loss -2.8846920824146896\n",
      "Epoch 63 | Loss -2.9123478656113346\n",
      "Epoch 64 | Loss -2.9430002578496985\n",
      "Epoch 65 | Loss -2.9770300643281478\n",
      "Epoch 66 | Loss -3.013659798250645\n",
      "Epoch 67 | Loss -3.0524794748583632\n",
      "Epoch 68 | Loss -3.0933273890107547\n",
      "Epoch 69 | Loss -3.135379355168163\n",
      "Epoch 70 | Loss -3.177408697111498\n",
      "Epoch 71 | Loss -3.219268846163864\n",
      "Epoch 72 | Loss -3.2616090727361984\n",
      "Epoch 73 | Loss -3.3036158229335006\n",
      "Epoch 74 | Loss -3.3429573005070417\n",
      "Epoch 75 | Loss -3.3774802047736356\n",
      "Epoch 76 | Loss -3.406513992244255\n",
      "Epoch 77 | Loss -3.4311495808606622\n",
      "Epoch 78 | Loss -3.4529890918277895\n",
      "Epoch 79 | Loss -3.4733140944953274\n",
      "Epoch 80 | Loss -3.4927338961101717\n",
      "Epoch 81 | Loss -3.5112121770779376\n",
      "Epoch 82 | Loss -3.5280557264730854\n",
      "Epoch 83 | Loss -3.542702574211204\n",
      "Epoch 84 | Loss -3.5552480705153844\n",
      "Epoch 85 | Loss -3.5659005047928654\n",
      "Epoch 86 | Loss -3.57529060745066\n",
      "Epoch 87 | Loss -3.584682248932425\n",
      "Epoch 88 | Loss -3.5948564065610755\n",
      "Epoch 89 | Loss -3.60539701817406\n",
      "Epoch 90 | Loss -3.615406297217887\n",
      "Epoch 91 | Loss -3.62421545883701\n",
      "Epoch 92 | Loss -3.6324055138773748\n",
      "Epoch 93 | Loss -3.642075306049329\n",
      "Epoch 94 | Loss -3.654727640490542\n",
      "Epoch 95 | Loss -3.6702721347672584\n",
      "Epoch 96 | Loss -3.6884009069240054\n",
      "Epoch 97 | Loss -3.708816911613087\n",
      "Epoch 98 | Loss -3.7301209311043895\n",
      "Epoch 99 | Loss -3.750114525310617\n",
      "Epoch 100 | Loss -3.766397091686126\n",
      "Epoch 101 | Loss -3.777961598394133\n",
      "Epoch 102 | Loss -3.7856750983622525\n",
      "Epoch 103 | Loss -3.790990927898397\n",
      "Epoch 104 | Loss -3.7959068720711504\n",
      "Epoch 105 | Loss -3.802195710188496\n",
      "Epoch 106 | Loss -3.810003987467331\n",
      "Epoch 107 | Loss -3.8180882149142894\n",
      "Epoch 108 | Loss -3.824505808438644\n",
      "Epoch 109 | Loss -3.828359142597277\n",
      "Epoch 110 | Loss -3.8298577309087136\n",
      "Epoch 111 | Loss -3.8297605041112086\n",
      "Epoch 112 | Loss -3.829621264092229\n",
      "Epoch 113 | Loss -3.8302630635439363\n",
      "Epoch 114 | Loss -3.831781747848858\n",
      "Epoch 115 | Loss -3.833626682021628\n",
      "Epoch 116 | Loss -3.835165986396417\n",
      "Epoch 117 | Loss -3.835924805089763\n",
      "Epoch 118 | Loss -3.8359962529677967\n",
      "Epoch 119 | Loss -3.8359408295287283\n",
      "Epoch 120 | Loss -3.8361216084847163\n",
      "Epoch 121 | Loss -3.8367722064925474\n",
      "Epoch 122 | Loss -3.8377634302536543\n",
      "Epoch 123 | Loss -3.8386964325428066\n",
      "Epoch 124 | Loss -3.839337340589755\n",
      "Epoch 125 | Loss -3.8397712564602355\n",
      "Epoch 126 | Loss -3.8401698922877046\n",
      "Epoch 127 | Loss -3.840696803401842\n",
      "Epoch 128 | Loss -3.8414218275720144\n",
      "Epoch 129 | Loss -3.8421439127127464\n",
      "Epoch 130 | Loss -3.8427370672765933\n",
      "Epoch 131 | Loss -3.8431648282787743\n",
      "Epoch 132 | Loss -3.843479251257832\n",
      "Epoch 133 | Loss -3.8437792323640343\n",
      "Epoch 134 | Loss -3.844129689858902\n",
      "Epoch 135 | Loss -3.8445280845156713\n",
      "Epoch 136 | Loss -3.8449173025738066\n",
      "Epoch 137 | Loss -3.8453002743660587\n",
      "Epoch 138 | Loss -3.845582137539323\n",
      "Epoch 139 | Loss -3.8457878690235168\n",
      "Epoch 140 | Loss -3.8459416085300653\n",
      "Epoch 141 | Loss -3.846098840565765\n",
      "Epoch 142 | Loss -3.8462643963647833\n",
      "Epoch 143 | Loss -3.846437717411682\n",
      "Epoch 144 | Loss -3.8465906236346905\n",
      "Epoch 145 | Loss -3.846744158491276\n",
      "Epoch 146 | Loss -3.846894570259552\n",
      "Epoch 147 | Loss -3.847067459474414\n",
      "Epoch 148 | Loss -3.8472854760945254\n",
      "Epoch 149 | Loss -3.8474996712941896\n",
      "Epoch 150 | Loss -3.8476798122219904\n",
      "Epoch 151 | Loss -3.847826842586244\n",
      "Epoch 152 | Loss -3.8479582597149613\n",
      "Epoch 153 | Loss -3.848085963656255\n",
      "Epoch 154 | Loss -3.8482035908005257\n",
      "Epoch 155 | Loss -3.8483460310522557\n",
      "Epoch 156 | Loss -3.848498645075941\n",
      "Epoch 157 | Loss -3.848657770105738\n",
      "Epoch 158 | Loss -3.8488105376729687\n",
      "Epoch 159 | Loss -3.8489498511405387\n",
      "Epoch 160 | Loss -3.8490736550687266\n",
      "Epoch 161 | Loss -3.8491983490121155\n",
      "Epoch 162 | Loss -3.849331084146137\n",
      "Epoch 163 | Loss -3.849460915615366\n",
      "Epoch 164 | Loss -3.8495613385635785\n",
      "Epoch 165 | Loss -3.849656034887012\n",
      "Epoch 166 | Loss -3.849764860413446\n",
      "Epoch 167 | Loss -3.8498739447525185\n",
      "Epoch 168 | Loss -3.84997609181712\n",
      "Epoch 169 | Loss -3.8500922225482235\n",
      "Epoch 170 | Loss -3.8502107971476676\n",
      "Epoch 171 | Loss -3.850318893250295\n",
      "Epoch 172 | Loss -3.850426441494681\n",
      "Epoch 173 | Loss -3.85053329072659\n",
      "Epoch 174 | Loss -3.850642250960127\n",
      "Epoch 175 | Loss -3.8507510193178907\n",
      "Epoch 176 | Loss -3.850867462926387\n",
      "Epoch 177 | Loss -3.8509686645027608\n",
      "Epoch 178 | Loss -3.8510848901222556\n",
      "Epoch 179 | Loss -3.8511844927671612\n",
      "Epoch 180 | Loss -3.85129319957778\n",
      "Epoch 181 | Loss -3.8513923878826013\n",
      "Epoch 182 | Loss -3.8515090335412596\n",
      "Epoch 183 | Loss -3.851620201900618\n",
      "Epoch 184 | Loss -3.8517140417200593\n",
      "Epoch 185 | Loss -3.8518019979938334\n",
      "Epoch 186 | Loss -3.8519232388205262\n",
      "Epoch 187 | Loss -3.8520229483361073\n",
      "Epoch 188 | Loss -3.8521220777534206\n",
      "Epoch 189 | Loss -3.852236230766047\n",
      "Epoch 190 | Loss -3.852325133128978\n",
      "Epoch 191 | Loss -3.8524173163653153\n",
      "Epoch 192 | Loss -3.8525226076238512\n",
      "Epoch 193 | Loss -3.852618658715106\n",
      "Epoch 194 | Loss -3.85271674505149\n",
      "Epoch 195 | Loss -3.8528268998111233\n",
      "Epoch 196 | Loss -3.852917344083436\n",
      "Epoch 197 | Loss -3.8530059253034796\n",
      "Epoch 198 | Loss -3.8531109264762597\n",
      "Epoch 199 | Loss -3.8532071308362306\n",
      "Epoch 200 | Loss -3.8532988216415935\n",
      "Epoch 201 | Loss -3.853383575681073\n",
      "Epoch 202 | Loss -3.8534717625293733\n",
      "Epoch 203 | Loss -3.8535644471652826\n",
      "Epoch 204 | Loss -3.853662947148461\n",
      "Epoch 205 | Loss -3.853741386968651\n",
      "Epoch 206 | Loss -3.8538342140413047\n",
      "Epoch 207 | Loss -3.853930325744768\n",
      "Epoch 208 | Loss -3.854011703626777\n",
      "Epoch 209 | Loss -3.854100310262834\n",
      "Epoch 210 | Loss -3.8541875260083343\n",
      "Epoch 211 | Loss -3.8542676005281327\n",
      "Epoch 212 | Loss -3.854348232883051\n",
      "Epoch 213 | Loss -3.854433499327516\n",
      "Epoch 214 | Loss -3.8545127961502033\n",
      "Epoch 215 | Loss -3.854593153455613\n",
      "Epoch 216 | Loss -3.854663798999782\n",
      "Epoch 217 | Loss -3.8547508487813342\n",
      "Epoch 218 | Loss -3.8548326486031015\n",
      "Epoch 219 | Loss -3.85491117137819\n",
      "Epoch 220 | Loss -3.854978152193107\n",
      "Epoch 221 | Loss -3.855050689574175\n",
      "Epoch 222 | Loss -3.8551265402780808\n",
      "Epoch 223 | Loss -3.8552075816686684\n",
      "Epoch 224 | Loss -3.8552685301232765\n",
      "Epoch 225 | Loss -3.8553416590475647\n",
      "Epoch 226 | Loss -3.855413540873078\n",
      "Epoch 227 | Loss -3.85548250950459\n",
      "Epoch 228 | Loss -3.8555315923685782\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 229 | Loss -3.8556002788880503\n",
      "Epoch 230 | Loss -3.8556702242968113\n",
      "Epoch 231 | Loss -3.8557397619387617\n",
      "Epoch 232 | Loss -3.855787357659043\n",
      "Epoch 233 | Loss -3.8558528606578406\n",
      "Epoch 234 | Loss -3.855908551948124\n",
      "Epoch 235 | Loss -3.8559790832821648\n",
      "Epoch 236 | Loss -3.85602197083628\n",
      "Epoch 237 | Loss -3.856087011774503\n",
      "Epoch 238 | Loss -3.856149030355737\n",
      "Epoch 239 | Loss -3.856194977844848\n",
      "Epoch 240 | Loss -3.856246964331209\n",
      "Epoch 241 | Loss -3.8562947685058098\n",
      "Epoch 242 | Loss -3.856356285616368\n",
      "Epoch 243 | Loss -3.85640079049384\n",
      "Epoch 244 | Loss -3.8564570867878145\n",
      "Epoch 245 | Loss -3.8565076890675907\n",
      "Epoch 246 | Loss -3.8565586764820132\n",
      "Epoch 247 | Loss -3.8566003272640543\n",
      "Epoch 248 | Loss -3.856639317811813\n",
      "Epoch 249 | Loss -3.856685540156676\n",
      "Epoch 250 | Loss -3.856750855033597\n",
      "Epoch 251 | Loss -3.8567809806533666\n",
      "Epoch 252 | Loss -3.8568095227734744\n",
      "Epoch 253 | Loss -3.8568700812106864\n",
      "Epoch 254 | Loss -3.8569119894676294\n",
      "Epoch 255 | Loss -3.8569549482185206\n",
      "Epoch 256 | Loss -3.856999587388335\n",
      "Epoch 257 | Loss -3.8570264984639193\n",
      "Epoch 258 | Loss -3.857072292950036\n",
      "Epoch 259 | Loss -3.857096267084849\n",
      "Epoch 260 | Loss -3.8571282796367323\n",
      "Epoch 261 | Loss -3.857179145554766\n",
      "Epoch 262 | Loss -3.8572166263905716\n",
      "Epoch 263 | Loss -3.857249417243498\n",
      "Epoch 264 | Loss -3.857283373139674\n",
      "Epoch 265 | Loss -3.8573217960107105\n",
      "Epoch 266 | Loss -3.8573593989388835\n",
      "Epoch 267 | Loss -3.8573776837679064\n",
      "Epoch 268 | Loss -3.8574143650048778\n",
      "Epoch 269 | Loss -3.857452025888603\n",
      "Epoch 270 | Loss -3.85748669095141\n",
      "Epoch 271 | Loss -3.857512760361034\n",
      "Epoch 272 | Loss -3.8575469865991883\n",
      "Epoch 273 | Loss -3.857572010549603\n",
      "Epoch 274 | Loss -3.857608824554851\n",
      "Epoch 275 | Loss -3.857635068025892\n",
      "Epoch 276 | Loss -3.857670138873038\n",
      "Epoch 277 | Loss -3.857685385112746\n",
      "Epoch 278 | Loss -3.8577108629869143\n",
      "Epoch 279 | Loss -3.8577476432115354\n",
      "Epoch 280 | Loss -3.8577713475208815\n",
      "Epoch 281 | Loss -3.857794095766014\n",
      "Epoch 282 | Loss -3.857823719237979\n",
      "Epoch 283 | Loss -3.8578583207663595\n",
      "Epoch 284 | Loss -3.8578823447361223\n",
      "Epoch 285 | Loss -3.8578965035800277\n",
      "Epoch 286 | Loss -3.857922283028047\n",
      "Epoch 287 | Loss -3.8579625301399756\n",
      "Epoch 288 | Loss -3.8579741477546974\n",
      "Epoch 289 | Loss -3.8579978476129924\n",
      "Epoch 290 | Loss -3.858020963273323\n",
      "Epoch 291 | Loss -3.8580446092715324\n",
      "Epoch 292 | Loss -3.85806401134204\n",
      "Epoch 293 | Loss -3.858094807306739\n",
      "Epoch 294 | Loss -3.8581109501036024\n",
      "Epoch 295 | Loss -3.8581283361580834\n",
      "Epoch 296 | Loss -3.8581571248083857\n",
      "Epoch 297 | Loss -3.8581833195938446\n",
      "Epoch 298 | Loss -3.8581957051538107\n",
      "Epoch 299 | Loss -3.858221248857694\n",
      "Epoch 300 | Loss -3.858238965771343\n",
      "Epoch 301 | Loss -3.85825927259558\n",
      "Epoch 302 | Loss -3.858281133990117\n",
      "Epoch 303 | Loss -3.8582957472137216\n",
      "Epoch 304 | Loss -3.8583154833224316\n",
      "Epoch 305 | Loss -3.8583621820360197\n",
      "Epoch 306 | Loss -3.8583598586382912\n",
      "Epoch 307 | Loss -3.8583867225384925\n",
      "Epoch 308 | Loss -3.8584055420080787\n",
      "Epoch 309 | Loss -3.85841801658652\n",
      "Epoch 310 | Loss -3.8584308801372473\n",
      "Epoch 311 | Loss -3.858448621170673\n",
      "Epoch 312 | Loss -3.858464001368375\n",
      "Epoch 313 | Loss -3.85850795113789\n",
      "Epoch 314 | Loss -3.85851617293394\n",
      "Epoch 315 | Loss -3.8585305895681796\n",
      "Epoch 316 | Loss -3.858549453221202\n",
      "Epoch 317 | Loss -3.858562903571709\n",
      "Epoch 318 | Loss -3.858581606001019\n",
      "Epoch 319 | Loss -3.858605076236751\n",
      "Epoch 320 | Loss -3.8586156385378225\n",
      "Epoch 321 | Loss -3.8586276095900525\n",
      "Epoch 322 | Loss -3.858634616181968\n",
      "Epoch 323 | Loss -3.85865184235708\n",
      "Epoch 324 | Loss -3.8586848496740296\n",
      "Epoch 325 | Loss -3.8587041226537773\n",
      "Epoch 326 | Loss -3.8587095879858078\n",
      "Epoch 327 | Loss -3.85873480432753\n",
      "Epoch 328 | Loss -3.858749290958265\n",
      "Epoch 329 | Loss -3.8587586457980683\n",
      "Epoch 330 | Loss -3.8587766823654093\n",
      "Epoch 331 | Loss -3.85880307723474\n",
      "Epoch 332 | Loss -3.858814652591597\n",
      "Epoch 333 | Loss -3.858829306218825\n",
      "Epoch 334 | Loss -3.8588318238739103\n",
      "Epoch 335 | Loss -3.858847969808389\n",
      "Epoch 336 | Loss -3.8588626765200393\n",
      "Epoch 337 | Loss -3.858888184044434\n",
      "Epoch 338 | Loss -3.858895588686649\n",
      "Epoch 339 | Loss -3.8589134492755566\n",
      "Epoch 340 | Loss -3.8589390925830003\n",
      "Epoch 341 | Loss -3.858945223164386\n",
      "Epoch 342 | Loss -3.858947946449783\n",
      "Epoch 343 | Loss -3.858965133450141\n",
      "Epoch 344 | Loss -3.85898676434869\n",
      "Epoch 345 | Loss -3.8590054937881066\n",
      "Epoch 346 | Loss -3.8590205498397863\n",
      "Epoch 347 | Loss -3.8590234826862155\n",
      "Epoch 348 | Loss -3.859038174667139\n",
      "Epoch 349 | Loss -3.859060043896902\n",
      "Epoch 350 | Loss -3.859081396790295\n",
      "Epoch 351 | Loss -3.859095372662346\n",
      "Epoch 352 | Loss -3.8591088740533723\n",
      "Epoch 353 | Loss -3.859123925122521\n",
      "Epoch 354 | Loss -3.859128143895987\n",
      "Epoch 355 | Loss -3.8591364915979325\n",
      "Epoch 356 | Loss -3.8591511264435656\n",
      "Epoch 357 | Loss -3.8591639332895418\n",
      "Epoch 358 | Loss -3.8591849825455244\n",
      "Epoch 359 | Loss -3.8591908087748603\n",
      "Epoch 360 | Loss -3.859218897896648\n",
      "Epoch 361 | Loss -3.8592193031679236\n",
      "Epoch 362 | Loss -3.8592385854971902\n",
      "Epoch 363 | Loss -3.859237599034579\n",
      "Epoch 364 | Loss -3.859256647714974\n",
      "Epoch 365 | Loss -3.859273172116167\n",
      "Epoch 366 | Loss -3.8592829737115695\n",
      "Epoch 367 | Loss -3.859302517841072\n",
      "Epoch 368 | Loss -3.859324722646724\n",
      "Epoch 369 | Loss -3.859317948384433\n",
      "Epoch 370 | Loss -3.859329809512116\n",
      "Epoch 371 | Loss -3.859341019441805\n",
      "Epoch 372 | Loss -3.8593544934657498\n",
      "Epoch 373 | Loss -3.8593853150708366\n",
      "Epoch 374 | Loss -3.8593856090902117\n",
      "Epoch 375 | Loss -3.859393134874282\n",
      "Epoch 376 | Loss -3.859406468794894\n",
      "Epoch 377 | Loss -3.8594168958402983\n",
      "Epoch 378 | Loss -3.859444176368588\n",
      "Epoch 379 | Loss -3.8594469868473027\n",
      "Epoch 380 | Loss -3.8594580868066535\n",
      "Epoch 381 | Loss -3.85947814130974\n",
      "Epoch 382 | Loss -3.8594974187009634\n",
      "Epoch 383 | Loss -3.859496298914425\n",
      "Epoch 384 | Loss -3.8595118637784767\n",
      "Epoch 385 | Loss -3.859505168721703\n",
      "Epoch 386 | Loss -3.859531518142186\n",
      "Epoch 387 | Loss -3.8595465656809917\n",
      "Epoch 388 | Loss -3.8595551207411942\n",
      "Epoch 389 | Loss -3.8595713643208627\n",
      "Epoch 390 | Loss -3.8595921416887453\n",
      "Epoch 391 | Loss -3.859585554714165\n",
      "Epoch 392 | Loss -3.8596064457288564\n",
      "Epoch 393 | Loss -3.859607581518442\n",
      "Epoch 394 | Loss -3.8596233983631434\n",
      "Epoch 395 | Loss -3.8596464710009677\n",
      "Epoch 396 | Loss -3.8596491202957486\n",
      "Epoch 397 | Loss -3.8596662003608304\n",
      "Epoch 398 | Loss -3.8596626757278583\n",
      "Epoch 399 | Loss -3.8596884384102452\n",
      "Epoch 400 | Loss -3.859694764557984\n",
      "Epoch 401 | Loss -3.859707732820337\n",
      "Epoch 402 | Loss -3.859713524354684\n",
      "Epoch 403 | Loss -3.8597288003810686\n",
      "Epoch 404 | Loss -3.85974232751754\n",
      "Epoch 405 | Loss -3.8597655385644734\n",
      "Epoch 406 | Loss -3.8597632774374473\n",
      "Epoch 407 | Loss -3.8597892069568145\n",
      "Epoch 408 | Loss -3.8597916093642994\n",
      "Epoch 409 | Loss -3.8598033847482602\n",
      "Epoch 410 | Loss -3.8598170925013306\n",
      "Epoch 411 | Loss -3.8598286849651013\n",
      "Epoch 412 | Loss -3.859839051290174\n",
      "Epoch 413 | Loss -3.859841350773728\n",
      "Epoch 414 | Loss -3.8598517614996894\n",
      "Epoch 415 | Loss -3.8598671986055457\n",
      "Epoch 416 | Loss -3.859867259906823\n",
      "Epoch 417 | Loss -3.8599012155625774\n",
      "Epoch 418 | Loss -3.8598992074380627\n",
      "Epoch 419 | Loss -3.8599098287330866\n",
      "Epoch 420 | Loss -3.859927896149771\n",
      "Epoch 421 | Loss -3.8599409210911872\n",
      "Epoch 422 | Loss -3.859953247605484\n",
      "Epoch 423 | Loss -3.8599646128732368\n",
      "Epoch 424 | Loss -3.8599851992157377\n",
      "Epoch 425 | Loss -3.859980319447289\n",
      "Epoch 426 | Loss -3.86000943711975\n",
      "Epoch 427 | Loss -3.860019658292264\n",
      "Epoch 428 | Loss -3.8600359436357365\n",
      "Epoch 429 | Loss -3.8600370634156747\n",
      "Epoch 430 | Loss -3.8600420604873955\n",
      "Epoch 431 | Loss -3.8600523170359198\n",
      "Epoch 432 | Loss -3.8600749869416338\n",
      "Epoch 433 | Loss -3.8600806602574407\n",
      "Epoch 434 | Loss -3.8600865410527883\n",
      "Epoch 435 | Loss -3.8601038998873074\n",
      "Epoch 436 | Loss -3.8601221999781528\n",
      "Epoch 437 | Loss -3.860128817417655\n",
      "Epoch 438 | Loss -3.860133715790824\n",
      "Epoch 439 | Loss -3.86015615503648\n",
      "Epoch 440 | Loss -3.8601651634283645\n",
      "Epoch 441 | Loss -3.8601819083620694\n",
      "Epoch 442 | Loss -3.860192935492262\n",
      "Epoch 443 | Loss -3.860199828683956\n",
      "Epoch 444 | Loss -3.860218073212605\n",
      "Epoch 445 | Loss -3.8602327046388853\n",
      "Epoch 446 | Loss -3.860244886921317\n",
      "Epoch 447 | Loss -3.8602534631045904\n",
      "Epoch 448 | Loss -3.860270283755775\n",
      "Epoch 449 | Loss -3.8602776533313183\n",
      "Epoch 450 | Loss -3.860295404560247\n",
      "Epoch 451 | Loss -3.860292129749123\n",
      "Epoch 452 | Loss -3.860297396300463\n",
      "Epoch 453 | Loss -3.860320513777754\n",
      "Epoch 454 | Loss -3.8603315459897227\n",
      "Epoch 455 | Loss -3.8603431612663828\n",
      "Epoch 456 | Loss -3.8603513311020663\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 457 | Loss -3.8603652654393548\n",
      "Epoch 458 | Loss -3.860379034149818\n",
      "Epoch 459 | Loss -3.860395506170957\n",
      "Epoch 460 | Loss -3.8604199476694996\n",
      "Epoch 461 | Loss -3.8604303610786594\n",
      "Epoch 462 | Loss -3.860433595254598\n",
      "Epoch 463 | Loss -3.8604462858221567\n",
      "Epoch 464 | Loss -3.8604585978331585\n",
      "Epoch 465 | Loss -3.860470547209214\n",
      "Epoch 466 | Loss -3.860475278985073\n",
      "Epoch 467 | Loss -3.860499993061843\n",
      "Epoch 468 | Loss -3.8604999772066186\n",
      "Epoch 469 | Loss -3.860519143877391\n",
      "Epoch 470 | Loss -3.8605213093125537\n",
      "Epoch 471 | Loss -3.8605339637319864\n",
      "Epoch 472 | Loss -3.8605590022655294\n",
      "Epoch 473 | Loss -3.8605701833610406\n",
      "Epoch 474 | Loss -3.860571593935339\n",
      "Epoch 475 | Loss -3.8605963534980554\n",
      "Epoch 476 | Loss -3.8606083434297647\n",
      "Epoch 477 | Loss -3.860627952727394\n",
      "Epoch 478 | Loss -3.8606395800096034\n",
      "Epoch 479 | Loss -3.860646969627177\n",
      "Epoch 480 | Loss -3.860651549380925\n",
      "Epoch 481 | Loss -3.860679817140586\n",
      "Epoch 482 | Loss -3.8606895163756865\n",
      "Epoch 483 | Loss -3.860702095904913\n",
      "Epoch 484 | Loss -3.8607259019687055\n",
      "Epoch 485 | Loss -3.8607297083651226\n",
      "Epoch 486 | Loss -3.8607464848275965\n",
      "Epoch 487 | Loss -3.8607553123689806\n",
      "Epoch 488 | Loss -3.8607600404556597\n",
      "Epoch 489 | Loss -3.8607884884806345\n",
      "Epoch 490 | Loss -3.8607973161377314\n",
      "Epoch 491 | Loss -3.860809254020524\n",
      "Epoch 492 | Loss -3.8608174992337965\n",
      "Epoch 493 | Loss -3.8608423434043355\n",
      "Epoch 494 | Loss -3.8608646856561424\n",
      "Epoch 495 | Loss -3.860868186070417\n",
      "Epoch 496 | Loss -3.86088729170062\n",
      "Epoch 497 | Loss -3.8608939569302803\n",
      "Epoch 498 | Loss -3.860916043399121\n",
      "Epoch 499 | Loss -3.8609211688463803\n",
      "Epoch 500 | Loss -3.8609458993635863\n"
     ]
    }
   ],
   "source": [
    "model = MIS(n_nodes, 20)\n",
    "\n",
    "optimizer = torch.optim.Adam(model.parameters(), lr=.02)\n",
    "epochs = 500\n",
    "\n",
    "\n",
    "for epoch in range(epochs):\n",
    "    optimizer.zero_grad()\n",
    "    loss = model(True)\n",
    "    loss.backward()\n",
    "    optimizer.step()\n",
    "    print(f\"Epoch {epoch+1} | Loss {loss}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 122,
   "id": "4f7a6f49",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'0000110101'"
      ]
     },
     "execution_count": 122,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "prob = model()\n",
    "mis = torch.argmax(prob)\n",
    "format(mis, '010b')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 81,
   "id": "7bf6d8b0",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(54)"
      ]
     },
     "execution_count": 81,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "mis"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 57,
   "id": "bfd64d73",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'1'"
      ]
     },
     "execution_count": 57,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "\"{0:b}\".format(1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 108,
   "id": "15ce4608",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(4.+0.j, dtype=torch.complex128)"
      ]
     },
     "execution_count": 108,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "sum_N(n_nodes)[mis]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8cb69112",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.8.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
