{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 109,
   "metadata": {},
   "outputs": [],
   "source": [
    "import tensorflow as tf\n",
    "from tensorflow import keras\n",
    "\n",
    "import tensorflow_quantum as tfq\n",
    "import math\n",
    "import cirq\n",
    "import sympy\n",
    "import numpy as np\n",
    "import seaborn as sns\n",
    "import collections\n",
    "from sklearn.decomposition import PCA\n",
    "# visualization tools\n",
    "%matplotlib inline\n",
    "import matplotlib.pyplot as plt\n",
    "from cirq.contrib.svg import SVGCircuit\n",
    "plt.style.use('ggplot')\n",
    "np.random.seed(0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 110,
   "metadata": {},
   "outputs": [],
   "source": [
    "def kernel(data,cx,cy):\n",
    "    k_data = []\n",
    "    for point in data:\n",
    "        dist = np.sqrt((cx-point[0])**2 + (cy-point[1])**2)\n",
    "        rotation = get_real_angle(point[0]-cx,point[1]-cy)\n",
    "        k_data.append([dist,rotation])\n",
    "    return np.array(k_data)\n",
    "\n",
    "# def qubitization(data):\n",
    "#     ry = []\n",
    "#     rz = []\n",
    "#     q_data = []\n",
    "#     for point in data:\n",
    "#         ry_rotations = []\n",
    "#         rz_rotations = []\n",
    "#         for i,data_point in enumerate(point):\n",
    "#             if (i+2)%2 == 0:\n",
    "#                 ry_rotations.append(2*np.arcsin(np.sqrt(data_point)))\n",
    "#             else:\n",
    "#                 rz_rotations.append(2*np.arcsin(np.sqrt(data_point))) # This part might be under scrutuny, check it later\n",
    "#         q_data.append([ry_rotations,rz_rotations])\n",
    "#     return np.array(q_data)\n",
    "\n",
    "#-----------------------------------------------------------------------------------------\n",
    "def qubitization(data):\n",
    "    rotations = []\n",
    "    for point in data:\n",
    "        point_rotations = []\n",
    "        for data_point in point:\n",
    "            point_rotations.append(2*np.arcsin(np.sqrt(data_point)))\n",
    "        rotations.append(point_rotations)\n",
    "    return np.array(rotations)\n",
    "#-----------------------------------------------------------------------------------------\n",
    "def get_real_angle(point0,point1):\n",
    "    value = np.arctan(point1/point0)\n",
    "    if value < 0:\n",
    "        #We know that the angle is between 90-180 or 270-360\n",
    "        if point1 < 0:\n",
    "            return abs(np.arctan(point1/point0)) + 3*np.pi/2 \n",
    "        else:\n",
    "            return abs(np.arctan(point1/point0)) + np.pi/2 \n",
    "    elif value > 0:\n",
    "        if point1 > 0 :\n",
    "            return np.arctan(point1/point0) + 0\n",
    "        else:\n",
    "            return abs(np.arctan(point1/point0)) + np.pi "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 111,
   "metadata": {},
   "outputs": [],
   "source": [
    "(x_train, y_train), (x_test, y_test) = tf.keras.datasets.mnist.load_data()\n",
    "z=1500\n",
    "x_train = x_train[:z]\n",
    "y_train = y_train[:z]\n",
    "# Rescale the images from [0,255] to the [0.0,1.0] range.\n",
    "x_train, x_test = x_train[..., np.newaxis]/255.0, x_test[..., np.newaxis]/255.0\n",
    "x_train = x_train.reshape((z,784))\n",
    "pca = PCA(n_components=4)\n",
    "pca.fit(x_train)\n",
    "x_train = pca.transform(x_train)\n",
    "for i in range(4):\n",
    "    if x_train[:,i].min() < 0: \n",
    "        x_train[:,i] -= x_train[:,i].min()\n",
    "    x_train[:,i] /= x_train[:,i].max()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 121,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(array([ 4.,  8.,  6., 17., 27., 28., 26., 13.,  7.,  4.]),\n",
       " array([0.26601923, 0.33941731, 0.41281539, 0.48621346, 0.55961154,\n",
       "        0.63300962, 0.70640769, 0.77980577, 0.85320385, 0.92660192,\n",
       "        1.        ]),\n",
       " <a list of 10 Patch objects>)"
      ]
     },
     "execution_count": 121,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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U1RU5GtGtzmLi4ud7du3O47jOPv27XPef/9933Pu37/fOXz4sDNv3rwOf/799987y5cvd1pbW52DBw86r7/+elLj9YolFDeX9fft27ftdktLS4cXHXV3RknauHGjJk+erCuuuMLXfH+z8BUJbjJu3bpVDz/8sDIyMiRJWVlZPTLnxSoqKjR69GgfE7rLGAgEdObMGUnSmTNnuuV6Dzc5jx8/roKCAknSLbfcoj179viec/jw4W3nXEf27Nmj+++/X4FAQEOHDlVzc7Pq6+s9j9crCryjy/rr6uraPW7z5s16+eWXtWHDBj3zzDN+RnSV8ciRI4rH47rzzjt9zXYxt6/ld999p/nz56u4uFjxeNzPiK4y1tTUqLa2Vm+++abeeOMN7d2719eMkvvXUrrw5/+JEyd06623+hVPkruMTzzxhL755hvNmTNHK1as0LPPPutrRsldzhtuuEHfffedJGn37t06e/asmpqafM2ZSF1dnUKhUNv9y50TbvSKAndcXtY/ceJErVq1SjNmzNDnn3/uR7Q2iTK2trZq/fr1mjVrlp+x2nHzWt55552KRqN69913VVBQoGg06lc8Se4ytra2qra2VkuWLNHcuXNVWlqq5uZmvyJKcn9eShdm33fffbeCQX8/km4yVlRUaNy4cSotLdXrr7+uVatWqbW11a+IktzlnDlzpqqrq/Xaa6+purpaOTk5SktL8yuiK505J9zoFQXe2cv6u2NZIFHGlpYWHTt2TG+99ZZefPFF/frrr3r77bd1+PDhHpVTkq6++uq2JZ5wOKzffvutx2XMycnRyJEjlZ6ermuvvVa5ubmqra3tcTn/tmvXLt13331+RWvjJuO2bdt0zz33SJKGDh2qc+fO+T6zdfuez58/X2+//bamT58u6dKl056gX79+l/zFmuxXkPSKAndzWf/FH96qqipdd911PSpj3759tWbNGkWjUUWjUd1000167bXXfN+F4ua1vHjNbs+ePb7v5nGT8a677tLPP/8sSWpsbFRtba0GDBjQ43JKF5Z7mpubNXToUF/zuc0YCoXaXsvjx4/r3LlzyszM7HE5Gxsb2/4y+OKLL/TAAw/4mtGNwsJClZeXy3Ec/fLLL+rbt29SBd5rrsSsqqrS+vXr1draqgceeEBTp07Vxo0bNWTIEBUWFmrt2rXat2+f0tLSlJGRoWeffVaDBg3qURkvtnTpUs2cObNbthEmyvnpp59qz549ba/l888/r7y8vB6V0XEcffzxx9q7d6+CwaCmTp3aLTNcN+95WVmZzp07pxkzZviez03G48eP68MPP1RLS4sk6amnntLtt9/e43J+++23+vTTTxUIBHTzzTfrueee830zwMqVK1VdXa2mpiZlZWVp2rRpOn/+vCRpwoQJchxHa9as0Y8//qj//e9/KioqSuoz3msKHAD+a3rFEgoA/BdR4ABgFAUOAEZR4ABgFAUOAEZR4ABgFAUOAEb9H7biLM4TIlL4AAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "ind1 = 0\n",
    "ind2 = 0\n",
    "data = x_train[:,0][y_train==ind1]\n",
    "data2 = x_train[:,1][y_train==ind1]\n",
    "# Ind 1 and ind2 are the numbers you want to binary classification between\n",
    "plt.hist(x_train[:,0][y_train==ind1])\n",
    "plt.hist(x_train[:,0][y_train==ind2])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 100,
   "metadata": {},
   "outputs": [],
   "source": [
    "#TENSORFLOW QUANTUM\n",
    "def convert_to_circuit(data):\n",
    "    \"\"\"Encode truncated classical image into quantum datapoint.\"\"\"\n",
    "    qubits = cirq.GridQubit.rect(1,2)\n",
    "    circuit = cirq.Circuit()\n",
    "    circuit.append(cirq.ry(data[0])(qubits[0]))\n",
    "    circuit.append(cirq.rz(data[1])(qubits[0]))\n",
    "    circuit.append(cirq.ry(data[2])(qubits[1]))\n",
    "    circuit.append(cirq.rz(data[3])(qubits[1]))\n",
    "    return circuit\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 101,
   "metadata": {},
   "outputs": [],
   "source": [
    "#TENSORFLOW QUANTUM\n",
    "def one_qubit_unitary(bit, symbols):\n",
    "    \"\"\"Make a Cirq circuit enacting a rotation of the bloch sphere about the X,\n",
    "    Y and Z axis, that depends on the values in `symbols`.\n",
    "    \"\"\"\n",
    "    return cirq.Circuit(\n",
    "        cirq.ry(symbols[0])(bit),\n",
    "        cirq.rz(symbols[1])(bit))\n",
    "\n",
    "def two_qubit_unitary(bits, symbols):\n",
    "    \"\"\"Make a Cirq circuit enacting a rotation of the bloch sphere about the X,\n",
    "    Y and Z axis, that depends on the values in `symbols`.\n",
    "    \"\"\"\n",
    "    return cirq.Circuit(\n",
    "        cirq.ry(symbols[0])(bits[0]),\n",
    "        cirq.rz(symbols[1])(bits[0]),\n",
    "        cirq.ry(symbols[0])(bits[1]),\n",
    "        cirq.rz(symbols[1])(bits[1]))\n",
    "\n",
    "def ent_qubit_unitary(bits,symbols,swap = False):\n",
    "    if swap:\n",
    "        return cirq.Circuit(\n",
    "            cirq.CNotPowGate(exponent=symbols[0])(bits[1],bits[0]),\n",
    "            cirq.CZPowGate(exponent=symbols[1])(bits[1],bits[0]))\n",
    "    else:\n",
    "        return cirq.Circuit(\n",
    "            cirq.CNotPowGate(exponent=symbols[0])(bits[0],bits[1]),\n",
    "            cirq.CZPowGate(exponent=symbols[1])(bits[0],bits[1]))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 102,
   "metadata": {},
   "outputs": [],
   "source": [
    "def create_model_circuit(qubits,get_qasm=False,old_circuit = None):\n",
    "    \"\"\"Create sequence of alternating convolution and pooling operators \n",
    "    which gradually shrink over time.\"\"\"\n",
    "    if not old_circuit:\n",
    "        model_circuit = cirq.Circuit()\n",
    "    else: \n",
    "        model_circuit = old_circuit \n",
    "    if get_qasm:\n",
    "        symbols= qcnn_model.trainable_variables[0].numpy()\n",
    "    else:\n",
    "        symbols = sympy.symbols('Param-0:63')\n",
    "    # Cirq uses sympy.Symbols to map learnable variables. TensorFlow Quantum\n",
    "    # scans incoming circuits and replaces these with TensorFlow variables.\n",
    "    model_circuit += one_qubit_unitary(qubits[0], symbols[0:2])\n",
    "    model_circuit += one_qubit_unitary(qubits[1], symbols[2:4])\n",
    "    model_circuit += ent_qubit_unitary(qubits[0:2],symbols[4:6],swap=True)\n",
    "    model_circuit += one_qubit_unitary(qubits[1],symbols[6:8])\n",
    "    if get_qasm:\n",
    "        return model_circuit\n",
    "    return model_circuit\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 103,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/svg+xml": [
       "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"980.6060156250002\" height=\"100.0\"><line x1=\"32.246796875\" x2=\"950.6060156250002\" y1=\"25.0\" y2=\"25.0\" stroke=\"#1967d2\" stroke-width=\"1\" /><line x1=\"32.246796875\" x2=\"950.6060156250002\" y1=\"75.0\" y2=\"75.0\" stroke=\"#1967d2\" stroke-width=\"1\" /><line x1=\"554.7992382812502\" x2=\"554.7992382812502\" y1=\"25.0\" y2=\"75.0\" stroke=\"black\" stroke-width=\"3\" /><line x1=\"667.3224023437501\" x2=\"667.3224023437501\" y1=\"25.0\" y2=\"75.0\" stroke=\"black\" stroke-width=\"3\" /><rect x=\"10.0\" y=\"5.0\" width=\"44.49359375\" height=\"40\" stroke=\"black\" fill=\"white\" stroke-width=\"0\" /><text x=\"32.246796875\" y=\"25.0\" dominant-baseline=\"middle\" text-anchor=\"middle\" font-size=\"14px\">(0, 0): </text><rect x=\"10.0\" y=\"55.0\" width=\"44.49359375\" height=\"40\" stroke=\"black\" fill=\"white\" stroke-width=\"0\" /><text x=\"32.246796875\" y=\"75.0\" dominant-baseline=\"middle\" text-anchor=\"middle\" font-size=\"14px\">(0, 1): </text><rect x=\"74.49359375\" y=\"5.0\" width=\"88.51101562500001\" height=\"40\" stroke=\"black\" fill=\"white\" stroke-width=\"1\" /><text x=\"118.74910156250002\" y=\"25.0\" dominant-baseline=\"middle\" text-anchor=\"middle\" font-size=\"14px\">Ry(Param-0)</text><rect x=\"183.00460937500003\" y=\"5.0\" width=\"88.51101562500001\" height=\"40\" stroke=\"black\" fill=\"white\" stroke-width=\"1\" /><text x=\"227.26011718750004\" y=\"25.0\" dominant-baseline=\"middle\" text-anchor=\"middle\" font-size=\"14px\">Rz(Param-1)</text><rect x=\"291.51562500000006\" y=\"55.0\" width=\"88.51101562500001\" height=\"40\" stroke=\"black\" fill=\"white\" stroke-width=\"1\" /><text x=\"335.77113281250007\" y=\"75.0\" dominant-baseline=\"middle\" text-anchor=\"middle\" font-size=\"14px\">Ry(Param-2)</text><rect x=\"400.0266406250001\" y=\"55.0\" width=\"88.51101562500001\" height=\"40\" stroke=\"black\" fill=\"white\" stroke-width=\"1\" /><text x=\"444.2821484375001\" y=\"75.0\" dominant-baseline=\"middle\" text-anchor=\"middle\" font-size=\"14px\">Rz(Param-3)</text><rect x=\"508.53765625000017\" y=\"55.0\" width=\"92.52316406250002\" height=\"40\" stroke=\"black\" fill=\"white\" stroke-width=\"1\" /><text x=\"554.7992382812502\" y=\"75.0\" dominant-baseline=\"middle\" text-anchor=\"middle\" font-size=\"14px\">@^(Param-4)</text><rect x=\"508.53765625000017\" y=\"5.0\" width=\"92.52316406250002\" height=\"40\" stroke=\"black\" fill=\"white\" stroke-width=\"1\" /><text x=\"554.7992382812502\" y=\"25.0\" dominant-baseline=\"middle\" text-anchor=\"middle\" font-size=\"18px\">X</text><rect x=\"621.0608203125001\" y=\"55.0\" width=\"92.52316406250002\" height=\"40\" stroke=\"black\" fill=\"white\" stroke-width=\"1\" /><text x=\"667.3224023437501\" y=\"75.0\" dominant-baseline=\"middle\" text-anchor=\"middle\" font-size=\"14px\">@^(Param-5)</text><circle cx=\"667.3224023437501\" cy=\"25.0\" r=\"10.0\" /><rect x=\"733.5839843750001\" y=\"55.0\" width=\"88.51101562500001\" height=\"40\" stroke=\"black\" fill=\"white\" stroke-width=\"1\" /><text x=\"777.8394921875001\" y=\"75.0\" dominant-baseline=\"middle\" text-anchor=\"middle\" font-size=\"14px\">Ry(Param-6)</text><rect x=\"842.0950000000001\" y=\"55.0\" width=\"88.51101562500001\" height=\"40\" stroke=\"black\" fill=\"white\" stroke-width=\"1\" /><text x=\"886.3505078125002\" y=\"75.0\" dominant-baseline=\"middle\" text-anchor=\"middle\" font-size=\"14px\">Rz(Param-7)</text></svg>"
      ],
      "text/plain": [
       "<cirq.contrib.svg.svg.SVGCircuit at 0x2c31c4e50c8>"
      ]
     },
     "execution_count": 103,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "SVGCircuit(create_model_circuit(cirq.GridQubit.rect(1,2),False))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 104,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "None\n"
     ]
    }
   ],
   "source": [
    "print(circ_data[0].append(create_model_circuit(cirq.GridQubit.rect(1,2),True)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 105,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "EPOCHS = 10\n",
    "# Custom accuracy metric.\n",
    "@tf.function\n",
    "def custom_accuracy(y_true, y_pred):\n",
    "    y_true = tf.squeeze(y_true)\n",
    "    y_pred = tf.map_fn(lambda x: 1.0 if x >= 0 else -1.0, y_pred)\n",
    "    return tf.keras.backend.mean(tf.keras.backend.equal(y_true, y_pred))\n",
    "\n",
    "\n",
    "def get_var_count(model_name):\n",
    "    return np.sum([np.prod(v.get_shape().as_list()) for v in model_name.trainable_variables])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 106,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Quantum Tensorflow\n",
    "d1 = x_train[y_train==ind1]\n",
    "d2 = x_train[y_train==ind2]\n",
    "comb_data = np.concatenate((d1,d2))\n",
    "circ_data = [convert_to_circuit(x) for x in qubitization(comb_data)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 107,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluating numbers 0 and 1:\n",
      "Train on 312 samples\n",
      "Epoch 1/25\n"
     ]
    },
    {
     "ename": "InternalError",
     "evalue": "2 root error(s) found.\n  (0) Internal:  Blas xGEMV launch failed : a.shape=[1,1,64], b.shape=[1,1,1], m=64, n=1, k=1\n\t [[{{node PartitionedCall/map/while/body/_15/einsum/Einsum}}]]\n\t [[PartitionedCall/Reshape_1/_58]]\n  (1) Internal:  Blas xGEMV launch failed : a.shape=[1,1,64], b.shape=[1,1,1], m=64, n=1, k=1\n\t [[{{node PartitionedCall/map/while/body/_15/einsum/Einsum}}]]\n0 successful operations.\n0 derived errors ignored. [Op:__inference_distributed_function_1277]\n\nFunction call stack:\ndistributed_function -> distributed_function\n",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[1;31mInternalError\u001b[0m                             Traceback (most recent call last)",
      "\u001b[1;32m<ipython-input-107-2ee5c30f40f5>\u001b[0m in \u001b[0;36m<module>\u001b[1;34m\u001b[0m\n\u001b[0;32m     37\u001b[0m                                  \u001b[0mbatch_size\u001b[0m\u001b[1;33m=\u001b[0m\u001b[1;36m8\u001b[0m\u001b[1;33m,\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m     38\u001b[0m                                  \u001b[0mepochs\u001b[0m\u001b[1;33m=\u001b[0m\u001b[0mEPOCHS\u001b[0m\u001b[1;33m,\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m---> 39\u001b[1;33m                                  verbose=2)\n\u001b[0m\u001b[0;32m     40\u001b[0m         \u001b[0mtracked_data\u001b[0m\u001b[1;33m[\u001b[0m\u001b[1;34m'{}_vs_{}_qcnn'\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mformat\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mind1\u001b[0m\u001b[1;33m,\u001b[0m\u001b[0mind2\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m]\u001b[0m \u001b[1;33m=\u001b[0m \u001b[0mquantum_history\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mhistory\u001b[0m\u001b[1;33m[\u001b[0m\u001b[1;34m'custom_accuracy'\u001b[0m\u001b[1;33m]\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m     41\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\lib\\site-packages\\tensorflow_core\\python\\keras\\engine\\training.py\u001b[0m in \u001b[0;36mfit\u001b[1;34m(self, x, y, batch_size, epochs, verbose, callbacks, validation_split, validation_data, shuffle, class_weight, sample_weight, initial_epoch, steps_per_epoch, validation_steps, validation_freq, max_queue_size, workers, use_multiprocessing, **kwargs)\u001b[0m\n\u001b[0;32m    817\u001b[0m         \u001b[0mmax_queue_size\u001b[0m\u001b[1;33m=\u001b[0m\u001b[0mmax_queue_size\u001b[0m\u001b[1;33m,\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    818\u001b[0m         \u001b[0mworkers\u001b[0m\u001b[1;33m=\u001b[0m\u001b[0mworkers\u001b[0m\u001b[1;33m,\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m--> 819\u001b[1;33m         use_multiprocessing=use_multiprocessing)\n\u001b[0m\u001b[0;32m    820\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    821\u001b[0m   def evaluate(self,\n",
      "\u001b[1;32m~\\anaconda3\\lib\\site-packages\\tensorflow_core\\python\\keras\\engine\\training_v2.py\u001b[0m in \u001b[0;36mfit\u001b[1;34m(self, model, x, y, batch_size, epochs, verbose, callbacks, validation_split, validation_data, shuffle, class_weight, sample_weight, initial_epoch, steps_per_epoch, validation_steps, validation_freq, max_queue_size, workers, use_multiprocessing, **kwargs)\u001b[0m\n\u001b[0;32m    340\u001b[0m                 \u001b[0mmode\u001b[0m\u001b[1;33m=\u001b[0m\u001b[0mModeKeys\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mTRAIN\u001b[0m\u001b[1;33m,\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    341\u001b[0m                 \u001b[0mtraining_context\u001b[0m\u001b[1;33m=\u001b[0m\u001b[0mtraining_context\u001b[0m\u001b[1;33m,\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m--> 342\u001b[1;33m                 total_epochs=epochs)\n\u001b[0m\u001b[0;32m    343\u001b[0m             \u001b[0mcbks\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mmake_logs\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mmodel\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mepoch_logs\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mtraining_result\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mModeKeys\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mTRAIN\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    344\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\lib\\site-packages\\tensorflow_core\\python\\keras\\engine\\training_v2.py\u001b[0m in \u001b[0;36mrun_one_epoch\u001b[1;34m(model, iterator, execution_function, dataset_size, batch_size, strategy, steps_per_epoch, num_samples, mode, training_context, total_epochs)\u001b[0m\n\u001b[0;32m    126\u001b[0m         step=step, mode=mode, size=current_batch_size) as batch_logs:\n\u001b[0;32m    127\u001b[0m       \u001b[1;32mtry\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m--> 128\u001b[1;33m         \u001b[0mbatch_outs\u001b[0m \u001b[1;33m=\u001b[0m \u001b[0mexecution_function\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0miterator\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m    129\u001b[0m       \u001b[1;32mexcept\u001b[0m \u001b[1;33m(\u001b[0m\u001b[0mStopIteration\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0merrors\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mOutOfRangeError\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    130\u001b[0m         \u001b[1;31m# TODO(kaftan): File bug about tf function and errors.OutOfRangeError?\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\lib\\site-packages\\tensorflow_core\\python\\keras\\engine\\training_v2_utils.py\u001b[0m in \u001b[0;36mexecution_function\u001b[1;34m(input_fn)\u001b[0m\n\u001b[0;32m     96\u001b[0m     \u001b[1;31m# `numpy` translates Tensors to values in Eager mode.\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m     97\u001b[0m     return nest.map_structure(_non_none_constant_value,\n\u001b[1;32m---> 98\u001b[1;33m                               distributed_function(input_fn))\n\u001b[0m\u001b[0;32m     99\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    100\u001b[0m   \u001b[1;32mreturn\u001b[0m \u001b[0mexecution_function\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\lib\\site-packages\\tensorflow_core\\python\\eager\\def_function.py\u001b[0m in \u001b[0;36m__call__\u001b[1;34m(self, *args, **kwds)\u001b[0m\n\u001b[0;32m    566\u001b[0m         \u001b[0mxla_context\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mExit\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    567\u001b[0m     \u001b[1;32melse\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m--> 568\u001b[1;33m       \u001b[0mresult\u001b[0m \u001b[1;33m=\u001b[0m \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0m_call\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m*\u001b[0m\u001b[0margs\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m**\u001b[0m\u001b[0mkwds\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m    569\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    570\u001b[0m     \u001b[1;32mif\u001b[0m \u001b[0mtracing_count\u001b[0m \u001b[1;33m==\u001b[0m \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0m_get_tracing_count\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\lib\\site-packages\\tensorflow_core\\python\\eager\\def_function.py\u001b[0m in \u001b[0;36m_call\u001b[1;34m(self, *args, **kwds)\u001b[0m\n\u001b[0;32m    630\u001b[0m         \u001b[1;31m# Lifting succeeded, so variables are initialized and we can run the\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    631\u001b[0m         \u001b[1;31m# stateless function.\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m--> 632\u001b[1;33m         \u001b[1;32mreturn\u001b[0m \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0m_stateless_fn\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m*\u001b[0m\u001b[0margs\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m**\u001b[0m\u001b[0mkwds\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m    633\u001b[0m     \u001b[1;32melse\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    634\u001b[0m       \u001b[0mcanon_args\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mcanon_kwds\u001b[0m \u001b[1;33m=\u001b[0m\u001b[0;31m \u001b[0m\u001b[0;31m\\\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\lib\\site-packages\\tensorflow_core\\python\\eager\\function.py\u001b[0m in \u001b[0;36m__call__\u001b[1;34m(self, *args, **kwargs)\u001b[0m\n\u001b[0;32m   2361\u001b[0m     \u001b[1;32mwith\u001b[0m \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0m_lock\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m   2362\u001b[0m       \u001b[0mgraph_function\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0margs\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mkwargs\u001b[0m \u001b[1;33m=\u001b[0m \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0m_maybe_define_function\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0margs\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mkwargs\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m-> 2363\u001b[1;33m     \u001b[1;32mreturn\u001b[0m \u001b[0mgraph_function\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0m_filtered_call\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0margs\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mkwargs\u001b[0m\u001b[1;33m)\u001b[0m  \u001b[1;31m# pylint: disable=protected-access\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m   2364\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m   2365\u001b[0m   \u001b[1;33m@\u001b[0m\u001b[0mproperty\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\lib\\site-packages\\tensorflow_core\\python\\eager\\function.py\u001b[0m in \u001b[0;36m_filtered_call\u001b[1;34m(self, args, kwargs)\u001b[0m\n\u001b[0;32m   1609\u001b[0m          if isinstance(t, (ops.Tensor,\n\u001b[0;32m   1610\u001b[0m                            resource_variable_ops.BaseResourceVariable))),\n\u001b[1;32m-> 1611\u001b[1;33m         self.captured_inputs)\n\u001b[0m\u001b[0;32m   1612\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m   1613\u001b[0m   \u001b[1;32mdef\u001b[0m \u001b[0m_call_flat\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mself\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0margs\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mcaptured_inputs\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mcancellation_manager\u001b[0m\u001b[1;33m=\u001b[0m\u001b[1;32mNone\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\lib\\site-packages\\tensorflow_core\\python\\eager\\function.py\u001b[0m in \u001b[0;36m_call_flat\u001b[1;34m(self, args, captured_inputs, cancellation_manager)\u001b[0m\n\u001b[0;32m   1690\u001b[0m       \u001b[1;31m# No tape is watching; skip to running the function.\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m   1691\u001b[0m       return self._build_call_outputs(self._inference_function.call(\n\u001b[1;32m-> 1692\u001b[1;33m           ctx, args, cancellation_manager=cancellation_manager))\n\u001b[0m\u001b[0;32m   1693\u001b[0m     forward_backward = self._select_forward_and_backward_functions(\n\u001b[0;32m   1694\u001b[0m         \u001b[0margs\u001b[0m\u001b[1;33m,\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\lib\\site-packages\\tensorflow_core\\python\\eager\\function.py\u001b[0m in \u001b[0;36mcall\u001b[1;34m(self, ctx, args, cancellation_manager)\u001b[0m\n\u001b[0;32m    543\u001b[0m               \u001b[0minputs\u001b[0m\u001b[1;33m=\u001b[0m\u001b[0margs\u001b[0m\u001b[1;33m,\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    544\u001b[0m               \u001b[0mattrs\u001b[0m\u001b[1;33m=\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;34m\"executor_type\"\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mexecutor_type\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;34m\"config_proto\"\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mconfig\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m,\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m--> 545\u001b[1;33m               ctx=ctx)\n\u001b[0m\u001b[0;32m    546\u001b[0m         \u001b[1;32melse\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    547\u001b[0m           outputs = execute.execute_with_cancellation(\n",
      "\u001b[1;32m~\\anaconda3\\lib\\site-packages\\tensorflow_core\\python\\eager\\execute.py\u001b[0m in \u001b[0;36mquick_execute\u001b[1;34m(op_name, num_outputs, inputs, attrs, ctx, name)\u001b[0m\n\u001b[0;32m     65\u001b[0m     \u001b[1;32melse\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m     66\u001b[0m       \u001b[0mmessage\u001b[0m \u001b[1;33m=\u001b[0m \u001b[0me\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mmessage\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m---> 67\u001b[1;33m     \u001b[0msix\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mraise_from\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mcore\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0m_status_to_exception\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0me\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mcode\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mmessage\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;32mNone\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m     68\u001b[0m   \u001b[1;32mexcept\u001b[0m \u001b[0mTypeError\u001b[0m \u001b[1;32mas\u001b[0m \u001b[0me\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m     69\u001b[0m     keras_symbolic_tensors = [\n",
      "\u001b[1;32m~\\anaconda3\\lib\\site-packages\\six.py\u001b[0m in \u001b[0;36mraise_from\u001b[1;34m(value, from_value)\u001b[0m\n",
      "\u001b[1;31mInternalError\u001b[0m: 2 root error(s) found.\n  (0) Internal:  Blas xGEMV launch failed : a.shape=[1,1,64], b.shape=[1,1,1], m=64, n=1, k=1\n\t [[{{node PartitionedCall/map/while/body/_15/einsum/Einsum}}]]\n\t [[PartitionedCall/Reshape_1/_58]]\n  (1) Internal:  Blas xGEMV launch failed : a.shape=[1,1,64], b.shape=[1,1,1], m=64, n=1, k=1\n\t [[{{node PartitionedCall/map/while/body/_15/einsum/Einsum}}]]\n0 successful operations.\n0 derived errors ignored. [Op:__inference_distributed_function_1277]\n\nFunction call stack:\ndistributed_function -> distributed_function\n"
     ]
    }
   ],
   "source": [
    "#QUantum Deep Neural Net\n",
    "EPOCHS=25\n",
    "tracked_data = {}\n",
    "for ind1 in range(10):\n",
    "    for ind2 in range(10):\n",
    "        if ind1==ind2:\n",
    "            continue\n",
    "        results = {}\n",
    "\n",
    "        cluster_state_bits = cirq.GridQubit.rect(1, 2)\n",
    "        readout_operators = cirq.Z(cluster_state_bits[-1])\n",
    "\n",
    "\n",
    "        print(f\"Evaluating numbers {ind1} and {ind2}:\")\n",
    "        data1_labels = np.zeros(len(x_train[y_train==ind1])) + 1\n",
    "        data2_labels = np.zeros(len(x_train[y_train==ind2])) -1\n",
    "        labels = np.concatenate((data1_labels,data2_labels))\n",
    "        d1 = x_train[y_train==ind1]\n",
    "        d2 = x_train[y_train==ind2]\n",
    "        comb_data = np.concatenate((d1,d2))\n",
    "        circ_data = [convert_to_circuit(x) for x in qubitization(comb_data)]\n",
    "        x_train_tfcirc = tfq.convert_to_tensor(circ_data)\n",
    "        excitation_input = tf.keras.Input(shape=(), dtype=tf.dtypes.string)\n",
    "        quantum_model = tfq.layers.PQC(create_model_circuit(cluster_state_bits),\n",
    "                                       readout_operators)(excitation_input)\n",
    "\n",
    "        qcnn_model = tf.keras.Model(inputs=[excitation_input], outputs=[quantum_model])\n",
    "        qcnn_model.compile(optimizer=tf.keras.optimizers.Adam(),\n",
    "                           loss=tf.losses.mse,\n",
    "                           metrics=[custom_accuracy])\n",
    "        tf.keras.utils.plot_model(qcnn_model,\n",
    "                                  show_shapes=True,\n",
    "                                  show_layer_names=False,\n",
    "                                  dpi=70)\n",
    "        quantum_history = qcnn_model.fit(x=x_train_tfcirc,\n",
    "                                 y=labels,\n",
    "                                 batch_size=8,\n",
    "                                 epochs=EPOCHS,\n",
    "                                 verbose=2)\n",
    "        tracked_data['{}_vs_{}_qcnn'.format(ind1,ind2)] = quantum_history.history['custom_accuracy']\n",
    "\n",
    "        # Classical Deep Neural Net\n",
    "\n",
    "        # Setup which numbers to binary classify \n",
    "\n",
    "        # Generating a classical neural network \n",
    "        model = keras.Sequential([\n",
    "            keras.layers.Dense(2),\n",
    "            keras.layers.Dense(1,activation='sigmoid')\n",
    "        ])\n",
    "\n",
    "        # Compiling a classical neural network\n",
    "        model.compile(optimizer='adam',\n",
    "                      loss=tf.keras.losses.mse,\n",
    "                      metrics=['accuracy'])\n",
    "\n",
    "        # Data preparation based on ind1 and ind2\n",
    "        d1 = x_train[y_train==ind1]\n",
    "        d2 = x_train[y_train==ind2]\n",
    "        data1_labels = np.zeros(len(d1)) + 1\n",
    "        data2_labels = np.zeros(len(d2)) \n",
    "        labels = np.concatenate((data1_labels,data2_labels))\n",
    "        comb_data = np.concatenate((d1,d2))\n",
    "\n",
    "        # Fitting the model\n",
    "        history = model.fit(comb_data, labels, epochs=EPOCHS,verbose=2)\n",
    "        p1 = get_var_count(model)\n",
    "        tracked_data['{}_vs_{}_CNN - {} Parameters'.format(ind1,ind2,p1)] = history.history['accuracy']\n",
    "        # Classical Deep Neural Net\n",
    "        # Setup which numbers to binary classify \n",
    "\n",
    "        # Generating a classical neural network \n",
    "        model = keras.Sequential([\n",
    "            keras.layers.Dense(8),\n",
    "            keras.layers.Dense(1,activation='sigmoid')\n",
    "        ])\n",
    "\n",
    "        # Compiling a classical neural network\n",
    "        model.compile(optimizer='adam',\n",
    "                      loss=tf.keras.losses.mse,\n",
    "                      metrics=['accuracy'])\n",
    "\n",
    "        # Data preparation based on ind1 and ind2\n",
    "        d1 = x_train[y_train==ind1]\n",
    "        d2 = x_train[y_train==ind2]\n",
    "        data1_labels = np.zeros(len(d1)) + 1\n",
    "        data2_labels = np.zeros(len(d2)) \n",
    "        labels = np.concatenate((data1_labels,data2_labels))\n",
    "        comb_data = np.concatenate((d1,d2))\n",
    "\n",
    "        # Fitting the model\n",
    "        history2 = model.fit(comb_data, labels, epochs=EPOCHS,verbose=2)\n",
    "        p2 = get_var_count(model)\n",
    "        tracked_data['{}_vs_{}_CNN - {} Parameters'.format(ind1,ind2,p2)] = history2.history['accuracy']\n",
    "\n",
    "\n",
    "        # Classical Deep Neural Net\n",
    "\n",
    "        # Setup which numbers to binary classify \n",
    "\n",
    "        # Generating a classical neural network \n",
    "        model = keras.Sequential([\n",
    "            keras.layers.Dense(16),\n",
    "            keras.layers.Dense(1,activation='sigmoid')\n",
    "        ])\n",
    "\n",
    "        # Compiling a classical neural network\n",
    "        model.compile(optimizer='adam',\n",
    "                      loss=tf.keras.losses.mse,\n",
    "                      metrics=['accuracy'])\n",
    "\n",
    "        # Data preparation based on ind1 and ind2\n",
    "        d1 = x_train[y_train==ind1]\n",
    "        d2 = x_train[y_train==ind2]\n",
    "        data1_labels = np.zeros(len(d1)) + 1\n",
    "        data2_labels = np.zeros(len(d2)) \n",
    "        labels = np.concatenate((data1_labels,data2_labels))\n",
    "        comb_data = np.concatenate((d1,d2))\n",
    "\n",
    "        # Fitting the model\n",
    "        history3 = model.fit(comb_data, labels, epochs=EPOCHS,verbose=2)\n",
    "\n",
    "        p3 = get_var_count(model)\n",
    "        tracked_data['{}_vs_{}_CNN - {} Parameters'.format(ind1,ind2,p3)] = history3.history['accuracy']\n",
    "        # Classical Deep Neural Net\n",
    "\n",
    "        # Setup which numbers to binary classify \n",
    "\n",
    "        # Generating a classical neural network \n",
    "        model = keras.Sequential([\n",
    "            keras.layers.Dense(32),\n",
    "            keras.layers.Dense(1,activation='sigmoid')\n",
    "        ])\n",
    "\n",
    "        # Compiling a classical neural network\n",
    "        model.compile(optimizer='adam',\n",
    "                      loss=tf.keras.losses.mse,\n",
    "                      metrics=['accuracy'])\n",
    "\n",
    "        # Data preparation based on ind1 and ind2\n",
    "        d1 = x_train[y_train==ind1]\n",
    "        d2 = x_train[y_train==ind2]\n",
    "        data1_labels = np.zeros(len(d1)) + 1\n",
    "        data2_labels = np.zeros(len(d2)) \n",
    "        labels = np.concatenate((data1_labels,data2_labels))\n",
    "        comb_data = np.concatenate((d1,d2))\n",
    "\n",
    "        # Fitting the model\n",
    "        history4 = model.fit(comb_data, labels, epochs=EPOCHS,verbose=2)\n",
    "        p4 = get_var_count(model)\n",
    "        tracked_data['{}_vs_{}_CNN - {} Parameters'.format(ind1,ind2,p4)] = history4.history['accuracy']\n",
    "\n",
    "        sns.lineplot(x=np.arange(0,25),y=data['0_vs_5_qcnn'],label='QNN - 8P')\n",
    "        sns.lineplot(x=np.arange(0,25),y=data['0_vs_5_CNN - 13 Parameters'],label='CNN - {}P'.format(13))\n",
    "        sns.lineplot(x=np.arange(0,25),y=data['0_vs_5_CNN - 49 Parameters'],label='CNN - {}P'.format(49))\n",
    "        sns.lineplot(x=np.arange(0,25),y=data['0_vs_5_CNN - 97 Parameters'],label='CNN - {}P'.format(97))\n",
    "        sns.lineplot(x=np.arange(0,25),y=data['0_vs_5_CNN - 193 Parameters'],label='CNN - {}P'.format(193))\n",
    "        plt.legend()\n",
    "        plt.tight_layout(rect= (0, 0.025, 1.025, 1))\n",
    "\n",
    "        plt.ylim((0,1))\n",
    "        plt.xlabel(\"Epochs\",fontsize=\"13\")\n",
    "        plt.ylabel(\"Accuracy\",fontsize=\"13\")\n",
    "        plt.savefig('binary_classification_{}_vs_{}.eps'.format(ind1,ind2),format='eps',pad_inches = 0)\n",
    "        plt.show()\n",
    "        #plt.savefig('binary_classification_{}_vs_{}.eps'.format(ind1,ind2),format='eps')\n",
    "        plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "with open('results.txt', 'r', encoding='cp1252') as f:\n",
    "    for line in f:\n",
    "        data = eval(line)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import seaborn as sns\n",
    "plt.style.use('default')\n",
    "penguins = sns.load_dataset(\"penguins\")\n",
    "\n",
    "# Draw a nested barplot by species and sex\n",
    "g = sns.catplot(\n",
    "    data=data, kind=\"bar\",\n",
    "    x=\"Numbers\", y=\"Acc\", hue=\"Real Vs Simulator\",\n",
    "    ci=\"sd\", palette=\"dark\", alpha=.6, height=6,legend=False\n",
    ")\n",
    "plt.ylim((0.6,1.0))\n",
    "plt.legend(loc='upper right')\n",
    "g.set_ylabels(\"Accuracy\",fontsize='13')\n",
    "g.set_xlabels(\"Numbers\",fontsize='13')\n",
    "plt.savefig('comparison_ibmq_real.eps',format='eps')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 89,
   "metadata": {},
   "outputs": [],
   "source": [
    "d = {'Real Vs Simulator':['IBMQ','Simulator','IBMQ','Simulator', 'IBMQ','Simulator','IBMQ','Simulator','IBMQ','Simulator'],\n",
    "     'Acc': [0.89,0.93, 0.97,0.97, 0.87,0.93, 0.931,0.948, 0.82,0.90],\n",
    "      'Numbers':['2-9','2-9','9-6','9-6','3-1','3-1','4-3','4-3','0-5','0-5']}\n",
    "import pandas as pd\n",
    "data = pd.DataFrame(d)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 67,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0    2/9\n",
       "1    2/9\n",
       "2    9/6\n",
       "3    9/6\n",
       "4    3/1\n",
       "5    3/1\n",
       "6    4/3\n",
       "7    4/3\n",
       "8    0/5\n",
       "9    0/5\n",
       "Name: Numbers, dtype: object"
      ]
     },
     "execution_count": 67,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "data['Numbers']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "metadata": {},
   "outputs": [
    {
     "ename": "AttributeError",
     "evalue": "'int' object has no attribute 'value'",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[1;31mAttributeError\u001b[0m                            Traceback (most recent call last)",
      "\u001b[1;32m<ipython-input-32-38d848195d93>\u001b[0m in \u001b[0;36m<module>\u001b[1;34m\u001b[0m\n\u001b[0;32m      5\u001b[0m     \u001b[0mshape\u001b[0m \u001b[1;33m=\u001b[0m \u001b[0mvariable\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mget_shape\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m)\u001b[0m  \u001b[1;31m#getting shape of a variable\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m      6\u001b[0m     \u001b[1;32mfor\u001b[0m \u001b[0mi\u001b[0m \u001b[1;32min\u001b[0m \u001b[0mshape\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m----> 7\u001b[1;33m         \u001b[0mlocal_parameters\u001b[0m\u001b[1;33m*=\u001b[0m\u001b[0mi\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mvalue\u001b[0m  \u001b[1;31m#mutiplying dimension values\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m      8\u001b[0m     \u001b[0mtotal_parameters\u001b[0m\u001b[1;33m+=\u001b[0m\u001b[0mlocal_parameters\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m      9\u001b[0m \u001b[0mprint\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mtotal_parameters\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;31mAttributeError\u001b[0m: 'int' object has no attribute 'value'"
     ]
    }
   ],
   "source": [
    "total_parameters = 0 \n",
    "#iterating over all variables \n",
    "for variable in model.trainable_variables:   \n",
    "    local_parameters=1 \n",
    "    shape = variable.get_shape()  #getting shape of a variable \n",
    "    for i in shape: \n",
    "        local_parameters*=i.value  #mutiplying dimension values \n",
    "    total_parameters+=local_parameters \n",
    "print(total_parameters)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 118,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "dim1 = [0.825,0.8,0.875,0.875,0.75,0.875,0.85,0.7,0.725,0.75,0.75,0.775,0.75,0.725,0.75,0.85,0.675,0.75,0.85,0.825,0.775]\n",
    "dim2 = [0.7,0.9,0.675,0.875,0.725,0.625,0.75,0.775,0.575,0.725,0.75,0.75,0.8,0.7,0.625,0.7,0.65,0.625,0.8,0.675]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 126,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/plain": [
       "<matplotlib.axes._subplots.AxesSubplot at 0x2c321454d48>"
      ]
     },
     "execution_count": 126,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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HWw2bLrOxeOcnDd2DGubQ7y1t4tXdrbxx49jM1ULnDbG5ju4b+m4Yf95+61Ou3btYtSoUZSUlACwaNEiPvzww4QUauM/roRQiObBHkDXIcsNeQWQV4A2qgxyC8BbjFZSBiWlkO+RC4FpKoJGa9RGWGlEjhRkl2aQpRtMcwQosUcos4epcgQpsUXQNHC73XR3mxxcDImmaVBSilZSCmcsxev10tzYAM2H4HADqqkh9rG9FTrbUQ37obMdIn9vwA3kfZQNWPiNG1n0uXMIRgx2twbY4fNT3xFK2Ea5mupnUvuaNWvYuHEj1157LQCrVq2itraWa6655hO3q6mpoaamBoBly5YlJKwQQoxE/TYtj1fHj9d/W11dzbJly4alSN9+++1DPkaiSLbBsWo2q+YCyTZY6Zit30Lt8XhoaWk5+nlLSwsFBQWDOpkQQoiB67dQT5w4kcbGRpqamohEIqxevZp58+YlI5sQQgjAdu+99957shvous6oUaN48sknWblyJUuWLGHhwoUJD1ZRUZHwcwyWZBscq2azai6QbIOVbtn6vZgohBDCXDJOTQghLE4KtRBCWJyp0/H6m5r+pz/9iddffx2bzUZubi7XXXcdRUVFlsj22muv8eqrr6LrOk6nk3/7t39LyCSgwWTrs2bNGh599FEefPBBJk6caHqut956i1/+8pcUFhYCcMEFF7B06dKE54onG8Dq1av59a9/jaZplJeXc8MNN1gi23PPPcfWrVsBCIVCdHR08Nxzz1kim8/nY8WKFfT09GAYBldccQVz5syxRLbm5mZ+/OMf09nZidvt5rvf/S4eT+I3mX7qqadYv349eXl5LF++/DPfV0rx7LPPsmHDBhwOB9dff33//dbKJNFoVH3nO99Rhw4dUuFwWN18882qvr7+E7fZvHmzCgQCSimlXn31VfXoo49aJltPT8/R/3/44Yfq/vvvt0w2pZTq7e1Vd999t7rzzjvVrl27LJHrzTffVD/96U8TnmUw2RoaGtQtt9yiurq6lFJKtbe3WybbsV555RW1YsUKy2R7+umn1auvvqqUUqq+vl5df/31lsm2fPly9eabbyqlYrXkiSeeSEq2rVu3qt27d6v//M//PO73161bpx544AFlGIbasWOHuuOOO/o9pmldH8dOTbfb7Uenph9rxowZOByxdWcrKytpbW21TLasrKyj/w8EAklbxCmebAAvvPACl1xyCRkZGZbKZYZ4sr3++uucf/75uN2xxXvy8vIsk+1Y7777LosXL7ZMNk3T6O2NbVHW29ubtDkW8WQ7cOAAM2fOBGD69OmsXbs2KdmmTZt29PfoeNauXcuZZ56JpmlUVVXR09NDW9vJd7Y3rVC3trZ+4m2Ix+M5aSF+4403mD17djKixZ1t5cqVfPe73+VXv/oV3/jGNyyTra6uDp/Px9y5c5OSKd5cAO+//z4333wzy5cvx5ekdb3jydbQ0EBjYyPf//73ueuuu9i4caNlsvVpbm6mqamJGTNmWCbbl770Jd555x2uvfZaHnzwQa6++mrLZCsvL+f9998H4IMPPsDv99PVZf5uPa2trXi93qOf91f7wMRCreKcmg6x9UX27NnDJZdckuhYQPzZLrjgAp588kmuvPJKfvvb3yYjWr/ZDMPg5z//OV/72teSkqdPPI/Z3LlzWbFiBY888ggzZ85kxYoVlslmGAaNjY3cc8893HDDDTz99NP09PRYIlufd999l4ULF6InaVGxeLK9++67nHXWWTz99NPccccdPPnkkxhG4jeEjSfbVVddxbZt27j11lvZtm0bhYWF2Cywdd5AnvM+phXqeKemf/TRR7z00kvceuutSXsbP9Bp88l8m99ftkAgQH19PT/4wQ/49re/TW1tLQ899BC7d+82NRdATk7O0eewurqaPXv2JDTTQLIVFhYyf/587HY7xcXFlJaW0tjYaIlsfVavXs0ZZ5yR8Ex94sn2xhtvcPrppwNQVVVFOBxOSqs13uf05ptv5qGHHuKf//mfgU92WZrF4/F84t1kPMtymFao45maXldXx//8z/9w6623Jq3PMN5sx76I169fz+jRoy2RLSsri2eeeYYVK1awYsUKKisrufXWWxM+6iOex+zYfri1a9cmbZRMPNkWLFjAli1bAOjs7KSxsfHo0r5mZ4NY10xPTw9VVVUJzzSQbF6v9+jjduDAAcLhMLm5uZbI1tnZebR1/9JLL3H22WcnPFc85s2bx6pVq1BKsXPnTrKysvot1KbOTFy/fj0///nPMQyDs88+m0svvZQXXniBiRMnMm/ePO677z72799Pfn4+EPuluO222yyR7dlnn2Xz5s3YbDbcbjdXX301Y8eOtUS2Y917771cddVVSRme11+u559/nrVr1x59zL75zW9SVlaW8FzxZFNK8Ytf/IKNGzei6zqXXnpp0lqv8TyfL774IuFwmCuvvDIpmeLNduDAAX7yk58QCAQA+OpXv8qsWbMskW3NmjU8//zzaJrG1KlTueaaa5Lyrvzxxx9n27ZtdHV1kZeXx+WXX07kyNrX5513HkopnnnmGTZt2kRmZibXX399v69PmUIuhBAWJzMThRDC4qRQCyGExUmhFkIIi5NCLYQQFieFWgghLE4KtRBCWJwUaiGEsLj/HwJbORUu28l2AAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "sns.distplot(data)\n",
    "sns.distplot(dim2)\n",
    "plt.show()\n",
    "sns.distplot(data2)\n",
    "sns.distplot(dim1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 130,
   "metadata": {},
   "outputs": [],
   "source": [
    "#Import svm model\n",
    "from sklearn import svm\n",
    "\n",
    "#Create a svm Classifier\n",
    "clf = svm.SVC(kernel='linear') # Linear Kernel\n",
    "\n",
    "#Train the model using the training sets\n",
    "clf.fit(comb_data, labels)\n",
    "\n",
    "#Predict the response for test dataset\n",
    "y_pred = clf.predict(comb_data)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 131,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: 0.8826530612244898\n"
     ]
    }
   ],
   "source": [
    "#Import scikit-learn metrics module for accuracy calculation\n",
    "from sklearn import metrics\n",
    "\n",
    "# Model Accuracy: how often is the classifier correct?\n",
    "print(\"Accuracy:\",metrics.accuracy_score(labels, y_pred))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "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.7.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
