{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/opt/anaconda3/envs/QPL/lib/python3.8/site-packages/scipy/__init__.py:146: UserWarning: A NumPy version >=1.16.5 and <1.23.0 is required for this version of SciPy (detected version 1.24.3\n",
      "  warnings.warn(f\"A NumPy version >={np_minversion} and <{np_maxversion}\"\n"
     ]
    }
   ],
   "source": [
    "import pygsti\n",
    "import numpy as np\n",
    "from matplotlib import pyplot as plt\n",
    "import json\n",
    "import pickle\n",
    "import os\n",
    "import pandas as pd\n",
    "\n",
    "from helper import *\n",
    "\n",
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Simulation notebook"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "This notebook simulates all of the circuits in a dataset. The first third of the notebook loads up or creates an error model. The second third simulates the random *i.i.d.*-layered circuits, and the final third simulates the random mirror circuits."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Select the dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 58,
   "metadata": {},
   "outputs": [],
   "source": [
    "exp_num = 4"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Set your hyperparameters"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 59,
   "metadata": {},
   "outputs": [],
   "source": [
    "typ = 'H'\n",
    "exp_path = f'./experiment_{exp_num}'\n",
    "sim_path = exp_path + '/simulation_results/'\n",
    "circ_path = exp_path + '/circuits/'\n",
    "\n",
    "try:\n",
    "    os.makedirs(sim_path)\n",
    "except:\n",
    "    print('Blah')\n",
    "\n",
    "with open('./experiment_0' + '/pspec.pkl', 'rb') as f:\n",
    "    pspec = pickle.load(f)\n",
    "\n",
    "max_stochastic = {'S': .0005, 'H': 0, 'H+S': .0001}\n",
    "max_hamiltonian = {'S': 0, 'H': .00025, 'H+S': .0001}\n",
    "max_strengths = {1: {'S': max_stochastic[typ], 'H': max_hamiltonian[typ]},\n",
    "                 2: {'S': 3*max_stochastic[typ], 'H': 3*max_hamiltonian[typ]}\n",
    "                 }\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Construct the noise models or load an existing noise model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 60,
   "metadata": {},
   "outputs": [],
   "source": [
    "target_model = pygsti.models.create_crosstalk_free_model(processor_spec = pspec)\n",
    "\n",
    "if exp_num == 0:\n",
    "    error_rates_dict = sample_error_rates_dict(pspec, max_strengths)\n",
    "    error_model = pygsti.models.create_crosstalk_free_model(pspec, lindblad_error_coeffs=error_rates_dict)\n",
    "\n",
    "    with open(exp_path+'./error_rates.json', 'wb') as f:\n",
    "        pickle.dump({'gate-errors': error_rates_dict}, \n",
    "                 f)\n",
    "\n",
    "    with open(exp_path + '/error_model.pkl', 'wb') as f:\n",
    "        pickle.dump(error_model, f)\n",
    "else:\n",
    "    with open('./experiment_0' + '/error_model.pkl', 'rb') as f:\n",
    "        error_model = pickle.load(f)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Simulate the random *i.i.d.*-layered circuits"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 61,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "CPU times: user 1.76 s, sys: 10.7 ms, total: 1.77 s\n",
      "Wall time: 1.77 s\n"
     ]
    }
   ],
   "source": [
    "%%time\n",
    "\n",
    "circuits = pygsti.io.read_circuit_list(circ_path + '/iid_layer_random_clifford_circuits_v1_instance_0.txt')\n",
    "good_circuits = []\n",
    "for c in circuits:\n",
    "    new_c = c.copy(editable = True)\n",
    "    # new_c.map_state_space_labels_inplace({0: 'Q0', 1: 'Q1', 2: 'Q2', 3: 'Q3'})\n",
    "    new_c.done_editing()\n",
    "    good_circuits.append(new_c)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 62,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0 ( 29 ),100 ( 19 ),200 ( 34 ),300 ( 167 ),400 ( 1 ),500 ( 30 ),600 ( 3 ),700 ( 5 ),800 ( 24 ),900 ( 13 ),1000 ( 5 ),1100 ( 12 ),1200 ( 44 ),1300 ( 14 ),1400 ( 69 ),1500 ( 6 ),1600 ( 19 ),1700 ( 6 ),1800 ( 94 ),1900 ( 21 ),2000 ( 8 ),2100 ( 19 ),2200 ( 39 ),2300 ( 9 ),2400 ( 143 ),2500 ( 6 ),2600 ( 180 ),2700 ( 22 ),2800 ( 26 ),2900 ( 26 ),3000 ( 19 ),3100 ( 30 ),3200 ( 1 ),3300 ( 13 ),3400 ( 28 ),3500 ( 21 ),3600 ( 20 ),3700 ( 7 ),3800 ( 14 ),3900 ( 17 ),4000 ( 16 ),4100 ( 3 ),4200 ( 127 ),4300 ( 5 ),4400 ( 14 ),4500 ( 14 ),4600 ( 4 ),4700 ( 11 ),4800 ( 1 ),4900 ( 178 ),CPU times: user 1h 39s, sys: 56.9 s, total: 1h 1min 36s\n",
      "Wall time: 10min 22s\n"
     ]
    }
   ],
   "source": [
    "%%time\n",
    "fids = []\n",
    "for i, c in enumerate(good_circuits):\n",
    "    #print(i, '(', (c.depth), ')', end=',')\n",
    "    if i % 100 == 0:\n",
    "        print(i, '(', (c.depth), ')', end=',')\n",
    "    pmatrix = error_model.circuit_operator(c).to_dense()\n",
    "    target_pmatrix = target_model.circuit_operator(c).to_dense()\n",
    "    fids.append(pygsti.entanglement_fidelity(pmatrix, target_pmatrix))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Visualize the distribution of fidelities."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 63,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(array([1.000e+00, 3.000e+00, 4.000e+00, 1.300e+01, 1.700e+01, 4.100e+01,\n",
       "        1.320e+02, 3.990e+02, 1.433e+03, 2.957e+03]),\n",
       " array([0.88503467, 0.8965275 , 0.90802034, 0.91951318, 0.93100601,\n",
       "        0.94249885, 0.95399169, 0.96548452, 0.97697736, 0.9884702 ,\n",
       "        0.99996303]),\n",
       " <BarContainer object of 10 artists>)"
      ]
     },
     "execution_count": 63,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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yspKSdLOnTvlOI4bMJKUkZEhx3Hcmcv5/X41NjYG3AAAQPfV4REzfvx4FRcX65133tFzzz2nPXv26L777pPf75ck1dXVKTIyUr179w54XWJiourq6tyZhISENttOSEhwZy5XVFTkvn/GcRylpKR08JEBAIDrSdA/TvoykydPdv87NTVVQ4cOVb9+/bR582Y99NBDV32dMUYej8e9/5f/fbWZv7R48WLNmzfPvd/Y2EjIAADQjYX8I9ZJSUnq16+fPvzwQ0mS1+tVS0uLGhoaAubq6+uVmJjozpw8ebLNtk6dOuXOXC4qKkqxsbEBNwAA0H2FPGJOnz6t48ePKykpSZKUlpamiIgIVVRUuDO1tbU6cOCAsrKyJEmZmZny+XzavXu3O7Nr1y75fD53BgAA3NiC/nHS2bNn9dFHH7n3jx49qpqaGsXFxSkuLk5LlizRww8/rKSkJB07dkxPPPGE4uPj9eCDD0qSHMfRtGnTNH/+fPXp00dxcXFasGCBhgwZ4n5aadCgQRo3bpymT5+uNWvWSJJmzJihnJwcPpkEAAAktSNi9u7dq5EjR7r3L70PZcqUKVq9erX279+vDRs26MyZM0pKStLIkSP12muvKSYmxn3NihUrFB4erkmTJqm5uVmjRo3S+vXrFRYW5s4UFxdr7ty57qeYcnNzr/ndNAAA4MbiMcaYrt6JUGhsbJTjOPL5fLw/BgBw3bt10eau3oWgHVt6f4dvM5jzN787CQAAWKnDP2INAEBXsvGKBtqHKzEAAMBKRAwAALASEQMAAKxExAAAACsRMQAAwEpEDAAAsBIRAwAArETEAAAAKxExAADASkQMAACwEhEDAACsRMQAAAArETEAAMBKRAwAALASEQMAAKxExAAAACsRMQAAwEpEDAAAsBIRAwAArETEAAAAKxExAADASkQMAACwEhEDAACsRMQAAAArETEAAMBKRAwAALASEQMAAKxExAAAACsRMQAAwEpEDAAAsBIRAwAArETEAAAAKxExAADASkQMAACwEhEDAACsRMQAAAArETEAAMBKRAwAALASEQMAAKxExAAAACsRMQAAwEpEDAAAsBIRAwAArETEAAAAKxExAADASkQMAACwUtAR895772nChAlKTk6Wx+PRW2+9FfC8MUZLlixRcnKyoqOjNWLECL3//vsBM36/X3PmzFF8fLx69eql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6/Y9MQAAANdCxAAAACsRMQAAwEpEDAAAsJKVEbNq1Sr1799fPXr0UFpamrZv337N+eLiYt19993q2bOnkpKS9IMf/ECnT58OmPn5z3+ugQMHKjo6WikpKfrhD3+oP//5z6E8jC4R7No9//zzGjRokKKjozVw4EBt2LChzcwbb7yhwYMHKyoqSoMHD1ZpaWmodr/LdPS6rV27Vvfee6969+6t3r17a/To0dq9e3coD6HLhOLv3CUlJSXyeDyaOHFiB+911wvFup05c0azZs1SUlKSevTooUGDBqmsrCxUh9BlQrF23f0c8d5772nChAlKTk6Wx+PRW2+99aWv2bZtm9LS0tSjRw/ddttt+tWvftVmJuTnB2OZkpISExERYdauXWsOHjxoHn/8cdOrVy/zP//zP1ec3759u7npppvML37xC3PkyBGzfft28+1vf9tMnDjRnXn55ZdNVFSUKS4uNkePHjW///3vTVJSkikoKOisw+oUwa7dqlWrTExMjCkpKTEff/yxefXVV83NN99sNm3a5M5UVlaasLAwU1hYaA4dOmQKCwtNeHi4qaqq6qzDCrlQrFteXp55/vnnzb59+8yhQ4fMD37wA+M4jjlx4kRnHVanCMXaXXLs2DHzzW9+09x7773mgQceCPGRdK5QrJvf7zdDhw41f/u3f2t27Nhhjh07ZrZv325qamo667A6RSjW7kY4R5SVlZknn3zSvPHGG0aSKS0tveb8kSNHTM+ePc3jjz9uDh48aNauXWsiIiLM66+/7s50xvnBuoj53ve+Zx577LGAx+68806zaNGiK87/9Kc/NbfddlvAY7/85S9N37593fuzZs0y9913X8DMvHnzzLBhwzpor68Pwa5dZmamWbBgQcBjjz/+uLnnnnvc+5MmTTLjxo0LmBk7dqx55JFHOmivu14o1u1yFy5cMDExMeY3v/nN19/h60io1u7ChQvmnnvuMS+88IKZMmVKt4uYUKzb6tWrzW233WZaWlo6foevI6FYuxvlHHHJV4mYhQsXmjvvvDPgsUcffdRkZGS49zvj/GDVj5NaWlpUXV2t7OzsgMezs7NVWVl5xddkZWXpxIkTKisrkzFGJ0+e1Ouvv67777/fnRk2bJiqq6vdy/lHjhxRWVlZwIzt2rN2fr9fPXr0CHgsOjpau3fv1vnz5yVJO3fubLPNsWPHXnWbtgnVul3u888/1/nz5xUXF9cxO34dCOXa/fjHP9Ytt9yiadOmdfyOd7FQrdumTZuUmZmpWbNmKTExUampqSosLFRra2toDqQLhGrtboRzRLCu9m//3r17O/X8YFXEfPbZZ2ptbW3zCyATExPb/KLIS7KyslRcXKzJkycrMjJSXq9X3/jGN/Qf//Ef7swjjzyip59+WsOGDVNERIRuv/12jRw5UosWLQrp8XSm9qzd2LFj9cILL6i6ulrGGO3du1cvvviizp8/r88++0ySVFdXF9Q2bROqdbvcokWL9M1vflOjR4/u8GPoKqFauz/84Q9at26d1q5dG/Jj6AqhWrcjR47o9ddfV2trq8rKyvQv//Iveu655/TMM8+E/Jg6S6jW7kY4RwTrav/2X7hwoVPPD1ZFzCWXfiv1JcaYNo9dcvDgQc2dO1f/9m//purqapWXl+vo0aMBv39p69ateuaZZ7Rq1Sr993//t95880397ne/09NPPx3S4+gKwazdv/7rv2r8+PHKyMhQRESEHnjgAU2dOlWSFBYW1q5t2ioU63bJsmXL9Oqrr+rNN99s8/8Iu4OOXLumpib9wz/8g9auXav4+PhQ73qX6ui/cxcvXlRCQoJ+/etfKy0tTY888oiefPJJrV69OqTH0RU6eu1upHNEMK60zpc/Hurzg1UREx8fr7CwsDYVV19f36b2LikqKtI999yjf/7nf9Zdd92lsWPHatWqVXrxxRdVW1sr6Yu/xPn5+frHf/xHDRkyRA8++KAKCwtVVFSkixcvhvy4OkN71i46OlovvviiPv/8cx07dkyffPKJbr31VsXExLgnEK/XG9Q2bROqdbvkZz/7mQoLC7VlyxbdddddITuOrhCKtfv444917NgxTZgwQeHh4QoPD9eGDRu0adMmhYeH6+OPP+6MQwupUP2dS0pK0h133BEQ0oMGDVJdXZ1aWlpCd0CdKFRrdyOcI4J1tX/7w8PD1adPn2vOdOT5waqIiYyMVFpamioqKgIer6ioUFZW1hVf8/nnn+ummwIP89L/iC9V49VmzBdvfO6o3e9S7Vm7SyIiItS3b1+FhYWppKREOTk57nplZma22eaWLVu+dJu2CNW6SdJPf/pTPf300yovL9fQoUNDsv9dKRRrd+edd2r//v2qqalxb7m5uRo5cqRqamqUkpISykPqFKH6O3fPPffoo48+Cjjp/ulPf1JSUpIiIyM7/kC6QKjW7kY4RwTrav/2Dx06VBEREdec6dDzQ4e9RbiTXPr43Lp168zBgwdNQUGB6dWrlzl27JgxxphFixaZ/Px8d/6ll14y4eHhZtWqVebjjz82O3bsMEOHDjXf+9733JmnnnrKxMTEmFdffdUcOXLEbNmyxdx+++1m0qRJnX58oRTs2h0+fNhs3LjR/OlPfzK7du0ykydPNnFxcebo0aPuzB/+8AcTFhZmli5dag4dOmSWLl3abT9i3ZHr9uyzz5rIyEjz+uuvm9raWvfW1NTU2YcXUqFYu8t1x08nhWLdPvnkE3PzzTeb2bNnm8OHD5vf/e53JiEhwfzkJz/p7MMLqVCs3Y1wjmhqajL79u0z+/btM5LM8uXLzb59+9yPpl++bpc+Yv3DH/7QHDx40Kxbt67NR6w74/xgXcQYY8zzzz9v+vXrZyIjI813vvMds23bNve5KVOmmOHDhwfM//KXvzSDBw820dHRJikpyfz93/99wPdxnD9/3ixZssTcfvvtpkePHiYlJcXMnDnTNDQ0dNIRdZ5g1u7gwYPmr//6r010dLSJjY01DzzwgPnggw/abPM///M/zcCBA01ERIS58847zRtvvNEZh9KpOnrd+vXrZyS1uT311FOddESdJxR/5/5Sd4wYY0KzbpWVlSY9Pd1ERUWZ2267zTzzzDPmwoULnXE4naqj1+5GOEe8++67V/w3acqUKcaYK59bt27dav7mb/7GREZGmltvvdWsXr26zXZDfX7wGHODXgsDAABWs+o9MQAAAJcQMQAAwEpEDAAAsBIRAwAArETEAAAAKxExAADASkQMAACwEhEDAACsRMQAAAArETEAAMBKRAwAALASEQMAAKz0/wAHi3/n7OWVcAAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.hist(fids)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Save the fidelites (as a json and a dataframe)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 64,
   "metadata": {},
   "outputs": [],
   "source": [
    "### As a json ###\n",
    "\n",
    "with open(sim_path+'/fidelities.json', 'w') as f:\n",
    "    json.dump(fids, f)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 65,
   "metadata": {},
   "outputs": [],
   "source": [
    "### As a dataframe ###\n",
    "\n",
    "def create_dataframe(circs, fidelities):\n",
    "    dfdict = {'Circuit': [],\n",
    "                'D:SP': [],\n",
    "                'F:Depth': [],\n",
    "                'F:Width': [],\n",
    "                'F:2QGC': []\n",
    "                }\n",
    "\n",
    "    for circ, fid in zip(circs, fidelities):\n",
    "        dfdict['Circuit'].append(circ.str)\n",
    "        dfdict['F:Width'].append(circ.width)\n",
    "        dfdict['F:Depth'].append(circ.depth)\n",
    "        dfdict['F:2QGC'].append(circ.two_q_gate_count())\n",
    "        dfdict['D:SP'].append(fid)\n",
    "\n",
    "    df = pd.DataFrame(dfdict)\n",
    "    return df\n",
    "df = create_dataframe(good_circuits, fids)\n",
    "df.drop_duplicates(subset='Circuit', keep='first', inplace=True) # drop any unnecesarry duplicates\n",
    "df.reset_index(drop=True, inplace=True)\n",
    "\n",
    "df.to_csv(sim_path+'/dataframe.csv', index = False)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Simulate the mirror circuits (if you have them)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Fill in."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 66,
   "metadata": {},
   "outputs": [],
   "source": [
    "with open('./experiment_0/error_model.pkl', 'rb') as f:\n",
    "    error_model = pickle.load(f)\n",
    "\n",
    "df_old = pd.read_csv(sim_path + '/dataframe.csv')\n",
    "max_depth = max(df_old['F:Depth'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 67,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "CPU times: user 612 ms, sys: 4.83 ms, total: 617 ms\n",
      "Wall time: 660 ms\n"
     ]
    }
   ],
   "source": [
    "%%time\n",
    "other_circuits = pygsti.io.read_circuit_list(circ_path + '/mirrored_random_clifford_circuits_v1_instance_0.txt')\n",
    "good_other_circuits = []\n",
    "for c in other_circuits:\n",
    "    new_c = c.copy(editable = True)\n",
    "    # new_c.map_state_space_labels_inplace({0: 'Q0', 1: 'Q1', 2: 'Q2', 3: 'Q3'})\n",
    "    new_c.done_editing()\n",
    "    good_other_circuits.append(new_c)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 68,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0 ( 172 ),100 ( 113 ),200 ( 158 ),300 ( 93 ),400 ( 124 ),500 ( 109 ),600 ( 133 ),700 ( 143 ),CPU times: user 25min 6s, sys: 20.3 s, total: 25min 27s\n",
      "Wall time: 4min 15s\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "(array([  5.,  13.,  19.,  31.,  48.,  59.,  81., 118., 155., 221.]),\n",
       " array([0.87303486, 0.88571747, 0.89840007, 0.91108268, 0.92376528,\n",
       "        0.93644789, 0.94913049, 0.9618131 , 0.9744957 , 0.98717831,\n",
       "        0.99986091]),\n",
       " <BarContainer object of 10 artists>)"
      ]
     },
     "execution_count": 68,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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dgQKAJAdgQIAZEegAADZESgAQHYECgCQHYECAGRHoAAA2REoAEB2BAoAkB2BAgBkR6AAANkRKABAdgQKAJAdgQIAZEegAADZESgAQHYECgCQHYECAGRHoAAA2REoAEB2yjt6AgCdwaDZz3T0FKBTcQcFAMiOQAEAsiNQAIDsCBQAIDsCBQDIjkABALIjUACA7AgUACA7AgUAyI5AAQCyI1AAgOwIFAAgOwIFAMiOQAEAsiNQAIDsCBQAIDsCBQDIjkABALIjUACA7AgUACA7AgUAyI5AAQCyI1AAgOwIFAAgOwIFAMiOQAEAsiNQAIDsCBQAIDsCBQDIjkABALIjUACA7AgUACA7AgUAyI5AAQCyU97REwA41KDZz3T0FIAO5g4KAJAdgQIAZEegAADZESgAQHYECgCQHYECAGRHoAAA2REoAEB2BAoAkJ2iB8q8efOirKys1VZTU1N4PqUU8+bNi7q6uujRo0dceOGF8corrxR7GgDACawkd1A+85nPxI4dOwrbSy+9VHjurrvuigULFsTChQvjhRdeiJqamrj44otj9+7dpZgKAHACKkmglJeXR01NTWE7+eSTI+I3d0/uueeemDt3blx55ZUxdOjQeOCBB+L999+PRx99tBRTAQBOQCUJlNdffz3q6upi8ODBcdVVV8XmzZsjImLLli3R2NgY48ePL4ytqKiICy64INatW1eKqQAAJ6Ci/zXjkSNHxoMPPhinn356/OpXv4rbb789xowZE6+88ko0NjZGRER1dXWr11RXV8cvf/nLox6zpaUlWlpaCo+bm5uLPW0AICNFD5SJEycW/nnYsGExevToOO200+KBBx6IUaNGRUREWVlZq9eklA7b9//Nnz8/vva1rxV7qgBApkr+MeNevXrFsGHD4vXXXy98mufgnZSDdu7cedhdlf9vzpw50dTUVNi2bdtW0jkDAB2r5IHS0tISr776atTW1sbgwYOjpqYmVq1aVXh+3759sXr16hgzZsxRj1FRURG9e/dutQEAnVfRf8Qza9asmDRpUnz605+OnTt3xu233x7Nzc0xderUKCsrixkzZkRDQ0MMGTIkhgwZEg0NDdGzZ8+4+uqriz0VICIGzX6mo6cA0GZFD5Tt27fHn/7pn8Y777wTJ598cowaNSqef/75GDhwYERE3HLLLbF37964/vrrY9euXTFy5MhYuXJlVFZWFnsqAMAJqiyllDp6Em3V3NwcVVVV0dTU5Mc98Fu4gwIci613XlL0Y7bl+7e/xQMAZEegAADZESgAQHYECgCQHYECAGRHoAAA2REoAEB2BAoAkB2BAgBkR6AAANkRKABAdgQKAJAdgQIAZEegAADZESgAQHYECgCQHYECAGRHoAAA2REoAEB2BAoAkB2BAgBkR6AAANkRKABAdgQKAJCd8o6eAJxIBs1+pqOnAPCx4A4KAJAdgQIAZEegAADZESgAQHYECgCQHYECAGRHoAAA2REoAEB2BAoAkB2BAgBkR6AAANkRKABAdgQKAJAdgQIAZEegAADZESgAQHYECgCQHYECAGSnvKMnwMfXoNnPdPQUAMiUOygAQHYECgCQHYECAGRHoAAA2REoAEB2BAoAkB2BAgBkR6AAANkRKABAdvwm2U7Cb2UFoDNxBwUAyI5AAQCyI1AAgOwIFAAgOwIFAMiOQAEAsiNQAIDsCBQAIDsCBQDIjkABALIjUACA7PhbPEfg79oAQMdyBwUAyI5AAQCyI1AAgOx0aKAsWrQoBg8eHN27d4/hw4fHmjVrOnI6AEAmOixQfvCDH8SMGTNi7ty5sWHDhjj//PNj4sSJ8dZbb3XUlACATHRYoCxYsCCmTZsWX/7yl+Oss86Ke+65J+rr62Px4sUdNSUAIBMd8jHjffv2xfr162P27Nmt9o8fPz7WrVt32PiWlpZoaWkpPG5qaoqIiObm5pLM78OW90tyXAA4UZTie+zBY6aUfuvYDgmUd955Jw4cOBDV1dWt9ldXV0djY+Nh4+fPnx9f+9rXDttfX19fsjkCwMdZ1T2lO/bu3bujqqrqI8d06C9qKysra/U4pXTYvoiIOXPmxMyZMwuPP/zww/if//mf6Nev3xHH5665uTnq6+tj27Zt0bt3746eTlaszdFZm6OzNkdnbY7O2hxdqdYmpRS7d++Ourq63zq2QwKlf//+0aVLl8PuluzcufOwuyoRERUVFVFRUdFq3yc/+clSTrFd9O7d238UR2Ftjs7aHJ21OTprc3TW5uhKsTa/7c7JQR3yJtlu3brF8OHDY9WqVa32r1q1KsaMGdMRUwIAMtJhP+KZOXNmXHPNNTFixIgYPXp0fPe734233norvvKVr3TUlACATHRYoEyZMiXefffd+PrXvx47duyIoUOHxooVK2LgwIEdNaV2U1FREbfeeuthP7bC2nwUa3N01uborM3RWZujy2FtytLv8lkfAIB25G/xAADZESgAQHYECgCQHYECAGRHoBTBokWLYvDgwdG9e/cYPnx4rFmz5iPHP/LII3HOOedEz549o7a2Nr70pS/Fu+++22rMPffcE2eccUb06NEj6uvr46tf/Wr87//+bylPoyTaujb33ntvnHXWWdGjR48444wz4sEHHzxszBNPPBFnn312VFRUxNlnnx3Lly8v1fRLqthrs2TJkjj//POjT58+0adPnxg3blz8/Oc/L+UplEwpvm4OWrZsWZSVlcUVV1xR5Fm3j1Ksza9//euYPn161NbWRvfu3eOss86KFStWlOoUSqoU63OiX4+fe+65mDRpUtTV1UVZWVk89dRTv/U1q1evjuHDh0f37t3j1FNPjX/6p386bEzJr8WJ47Js2bLUtWvXtGTJkrRp06Z00003pV69eqVf/vKXRxy/Zs2adNJJJ6V/+Id/SJs3b05r1qxJn/nMZ9IVV1xRGPPwww+nioqK9Mgjj6QtW7akf/3Xf021tbVpxowZ7XVaRdHWtVm0aFGqrKxMy5YtS2+++WZ67LHH0ic+8Yn09NNPF8asW7cudenSJTU0NKRXX301NTQ0pPLy8vT888+312kVRSnW5uqrr0733ntv2rBhQ3r11VfTl770pVRVVZW2b9/eXqdVFKVYm4O2bt2aPvWpT6Xzzz8/XX755SU+k+Irxdq0tLSkESNGpD/+4z9Oa9euTVu3bk1r1qxJGzdubK/TKppSrE9nuB6vWLEizZ07Nz3xxBMpItLy5cs/cvzmzZtTz54900033ZQ2bdqUlixZkrp27Zoef/zxwpj2uBYLlOP0uc99Ln3lK19pte/MM89Ms2fPPuL4b37zm+nUU09tte873/lOOuWUUwqPp0+fnj7/+c+3GjNz5sx03nnnFWnW7aOtazN69Og0a9asVvtuuummdO655xYeT548OX3hC19oNWbChAnpqquuKtKs20cp1uZQH3zwQaqsrEwPPPDA8U+4HZVqbT744IN07rnnpu9973tp6tSpJ2SglGJtFi9enE499dS0b9++4k+4nZVifTrL9fig3yVQbrnllnTmmWe22nfdddelUaNGFR63x7XYj3iOw759+2L9+vUxfvz4VvvHjx8f69atO+JrxowZE9u3b48VK1ZESil+9atfxeOPPx6XXHJJYcx5550X69evL9ye37x5c6xYsaLVmNwdy9q0tLRE9+7dW+3r0aNH/PznP4/9+/dHRMS//du/HXbMCRMmHPWYOSrV2hzq/fffj/3790ffvn2LM/F2UMq1+frXvx4nn3xyTJs2rfgTbwelWpunn346Ro8eHdOnT4/q6uoYOnRoNDQ0xIEDB0pzIiVSqvXpDNfjtjradfbFF19s12uxQDkO77zzThw4cOCwP3BYXV192B9CPGjMmDHxyCOPxJQpU6Jbt25RU1MTn/zkJ+Mf//EfC2OuuuqquO222+K8886Lrl27xmmnnRZjx46N2bNnl/R8iulY1mbChAnxve99L9avXx8ppXjxxRfjvvvui/3798c777wTERGNjY1tOmaOSrU2h5o9e3Z86lOfinHjxhX9HEqlVGvzs5/9LJYuXRpLliwp+TmUSqnWZvPmzfH444/HgQMHYsWKFfG3f/u38e1vfzvuuOOOkp9TMZVqfTrD9bitjnad/eCDD9r1WixQiqCsrKzV45TSYfsO2rRpU/zlX/5l/P3f/32sX78+fvSjH8WWLVta/Q2iZ599Nu64445YtGhR/Md//Ec8+eST8cMf/jBuu+22kp5HKbRlbf7u7/4uJk6cGKNGjYquXbvG5ZdfHtdee21ERHTp0uWYjpmzUqzNQXfddVc89thj8eSTTx72f4gngmKuze7du+PP//zPY8mSJdG/f/9ST73kiv118+GHH8aAAQPiu9/9bgwfPjyuuuqqmDt3bixevLik51EqxV6fznQ9bosjreOh+0t9LRYox6F///7RpUuXw4px586dh5XlQfPnz49zzz03/uqv/io++9nPxoQJE2LRokVx3333xY4dOyLiN//RXHPNNfHlL385hg0bFl/84hejoaEh5s+fHx9++GHJz6sYjmVtevToEffdd1+8//77sXXr1njrrbdi0KBBUVlZWfjGUlNT06Zj5qhUa3PQt771rWhoaIiVK1fGZz/72ZKdRymUYm3efPPN2Lp1a0yaNCnKy8ujvLw8HnzwwXj66aejvLw83nzzzfY4teNWqq+b2traOP3001uF7llnnRWNjY2xb9++0p1QkZVqfTrD9bitjnadLS8vj379+n3kmGJeiwXKcejWrVsMHz48Vq1a1Wr/qlWrYsyYMUd8zfvvvx8nndR62Q9eGA4W6tHGpN+8qblY0y+pY1mbg7p27RqnnHJKdOnSJZYtWxaXXnppYT1Gjx592DFXrlz5W4+Zk1KtTUTEN7/5zbjtttviRz/6UYwYMaIk8y+lUqzNmWeeGS+99FJs3LixsF122WUxduzY2LhxY9TX15fylIqmVF835557brzxxhutvtn+4he/iNra2ujWrVvxT6RESrU+neF63FZHu86OGDEiunbt+pFjinotLtrbbT+mDn6sbenSpWnTpk1pxowZqVevXmnr1q0ppZRmz56drrnmmsL473//+6m8vDwtWrQovfnmm2nt2rVpxIgR6XOf+1xhzK233poqKyvTY489ljZv3pxWrlyZTjvttDR58uR2P7/j0da1ee2119JDDz2UfvGLX6R///d/T1OmTEl9+/ZNW7ZsKYz52c9+lrp06ZLuvPPO9Oqrr6Y777zzhP6YcTHX5hvf+Ebq1q1bevzxx9OOHTsK2+7du9v79I5LKdbmUCfqp3hKsTZvvfVW+sQnPpFuuOGG9Nprr6Uf/vCHacCAAen2229v79M7bqVYn85wPd69e3fasGFD2rBhQ4qItGDBgrRhw4bCx68PXZeDHzP+6le/mjZt2pSWLl162MeM2+NaLFCK4N57700DBw5M3bp1S3/wB3+QVq9eXXhu6tSp6YILLmg1/jvf+U46++yzU48ePVJtbW36sz/7s1a/q2L//v1p3rx56bTTTkvdu3dP9fX16frrr0+7du1qpzMqnraszaZNm9Lv/d7vpR49eqTevXunyy+/PP3Xf/3XYcf853/+53TGGWekrl27pjPPPDM98cQT7XEqRVfstRk4cGCKiMO2W2+9tZ3OqHhK8XXz/52ogZJSadZm3bp1aeTIkamioiKdeuqp6Y477kgffPBBe5xO0RV7fTrD9finP/3pEa8NU6dOTSkd+fvUs88+m37/938/devWLQ0aNCgtXrz4sOOW+lpcllInvUcFAJywvAcFAMiOQAEAsiNQAIDsCBQAIDsCBQDIjkABALIjUACA7AgUACA7AgUAyI5AAQCyI1AAgOwIFAAgO/8HToxNOEJObe4AAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "%%time\n",
    "other_fids = []\n",
    "for i, c in enumerate(good_other_circuits):\n",
    "    #print(i, '(', (c.depth), ')', end=',')\n",
    "    if i % 100 == 0:\n",
    "        print(i, '(', (c.depth), ')', end=',')\n",
    "    pmatrix = error_model.circuit_operator(c).to_dense()\n",
    "    target_pmatrix = target_model.circuit_operator(c).to_dense()\n",
    "    other_fids.append(pygsti.entanglement_fidelity(pmatrix, target_pmatrix))\n",
    "plt.hist(other_fids)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 69,
   "metadata": {},
   "outputs": [],
   "source": [
    "def create_dataframe(circs, fidelities):\n",
    "    dfdict = {'Circuit': [],\n",
    "                'D:SP': [],\n",
    "                'F:Depth': [],\n",
    "                'F:Width': [],\n",
    "                'F:2QGC': []\n",
    "                }\n",
    "\n",
    "    for circ, fid in zip(circs, fidelities):\n",
    "        dfdict['Circuit'].append(circ.str)\n",
    "        dfdict['F:Width'].append(circ.width)\n",
    "        dfdict['F:Depth'].append(circ.depth)\n",
    "        dfdict['F:2QGC'].append(circ.two_q_gate_count())\n",
    "        dfdict['D:SP'].append(fid)\n",
    "\n",
    "    df = pd.DataFrame(dfdict)\n",
    "    return df\n",
    "df_other = create_dataframe(good_other_circuits, other_fids)\n",
    "df_other.drop_duplicates(subset='Circuit', keep='first', inplace=True) # drop any unnecesarry duplicates\n",
    "df_other = df_other[df_other['F:Depth'] <= max_depth]\n",
    "good_other_circ_indices = np.array(df_other.index)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 70,
   "metadata": {},
   "outputs": [],
   "source": [
    "# get rid of the circuits that are too long\n",
    "\n",
    "good_other_fids = np.array(other_fids)\n",
    "good_other_fids = good_other_fids[good_other_circ_indices]\n",
    "good_other_fids = list(good_other_fids)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 71,
   "metadata": {},
   "outputs": [],
   "source": [
    "with open(sim_path+'/mirrored_fidelities.json', 'w') as f:\n",
    "    json.dump(good_other_fids, f)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 72,
   "metadata": {},
   "outputs": [],
   "source": [
    "df_other.reset_index(drop=True, inplace=True)\n",
    "df_other.to_csv(sim_path+'/mirrored_dataframe.csv', index = False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "QPL",
   "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.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
