{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "9d236be4",
   "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, RY, X\n",
    "from pyqtorch.core.measurement import total_magnetization\n",
    "\n",
    "from pyqtorch.matrices_sparse import general_hamiltonian, get_sparse_torch"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "79b5a0d7",
   "metadata": {},
   "outputs": [],
   "source": [
    "sigma_x = torch.tensor([[0, 1],\n",
    "                        [1, 0]], \n",
    "                       dtype=torch.cdouble)\n",
    "\n",
    "sigma_y = torch.tensor([[0, -1j],\n",
    "                        [1j, 0]], \n",
    "                       dtype=torch.cdouble)\n",
    "\n",
    "sigma_z = torch.tensor([[1, 0],\n",
    "                        [0, 1]], \n",
    "                       dtype=torch.cdouble)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "3acf79b4",
   "metadata": {},
   "outputs": [],
   "source": [
    "def RY_(theta):\n",
    "    return torch.tensor([[torch.cos(theta/2),-torch.sin(theta/2)],\n",
    "                         [torch.sin(theta/2),torch.cos(theta/2)]])\n",
    "\n",
    "def RX_(theta):\n",
    "    return torch.tensor([[torch.cos(theta/2),-1j*torch.sin(theta/2)],\n",
    "                         [-1j*torch.sin(theta/2),torch.cos(theta/2)]])\n",
    "\n",
    "def RZ_(theta):\n",
    "    return torch.tensor([[torch.exp(-1j*theta/2),0],\n",
    "                         [0,torch.exp(1j*theta/2)]])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 241,
   "id": "5b60ea60",
   "metadata": {},
   "outputs": [],
   "source": [
    "def eigenvalues_from_hamiltonian(H):\n",
    "    Lambda, _ = torch.eig(H, eigenvectors=True)\n",
    "    return np.unique(np.round(Lambda.real,8))\n",
    "\n",
    "def hamiltonian_to_unitary(H, theta=1):\n",
    "    Lambda, V = torch.eig(H, eigenvectors=True)\n",
    "    U = torch.matmul(\n",
    "            torch.matmul(V,torch.diag(torch.exp(-1j*theta/2*Lambda))),\n",
    "            torch.conj(V).T) #U = V exp(-i*Lamda) V^T\n",
    "    return U"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 186,
   "id": "66041dc0",
   "metadata": {},
   "outputs": [],
   "source": [
    "theta = torch.pi/2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 189,
   "id": "f851c8a9",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[ 0.7071+2.2204e-16j, -0.7071-5.5511e-17j],\n",
       "        [ 0.7071+5.5511e-17j,  0.7071+0.0000e+00j]], dtype=torch.complex128)"
      ]
     },
     "execution_count": 189,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "Ry = hamiltonian_to_unitary(sigma_y, theta)\n",
    "Ry"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 181,
   "id": "696a8756",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[ 0.7071, -0.7071],\n",
       "        [ 0.7071,  0.7071]])"
      ]
     },
     "execution_count": 181,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "RY_(torch.tensor(theta))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 190,
   "id": "125effa4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[7.0711e-01+2.2204e-16j, 5.5511e-17-7.0711e-01j],\n",
       "        [5.5511e-17-7.0711e-01j, 7.0711e-01+0.0000e+00j]],\n",
       "       dtype=torch.complex128)"
      ]
     },
     "execution_count": 190,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "Rx = hamiltonian_to_unitary(sigma_x, theta)\n",
    "Rx"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 182,
   "id": "e4a4be2e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[0.7071+0.0000j, 0.0000-0.7071j],\n",
       "        [0.0000-0.7071j, 0.7071+0.0000j]])"
      ]
     },
     "execution_count": 182,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "RX_(torch.tensor(theta))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 191,
   "id": "e9bbf351",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[0.7071-0.7071j, 0.0000+0.0000j],\n",
       "        [0.0000+0.0000j, 0.7071-0.7071j]], dtype=torch.complex128)"
      ]
     },
     "execution_count": 191,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "Rz = hamiltonian_to_unitary(sigma_z, theta)\n",
    "Rz"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 192,
   "id": "973f57d0",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[0.7071-0.7071j, 0.0000+0.0000j],\n",
       "        [0.0000+0.0000j, 0.7071+0.7071j]])"
      ]
     },
     "execution_count": 192,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "RZ_(torch.tensor(theta))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "a757a6a9",
   "metadata": {},
   "outputs": [],
   "source": [
    "IMAT = torch.eye(2, dtype=torch.cdouble)\n",
    "XMAT = torch.tensor([[0, 1], [1, 0]], dtype=torch.cdouble)\n",
    "YMAT = torch.tensor([[0, -1j], [1j, 0]], dtype=torch.cdouble)\n",
    "ZMAT = torch.tensor([[1, 0], [0, -1]], dtype=torch.cdouble)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "ee870e9f",
   "metadata": {},
   "outputs": [],
   "source": [
    "def XX(N, i=0, j=0, device='cpu'):\n",
    "    if i == j:\n",
    "        return torch.ones(2**N).to(device)\n",
    "    op_list = [XMAT.to(device) if k in [i, j]\n",
    "               else IMAT.to(device) for k in range(N)]\n",
    "    operator = op_list[0]\n",
    "    for op in op_list[1::]:\n",
    "        operator = torch.kron(operator, op)\n",
    "    return operator    \n",
    "    \n",
    "def YY(N, i=0, j=0, device='cpu'):\n",
    "    if i == j:\n",
    "        return torch.ones(2**N).to(device)\n",
    "    op_list = [YMAT.to(device) if k in [i, j]\n",
    "               else IMAT.to(device) for k in range(N)]\n",
    "    operator = op_list[0]\n",
    "    for op in op_list[1::]:\n",
    "        operator = torch.kron(operator, op)\n",
    "    return operator\n",
    "    \n",
    "def ZZ(N, i=0, j=0, device='cpu'):\n",
    "    if i == j:\n",
    "        return torch.ones(2**N).to(device)\n",
    "\n",
    "    op_list = [ZMAT.to(device) if k in [i, j]\n",
    "               else IMAT.to(device) for k in range(N)]\n",
    "    operator = op_list[0]\n",
    "    for op in op_list[1::]:\n",
    "        operator = torch.kron(operator, op)\n",
    "\n",
    "    return operator\n",
    "\n",
    "\n",
    "\n",
    "def general_hamiltonian(graph, alpha, beta, gamma, device='cpu'):\n",
    "    #alpha, beta, gamma: matrices of parameters (e.g. alpha_ij)\n",
    "    #connectivity_graph: which qubits are connected\n",
    "    #no 1 body terms\n",
    "    \n",
    "    N = graph.number_of_nodes()\n",
    "    # construct the hamiltonian\n",
    "    H = torch.zeros((2**N,2**N), dtype=torch.cdouble).to(device)\n",
    "\n",
    "    for edge in graph.edges.data():\n",
    "        H += alpha[edge[0], edge[1]] * XX(N, edge[0], edge[1], device) #alpha_ij * XX\n",
    "        H += beta[edge[0], edge[1]]  * YY(N, edge[0], edge[1], device) #beta_ij * YY\n",
    "        H += gamma[edge[0], edge[1]] * ZZ(N, edge[0], edge[1], device) #gamma_ij * ZZ\n",
    "\n",
    "    return H\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "056ff281",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[ 1.+0.j,  0.+0.j,  0.+0.j,  0.+0.j],\n",
       "        [ 0.+0.j, -1.+0.j,  0.+0.j, -0.+0.j],\n",
       "        [ 0.+0.j,  0.+0.j, -1.+0.j, -0.+0.j],\n",
       "        [ 0.+0.j, -0.+0.j, -0.+0.j,  1.-0.j]], dtype=torch.complex128)"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "ZZ(2,0,1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "3a03dbd8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[0.+0.j, 0.+0.j, 0.+0.j, 1.+0.j],\n",
       "        [0.+0.j, 0.+0.j, 1.+0.j, 0.+0.j],\n",
       "        [0.+0.j, 1.+0.j, 0.+0.j, 0.+0.j],\n",
       "        [1.+0.j, 0.+0.j, 0.+0.j, 0.+0.j]], dtype=torch.complex128)"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "XX(2,0,1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "a66982ae",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[ 0.+0.j,  0.-0.j,  0.-0.j, -1.+0.j],\n",
       "        [ 0.+0.j,  0.+0.j,  1.-0.j,  0.-0.j],\n",
       "        [ 0.+0.j,  1.-0.j,  0.+0.j,  0.-0.j],\n",
       "        [-1.+0.j,  0.+0.j,  0.+0.j,  0.+0.j]], dtype=torch.complex128)"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "YY(2,0,1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "41b39705",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "n_nodes = 8\n",
    "\n",
    "graph = nx.gnp_random_graph(n_nodes, p=.4, seed=42)\n",
    "nx.draw(graph, with_labels=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "c9992001",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[15.+0.j,  0.+0.j,  0.+0.j,  ...,  0.+0.j,  0.+0.j,  0.+0.j],\n",
       "        [ 0.+0.j,  9.+0.j,  2.+0.j,  ...,  0.+0.j,  0.+0.j,  0.+0.j],\n",
       "        [ 0.+0.j,  2.+0.j,  7.+0.j,  ...,  0.+0.j,  0.+0.j,  0.+0.j],\n",
       "        ...,\n",
       "        [ 0.+0.j,  0.+0.j,  0.+0.j,  ...,  7.+0.j,  2.+0.j,  0.+0.j],\n",
       "        [ 0.+0.j,  0.+0.j,  0.+0.j,  ...,  2.+0.j,  9.+0.j,  0.+0.j],\n",
       "        [ 0.+0.j,  0.+0.j,  0.+0.j,  ...,  0.+0.j,  0.+0.j, 15.+0.j]],\n",
       "       dtype=torch.complex128)"
      ]
     },
     "execution_count": 22,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "alpha = np.array([[1,2,3],\n",
    "                     [2,1,2.5],\n",
    "                     [3,2.5,1]])\n",
    "\n",
    "beta = np.array([[1,0.2,-1],\n",
    "                     [0.2,1,2.5],\n",
    "                     [-1,2.5,1]])\n",
    "\n",
    "gamma = np.array([[1,-0.3,-0.5],\n",
    "                     [-0.3,1,2],\n",
    "                     [-0.5,2,1]])\n",
    "\n",
    "adj = nx.adjacency_matrix(graph).todense()\n",
    "\n",
    "H = general_hamiltonian(graph, adj, adj, adj)\n",
    "H"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "b79d630b",
   "metadata": {},
   "outputs": [
    {
     "ename": "AttributeError",
     "evalue": "'Tensor' object has no attribute 'row'",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mAttributeError\u001b[0m                            Traceback (most recent call last)",
      "\u001b[0;32m/var/folders/2x/4nhwbscj4_v6grk7mn3gtdl40000gn/T/ipykernel_81867/1818841959.py\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[0;32m----> 1\u001b[0;31m \u001b[0mH\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mget_sparse_torch\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mH\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m",
      "\u001b[0;32m~/Documents/LIP6/qml-research/PyQ/pyqtorch/matrices_sparse.py\u001b[0m in \u001b[0;36mget_sparse_torch\u001b[0;34m(coo_matrix)\u001b[0m\n\u001b[1;32m    142\u001b[0m \u001b[0;32mdef\u001b[0m \u001b[0mget_sparse_torch\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mcoo_matrix\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    143\u001b[0m     \u001b[0mvalues\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mcoo_matrix\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdata\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 144\u001b[0;31m     \u001b[0mindices\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mnp\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mvstack\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mcoo_matrix\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mrow\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcoo_matrix\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mcol\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    145\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    146\u001b[0m     \u001b[0mi\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mtorch\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mLongTensor\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mindices\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;31mAttributeError\u001b[0m: 'Tensor' object has no attribute 'row'"
     ]
    }
   ],
   "source": [
    "H = get_sparse_torch(H)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "id": "8f54acc3",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/Users/slimanethabet/.pyenv/versions/3.9.7/envs/my-env/lib/python3.9/site-packages/scipy/sparse/linalg/_dsolve/linsolve.py:322: SparseEfficiencyWarning: splu requires CSC matrix format\n",
      "  warn('splu requires CSC matrix format', SparseEfficiencyWarning)\n",
      "/Users/slimanethabet/.pyenv/versions/3.9.7/envs/my-env/lib/python3.9/site-packages/scipy/sparse/linalg/_dsolve/linsolve.py:215: SparseEfficiencyWarning: spsolve is more efficient when sparse b is in the CSC matrix format\n",
      "  warn('spsolve is more efficient when sparse b '\n"
     ]
    }
   ],
   "source": [
    "from scipy.sparse.linalg import eigsh, expm\n",
    "\n",
    "def eigenvalues_from_hamiltonian(H):\n",
    "    Lambda, _ = scipy.linalg.eigh(H, eigenvectors=True)\n",
    "    return np.unique(np.round(Lambda.real,8))\n",
    "\n",
    "def hamiltonian_to_unitary(H, theta=1):\n",
    "    Lambda, V = eigsh(H, k=H.shape[0]-2)\n",
    "    U = np.dot(\n",
    "            np.dot(V,np.diag(np.exp(-1j*theta/2*Lambda))),\n",
    "            np.conj(V).T) #U = V exp(-i*Lamda) V^T\n",
    "    return U\n",
    "\n",
    "#U = hamiltonian_to_unitary(H,theta=torch.pi)\n",
    "U = expm(-1j * np.pi * H)\n",
    "\n",
    "U"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "40222151",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([1.+0.j, 1.+0.j, 1.+0.j, ..., 1.+0.j, 1.+0.j, 1.+0.j])"
      ]
     },
     "execution_count": 34,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.diag(np.dot(np.conj(U.T), U).todense())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 242,
   "id": "8fffa9b7",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([-7.41006561, -2.4710468 ,  2.32457988,  7.55653252])"
      ]
     },
     "execution_count": 242,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "eigenvalues_from_hamiltonian(H)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "30a22919",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([[ 0.+0.j,  0.+0.j,  1.+0.j,  0.+0.j],\n",
       "       [ 0.+0.j, -0.+0.j,  0.+0.j, -1.+0.j],\n",
       "       [ 1.+0.j,  0.+0.j,  0.+0.j,  0.+0.j],\n",
       "       [ 0.+0.j, -1.+0.j,  0.+0.j, -0.+0.j]])"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "XMAT = np.array([[0, 1], [1, 0]], dtype=np.cdouble)\n",
    "YMAT = np.array([[0, -1j], [1j, 0]], dtype=np.cdouble)\n",
    "ZMAT = np.array([[1, 0], [0, -1]], dtype=np.cdouble)\n",
    "\n",
    "np.kron(XMAT, ZMAT)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "9419fd98",
   "metadata": {},
   "outputs": [],
   "source": [
    "def hamiltonian_evolution(H, state, t, n_qubits, n_steps=100):\n",
    "    batch_size = len(t)\n",
    "    state = state.reshape((-1, batch_size))\n",
    "    h = t.reshape((1, -1)).expand(state.shape[0], -1)/n_steps\n",
    "    for _ in range(n_steps):\n",
    "        k1 = -1j * torch.matmul(H, state)\n",
    "        k2 = -1j * torch.matmul(H, state + h/2 * k1)\n",
    "        k3 = -1j * torch.matmul(H, state + h/2 * k2)\n",
    "        k4 = -1j * torch.matmul(H, state + h * k3)\n",
    "        state += h/6 * (k1 + 2*k2 + 2*k3 + k4)\n",
    "    return state.reshape([2]*n_qubits + [batch_size])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "c75c4a08",
   "metadata": {},
   "outputs": [
    {
     "ename": "TypeError",
     "evalue": "'NoneType' object is not subscriptable",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mTypeError\u001b[0m                                 Traceback (most recent call last)",
      "\u001b[0;32m/var/folders/2x/4nhwbscj4_v6grk7mn3gtdl40000gn/T/ipykernel_81867/2979760702.py\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[0;32m----> 1\u001b[0;31m \u001b[0mH\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mgeneral_hamiltonian\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mgraph\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0malpha\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mnx\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0madjacency_matrix\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mgraph\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtodense\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mbeta\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;32mNone\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mgamma\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;32mNone\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mdevice\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;34m'cpu'\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m",
      "\u001b[0;32m/var/folders/2x/4nhwbscj4_v6grk7mn3gtdl40000gn/T/ipykernel_81867/4153909538.py\u001b[0m in \u001b[0;36mgeneral_hamiltonian\u001b[0;34m(graph, alpha, beta, gamma, device)\u001b[0m\n\u001b[1;32m     44\u001b[0m     \u001b[0;32mfor\u001b[0m \u001b[0medge\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mgraph\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0medges\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdata\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     45\u001b[0m         \u001b[0mH\u001b[0m \u001b[0;34m+=\u001b[0m \u001b[0malpha\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0medge\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;36m0\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0medge\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;36m1\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m]\u001b[0m \u001b[0;34m*\u001b[0m \u001b[0mXX\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mN\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0medge\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;36m0\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0medge\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;36m1\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mdevice\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;31m#alpha_ij * XX\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 46\u001b[0;31m         \u001b[0mH\u001b[0m \u001b[0;34m+=\u001b[0m \u001b[0mbeta\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0medge\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;36m0\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0medge\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;36m1\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m]\u001b[0m  \u001b[0;34m*\u001b[0m \u001b[0mYY\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mN\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0medge\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;36m0\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0medge\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;36m1\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mdevice\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;31m#beta_ij * YY\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     47\u001b[0m         \u001b[0mH\u001b[0m \u001b[0;34m+=\u001b[0m \u001b[0mgamma\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0medge\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;36m0\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0medge\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;36m1\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m]\u001b[0m \u001b[0;34m*\u001b[0m \u001b[0mZZ\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mN\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0medge\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;36m0\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0medge\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;36m1\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mdevice\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;31m#gamma_ij * ZZ\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     48\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;31mTypeError\u001b[0m: 'NoneType' object is not subscriptable"
     ]
    }
   ],
   "source": [
    "H = general_hamiltonian(graph, alpha=nx.adjacency_matrix(graph).todense(), beta=None, gamma=None, device='cpu')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "c2634519",
   "metadata": {},
   "outputs": [],
   "source": [
    "#H = get_sparse_torch(H)\n",
    "circuit = QuantumCircuit(n_qubits=8)\n",
    "\n",
    "t = torch.tensor([.5, .9], requires_grad=True)\n",
    "\n",
    "state = circuit.uniform_state(batch_size=2)\n",
    "state = hamiltonian_evolution(H, state, t, 8, n_steps=200)\n",
    "\n",
    "l = torch.sum(torch.conj(state) * X(state, [3], 8)).real\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "10981a34",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([-2.0979e-05, -3.9292e-04])"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "l.backward()\n",
    "t.grad"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "a325baa3",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 1])"
      ]
     },
     "execution_count": 27,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "state.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "1f3f0bc5",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([1.0000, 1.0000], dtype=torch.float64, grad_fn=<SumBackward1>)"
      ]
     },
     "execution_count": 29,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "\n",
    "torch.sum(torch.abs(state.reshape((-1, 2)))**2, axis=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4f75c63f",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "interpreter": {
   "hash": "95c228e3bcdd1cffff285f62e442dc1cbf9819f87fb728ec53d42aacc8686033"
  },
  "kernelspec": {
   "display_name": "Python 3.9.7 64-bit ('my-env')",
   "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.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
