{
  "cells": [
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "dMaBZ6e3p0KY"
      },
      "source": [
        "# $\\ell^1$-iOT resolution\n",
        "\n",
        "Use the Lightspeed iOT toolbox to compute the solution of the sparse iOT problem."
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 3,
      "metadata": {
        "id": "ZrtrQcJZUwr6",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "aea23195-eaf9-4987-8461-1d30b3fcc9e5"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Thu Sep 28 12:47:59 2023       \n",
            "+-----------------------------------------------------------------------------+\n",
            "| NVIDIA-SMI 525.105.17   Driver Version: 525.105.17   CUDA Version: 12.0     |\n",
            "|-------------------------------+----------------------+----------------------+\n",
            "| GPU  Name        Persistence-M| Bus-Id        Disp.A | Volatile Uncorr. ECC |\n",
            "| Fan  Temp  Perf  Pwr:Usage/Cap|         Memory-Usage | GPU-Util  Compute M. |\n",
            "|                               |                      |               MIG M. |\n",
            "|===============================+======================+======================|\n",
            "|   0  Tesla V100-SXM2...  Off  | 00000000:00:04.0 Off |                    0 |\n",
            "| N/A   38C    P0    24W / 300W |      0MiB / 16384MiB |      0%      Default |\n",
            "|                               |                      |                  N/A |\n",
            "+-------------------------------+----------------------+----------------------+\n",
            "                                                                               \n",
            "+-----------------------------------------------------------------------------+\n",
            "| Processes:                                                                  |\n",
            "|  GPU   GI   CI        PID   Type   Process name                  GPU Memory |\n",
            "|        ID   ID                                                   Usage      |\n",
            "|=============================================================================|\n",
            "|  No running processes found                                                 |\n",
            "+-----------------------------------------------------------------------------+\n"
          ]
        }
      ],
      "source": [
        "gpu_info = !nvidia-smi\n",
        "gpu_info = '\\n'.join(gpu_info)\n",
        "if gpu_info.find('failed') >= 0:\n",
        "  print('Not connected to a GPU')\n",
        "else:\n",
        "  print(gpu_info)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 4,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "BtgmrRwUp0Ka",
        "outputId": "3c8919c1-20ba-467f-eebf-f1e5bc77b5be"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "The autoreload extension is already loaded. To reload it, use:\n",
            "  %reload_ext autoreload\n"
          ]
        }
      ],
      "source": [
        "%load_ext autoreload\n",
        "%autoreload 2\n",
        "import jax.numpy as jnp\n",
        "from jax import grad, jit, vmap,jacfwd, jacrev, jacobian\n",
        "from jax import random\n",
        "import matplotlib.pyplot as plt\n",
        "import numpy as np\n",
        "import graph_helpers as helpers\n",
        "import gaussian_certificate_asym as cert"
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "Load the graph."
      ],
      "metadata": {
        "id": "cEGVCu-64zk2"
      }
    },
    {
      "cell_type": "code",
      "execution_count": 5,
      "metadata": {
        "id": "AFm9rZlHp0Kb"
      },
      "outputs": [],
      "source": [
        "n = 10\n",
        "Sigma_X = jnp.eye(n)\n",
        "Sigma_Y = jnp.eye(n)\n",
        "graph_type = 'planar'\n",
        "graph_type = 'erdos'\n",
        "graph_type = 'rbm'\n",
        "graph_type = 'circular'\n",
        "A,X = helpers.generate_graph(graph_type,n)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 6,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 448
        },
        "id": "G8YBlOhsp0Kc",
        "outputId": "54dca5e2-7aae-4c57-9962-1ffeae7cd417"
      },
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "[<matplotlib.lines.Line2D at 0x79fffc2c3fd0>]"
            ]
          },
          "metadata": {},
          "execution_count": 6
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ],
            "image/png": "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unWrs69r1656//33TUrk3377299qxowZuu222yRJycnJOnTokNLS0igvl4GZlwsIDg5W3759lZ6eXrvP5XIpPT1dKSkpJibzX4ZhaOrUqVqyZIlWrlypxMREsyP5tREjRmj79u3KyMio3fr166fbb79dGRkZFJdGNnjw4B/dOmDv3r1q166dSYn8W2lpqQIC6l5qAwMD5XK5TErkG1h5uQjTp0/XHXfcoX79+mnAgAGaN2+eSkpKdOedd5odzS9NmTJFixYt0tKlSxUeHq6CggJJks1mU2hoqMnp/E94ePiP5o2aNm2qVq1aMYdkggcffFCDBg3S3Llz9bOf/UwbN27UggULtGDBArOj+aWbbrpJTz/9tBISEtS9e3d9++23+uMf/6j//d//NTuadzNwUV588UUjISHBCA4ONgYMGGB88803ZkfyW5LOur3xxhtmR8MZQ4cONR544AGzY/itjz76yOjRo4dhtVqNpKQkY8GCBWZH8lsOh8N44IEHjISEBCMkJMTo0KGD8fvf/96oqKgwO5pX4z4vAADAqzDzAgAAvArlBQAAeBXKCwAA8CqUFwAA4FUoLwAAwKtQXgAAgFehvAAAAK9CeQEAAF6F8gIAALwK5QUAAHgVygsAAPAqlBcAAOBV/j8KSSuGx2M9iAAAAABJRU5ErkJggg==\n"
          },
          "metadata": {}
        }
      ],
      "source": [
        "v = np.real(np.linalg.eigvals(A))\n",
        "plt.plot( np.sort(v), '.-' )"
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "_dewZ2j1p0Kc"
      },
      "source": [
        "We define the cost as a shifted Laplacian."
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 7,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 448
        },
        "id": "dJlYNJb2p0Kc",
        "outputId": "003cbfcf-6ffa-4f10-8d4a-a7cbca6ff7aa"
      },
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "[<matplotlib.lines.Line2D at 0x79fffc1d7580>]"
            ]
          },
          "metadata": {},
          "execution_count": 7
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ],
            "image/png": "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\n"
          },
          "metadata": {}
        }
      ],
      "source": [
        "cost_mode = 'adjacency'\n",
        "cost_mode = 'laplacian'\n",
        "delta = .03\n",
        "if cost_mode=='laplacian':\n",
        "    # Laplacian to have >0 eigenvalues\n",
        "    C = np.diag(np.sum(A, axis=0)) - A\n",
        "    C = C/np.max(np.real(np.linalg.eigvals(C)))\n",
        "    C = C + delta * jnp.eye(n)\n",
        "elif cost_mode=='adjacency':\n",
        "    C = A/np.max(np.abs(np.linalg.eigvals(A)))\n",
        "    C = C + (1+delta) * jnp.eye(n)\n",
        "v = np.real(np.linalg.eigvals(C))\n",
        "plt.plot( np.sort(v), '.-' )"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 8,
      "metadata": {
        "id": "VAe3ZYkqp0Kc"
      },
      "outputs": [],
      "source": [
        "eps = jnp.inf\n",
        "eps = 1\n",
        "eps = 1\n",
        "eta = cert.compute_precertificate(C, Sigma_X, Sigma_Y, eps)\n",
        "eta = eta.reshape((n,n))"
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "wBtX1N1Wp0Kc"
      },
      "source": [
        "Display the values of the certificates as a function of the geodesic distance (so in particular, when using combinatorial geodesic distances, it must be ==1 for distance=0 and 1)."
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 9,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 448
        },
        "id": "on7y-ae2p0Kc",
        "outputId": "972c4f4b-4029-4b75-9a1a-1f903ce3fc59"
      },
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "[<matplotlib.lines.Line2D at 0x79ffe47e9a80>]"
            ]
          },
          "metadata": {},
          "execution_count": 9
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ],
            "image/png": "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\n"
          },
          "metadata": {}
        }
      ],
      "source": [
        "D = helpers.floyd_warshall(A)\n",
        "plt.plot( D.flatten(), eta.flatten(), '.' )"
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "ymVA9Ukep0Kd"
      },
      "source": [
        "## Test for recovery using samples\n",
        "\n",
        "Compute the covariance of the transport plan."
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 10,
      "metadata": {
        "id": "-lEtVvtvp0Kd"
      },
      "outputs": [],
      "source": [
        "epsilon = 2\n",
        "Sigma_XY = cert.Sigma_xy_fun(Sigma_X, Sigma_Y,C,epsilon)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 11,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 448
        },
        "id": "lNBCLt5Qp0Kd",
        "outputId": "154cccc7-45ff-42a4-dfa5-0622517c04d2"
      },
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "<matplotlib.image.AxesImage at 0x79ffe43b9960>"
            ]
          },
          "metadata": {},
          "execution_count": 11
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ],
            "image/png": "iVBORw0KGgoAAAANSUhEUgAAAZgAAAGdCAYAAAAv9mXmAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/bCgiHAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAU7ElEQVR4nO3df2zVhb3/8TcUWqopBHGgjUWZWYICCgga5Ru3RSL69cdMFqcJJgQTt2xFQRIz2KLGOKgsmyERh2I2RzJRTBaiMxHiZVHmJgFBjWabsGu+rpMAmphWa1Jre+4fu7f38j3T2wN98/mc+ngk5w9PzvG88uG0z3x62nNGVSqVSgDAMBtd9AAARiaBASCFwACQQmAASCEwAKQQGABSCAwAKQQGgBRjTvYDDgwMxKFDh6KlpSVGjRp1sh8egBNQqVTio48+itbW1hg9+ovPUU56YA4dOhRtbW0n+2EBGEadnZ1x1llnfeFtTnpgWlpaIiLi/8T/jTEx9mQ//OfaduDNoidUuenfFxY9oconfeX5N4MMp4ztK3pCla3n/lvREwZ1fzwQZ8/9f4Pfy7/ISQ/Mf/1YbEyMjTGjyvPNanxL+V6OGntqY9ETqozpK98mGE5jx5bvR/dl/P40lJc4yrcagBFBYABIITAApBAYAFIIDAApBAaAFAIDQAqBASCFwACQQmAASCEwAKQ4rsA8/PDDcc4558S4cePikksuiT179gz3LgDqXM2B2bp1a6xcuTLuvffe2L9/f1x44YWxaNGiOHr0aMY+AOpUzYF58MEH47bbboulS5fG+eefH4888kiccsop8atf/SpjHwB1qqbAfPrpp7Fv375YuPC/P6dk9OjRsXDhwnjllVf+5X16e3uju7v7mAsAI19Ngfnggw+iv78/pkyZcsz1U6ZMicOHD//L+3R0dMSECRMGLz7NEuDLIf23yFavXh1dXV2Dl87OzuyHBKAEavpEy9NPPz0aGhriyJEjx1x/5MiROOOMM/7lfZqamqKpqen4FwJQl2o6g2lsbIyLLroodu7cOXjdwMBA7Ny5My699NJhHwdA/arpDCYiYuXKlbFkyZKYN29eXHzxxbF+/fro6emJpUuXZuwDoE7VHJibbrop3n///bjnnnvi8OHDMXv27Ni+fXvVC/8AfLnVHJiIiGXLlsWyZcuGewsAI4j3IgMghcAAkEJgAEghMACkEBgAUggMACkEBoAUAgNACoEBIIXAAJBCYABIcVzvRTYcth14M8a3lKdvi1pnFz2hyo5D24ueUOX6g1cVPaFKT19j0RMYQcr4fCrT111fz6cRsWlIty3Pd3gARhSBASCFwACQQmAASCEwAKQQGABSCAwAKQQGgBQCA0AKgQEghcAAkEJgAEghMACkEBgAUggMACkEBoAUAgNACoEBIIXAAJBCYABIITAApBAYAFIIDAApBAaAFAIDQAqBASCFwACQQmAASCEwAKQQGABSCAwAKQQGgBQCA0AKgQEghcAAkEJgAEgxpqgHvunfF8bYUxuLevgqOw5tL3pClUWts4ueUKWMx+n6g1cVPaFKT195ntvUvzI9nz7rqwz5ts5gAEghMACkEBgAUggMACkEBoAUAgNACoEBIIXAAJBCYABIITAApBAYAFIIDAApBAaAFAIDQIqaAtPR0RHz58+PlpaWmDx5ctxwww3x9ttvZ20DoI7VFJiXXnop2tvbY/fu3fHCCy9EX19fXHnlldHT05O1D4A6VdMHjm3ffuyHTf3617+OyZMnx759++Lyyy8f1mEA1LcT+kTLrq6uiIg47bTTPvc2vb290dvbO/jf3d3dJ/KQANSJ436Rf2BgIFasWBELFiyImTNnfu7tOjo6YsKECYOXtra2431IAOrIcQemvb093nrrrXjqqae+8HarV6+Orq6uwUtnZ+fxPiQAdeS4fkS2bNmyeO6552LXrl1x1llnfeFtm5qaoqmp6bjGAVC/agpMpVKJ22+/PbZt2xYvvvhiTJs2LWsXAHWupsC0t7fHli1b4plnnomWlpY4fPhwRERMmDAhmpubUwYCUJ9qeg1m48aN0dXVFd/4xjfizDPPHLxs3bo1ax8AdarmH5EBwFB4LzIAUggMACkEBoAUAgNACoEBIIXAAJBCYABIITAApBAYAFIIDAApBAaAFCf0kckn4pO+sTGmr7Goh69y/cGrip5QZceh7UVPqLKodXbRE6qU8TiV8fnUU6KvN74cnMEAkEJgAEghMACkEBgAUggMACkEBoAUAgNACoEBIIXAAJBCYABIITAApBAYAFIIDAApBAaAFAIDQAqBASCFwACQQmAASCEwAKQQGABSCAwAKQQGgBQCA0AKgQEghcAAkEJgAEghMACkEBgAUggMACkEBoAUAgNACoEBIIXAAJBCYABIITAApBAYAFKMKXpAWfT0NRY9ocr1B68qekKVHYe2Fz2hyqLW2UVPqFLG41TG51MZv+4YPs5gAEghMACkEBgAUggMACkEBoAUAgNACoEBIIXAAJBCYABIITAApBAYAFIIDAApBAaAFAIDQIoTCswDDzwQo0aNihUrVgzTHABGiuMOzN69e+PRRx+NCy64YDj3ADBCHFdgPv7441i8eHE89thjMXHixOHeBMAIcFyBaW9vj2uuuSYWLlz4v962t7c3uru7j7kAMPLV/JHJTz31VOzfvz/27t07pNt3dHTEfffdV/MwAOpbTWcwnZ2dsXz58njiiSdi3LhxQ7rP6tWro6ura/DS2dl5XEMBqC81ncHs27cvjh49GnPnzh28rr+/P3bt2hUbNmyI3t7eaGhoOOY+TU1N0dTUNDxrAagbNQXmiiuuiDfffPOY65YuXRrTp0+PH/7wh1VxAeDLq6bAtLS0xMyZM4+57tRTT41JkyZVXQ/Al5u/5AcgRc2/Rfb/e/HFF4dhBgAjjTMYAFIIDAApBAaAFAIDQAqBASCFwACQQmAASCEwAKQQGABSCAwAKQQGgBQn/F5k5Onpayx6QpXrD15V9IQqOw5tL3pClUWts4ueUKWMx6mMz6cyft3VK2cwAKQQGABSCAwAKQQGgBQCA0AKgQEghcAAkEJgAEghMACkEBgAUggMACkEBoAUAgNACoEBIIXAAJBCYABIITAApBAYAFIIDAApBAaAFAIDQAqBASCFwACQQmAASCEwAKQQGABSCAwAKQQGgBQCA0AKgQEghcAAkEJgAEghMACkEBgAUggMACkEBoAUY4oeQH3p6WssekKV6w9eVfSEKjsObS96QpVFrbOLnlCljMepjM+nMn7dDYUzGABSCAwAKQQGgBQCA0AKgQEghcAAkEJgAEghMACkEBgAUggMACkEBoAUAgNACoEBIIXAAJCi5sC89957ccstt8SkSZOiubk5Zs2aFa+++mrGNgDqWE2fB/Phhx/GggUL4pvf/GY8//zz8ZWvfCUOHjwYEydOzNoHQJ2qKTDr1q2Ltra2ePzxxwevmzZt2rCPAqD+1fQjsmeffTbmzZsXN954Y0yePDnmzJkTjz322Bfep7e3N7q7u4+5ADDy1RSYd955JzZu3Bhf+9rXYseOHfH9738/7rjjjti8efPn3qejoyMmTJgweGlrazvh0QCUX02BGRgYiLlz58batWtjzpw58d3vfjduu+22eOSRRz73PqtXr46urq7BS2dn5wmPBqD8agrMmWeeGeeff/4x15133nnx97///XPv09TUFOPHjz/mAsDIV1NgFixYEG+//fYx1x04cCDOPvvsYR0FQP2rKTB33nln7N69O9auXRt/+9vfYsuWLbFp06Zob2/P2gdAnaopMPPnz49t27bFk08+GTNnzoz7778/1q9fH4sXL87aB0CdqunvYCIirr322rj22msztgAwgngvMgBSCAwAKQQGgBQCA0AKgQEghcAAkEJgAEghMACkEBgAUggMACkEBoAUNb8XGZRNT19j0ROqXH/wqqInVNlxaHvRE6osap1d9IQqZTxOZXo+9Y3tG/JtncEAkEJgAEghMACkEBgAUggMACkEBoAUAgNACoEBIIXAAJBCYABIITAApBAYAFIIDAApBAaAFAIDQAqBASCFwACQQmAASCEwAKQQGABSCAwAKQQGgBQCA0AKgQEghcAAkEJgAEghMACkEBgAUggMACkEBoAUAgNACoEBIIXAAJBCYABIITAApBAYAFKMKeqBTxnbF2PHjirq4av09DUWPYERpIzPp+sPXlX0hCo7Dm0vekKVRa2zi55QpUzHqfujgZg4xNs6gwEghcAAkEJgAEghMACkEBgAUggMACkEBoAUAgNACoEBIIXAAJBCYABIITAApBAYAFIIDAApagpMf39/3H333TFt2rRobm6Oc889N+6///6oVCpZ+wCoUzV9Hsy6deti48aNsXnz5pgxY0a8+uqrsXTp0pgwYULccccdWRsBqEM1BeZPf/pTfOtb34prrrkmIiLOOeecePLJJ2PPnj0p4wCoXzX9iOyyyy6LnTt3xoEDByIi4o033oiXX345rr766s+9T29vb3R3dx9zAWDkq+kMZtWqVdHd3R3Tp0+PhoaG6O/vjzVr1sTixYs/9z4dHR1x3333nfBQAOpLTWcwTz/9dDzxxBOxZcuW2L9/f2zevDl+9rOfxebNmz/3PqtXr46urq7BS2dn5wmPBqD8ajqDueuuu2LVqlVx8803R0TErFmz4t13342Ojo5YsmTJv7xPU1NTNDU1nfhSAOpKTWcwn3zySYwefexdGhoaYmBgYFhHAVD/ajqDue6662LNmjUxderUmDFjRrz22mvx4IMPxq233pq1D4A6VVNgHnroobj77rvjBz/4QRw9ejRaW1vje9/7Xtxzzz1Z+wCoUzUFpqWlJdavXx/r169PmgPASOG9yABIITAApBAYAFIIDAApBAaAFAIDQAqBASCFwACQQmAASCEwAKQQGABS1PReZMNp67n/FuNbytO36w9eVfSEKj19jUVPYAQp4/OpjF93Ow5tL3pClUWts4ueMOizSl9EvDOk25bnOzwAI4rAAJBCYABIITAApBAYAFIIDAApBAaAFAIDQAqBASCFwACQQmAASCEwAKQQGABSCAwAKQQGgBQCA0AKgQEghcAAkEJgAEghMACkEBgAUggMACkEBoAUAgNACoEBIIXAAJBCYABIITAApBAYAFIIDAApBAaAFAIDQAqBASCFwACQQmAASDHmZD9gpVKJiIjujwdO9kN/ob6eT4ueUOWzvkrREyBV39i+oidU6f6oXN+bIiI+q5TnOH0W/9zyX9/Lv8ioylBuNYz+8Y9/RFtb28l8SACGWWdnZ5x11llfeJuTHpiBgYE4dOhQtLS0xKhRo477/9Pd3R1tbW3R2dkZ48ePH8aFI4vjNDSO09A4TkMzko9TpVKJjz76KFpbW2P06C9+leWk/4hs9OjR/2v1ajF+/PgR9w+YwXEaGsdpaBynoRmpx2nChAlDup0X+QFIITAApKjbwDQ1NcW9994bTU1NRU8pNcdpaBynoXGchsZx+qeT/iI/AF8OdXsGA0C5CQwAKQQGgBQCA0CKug3Mww8/HOecc06MGzcuLrnkktizZ0/Rk0qlo6Mj5s+fHy0tLTF58uS44YYb4u233y56Vqk98MADMWrUqFixYkXRU0rnvffei1tuuSUmTZoUzc3NMWvWrHj11VeLnlUq/f39cffdd8e0adOiubk5zj333Lj//vuH9J5dI1VdBmbr1q2xcuXKuPfee2P//v1x4YUXxqJFi+Lo0aNFTyuNl156Kdrb22P37t3xwgsvRF9fX1x55ZXR09NT9LRS2rt3bzz66KNxwQUXFD2ldD788MNYsGBBjB07Np5//vn485//HD//+c9j4sSJRU8rlXXr1sXGjRtjw4YN8Ze//CXWrVsXP/3pT+Ohhx4qelph6vLXlC+55JKYP39+bNiwISL++f5mbW1tcfvtt8eqVasKXldO77//fkyePDleeumluPzyy4ueUyoff/xxzJ07N37xi1/ET37yk5g9e3asX7++6FmlsWrVqvjjH/8Yf/jDH4qeUmrXXnttTJkyJX75y18OXvftb387mpub4ze/+U2By4pTd2cwn376aezbty8WLlw4eN3o0aNj4cKF8corrxS4rNy6uroiIuK0004reEn5tLe3xzXXXHPMc4r/9uyzz8a8efPixhtvjMmTJ8ecOXPiscceK3pW6Vx22WWxc+fOOHDgQEREvPHGG/Hyyy/H1VdfXfCy4pz0N7s8UR988EH09/fHlClTjrl+ypQp8de//rWgVeU2MDAQK1asiAULFsTMmTOLnlMqTz31VOzfvz/27t1b9JTSeuedd2Ljxo2xcuXK+NGPfhR79+6NO+64IxobG2PJkiVFzyuNVatWRXd3d0yfPj0aGhqiv78/1qxZE4sXLy56WmHqLjDUrr29Pd566614+eWXi55SKp2dnbF8+fJ44YUXYty4cUXPKa2BgYGYN29erF27NiIi5syZE2+99VY88sgjAvM/PP300/HEE0/Eli1bYsaMGfH666/HihUrorW19Ut7nOouMKeffno0NDTEkSNHjrn+yJEjccYZZxS0qryWLVsWzz33XOzatWtYPyZhJNi3b18cPXo05s6dO3hdf39/7Nq1KzZs2BC9vb3R0NBQ4MJyOPPMM+P8888/5rrzzjsvfvvb3xa0qJzuuuuuWLVqVdx8880RETFr1qx49913o6Oj40sbmLp7DaaxsTEuuuii2Llz5+B1AwMDsXPnzrj00ksLXFYulUolli1bFtu2bYvf//73MW3atKInlc4VV1wRb775Zrz++uuDl3nz5sXixYvj9ddfF5f/tGDBgqpfcT9w4ECcffbZBS0qp08++aTqA7gaGhpiYKB8H8F8stTdGUxExMqVK2PJkiUxb968uPjii2P9+vXR09MTS5cuLXpaabS3t8eWLVvimWeeiZaWljh8+HBE/PODgpqbmwteVw4tLS1Vr0mdeuqpMWnSJK9V/Q933nlnXHbZZbF27dr4zne+E3v27IlNmzbFpk2bip5WKtddd12sWbMmpk6dGjNmzIjXXnstHnzwwbj11luLnlacSp166KGHKlOnTq00NjZWLr744sru3buLnlQqEfEvL48//njR00rt61//emX58uVFzyid3/3ud5WZM2dWmpqaKtOnT69s2rSp6Eml093dXVm+fHll6tSplXHjxlW++tWvVn784x9Xent7i55WmLr8OxgAyq/uXoMBoD4IDAApBAaAFAIDQAqBASCFwACQQmAASCEwAKQQGABSCAwAKQQGgBQCA0CK/wABxayp8yNafQAAAABJRU5ErkJggg==\n"
          },
          "metadata": {}
        }
      ],
      "source": [
        "plt.imshow(Sigma_XY)"
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "Call the cvxpy solver leveraging the parametric problem (not real unparameteric iOT)"
      ],
      "metadata": {
        "id": "9BJEL9I_5DfY"
      }
    },
    {
      "cell_type": "code",
      "execution_count": 12,
      "metadata": {
        "id": "N64461Ndp0Ke"
      },
      "outputs": [],
      "source": [
        "if 0:\n",
        "  gamma = 0.01 # regul param\n",
        "  SampleSigma_x=SampleCov[:n,:n]\n",
        "  SampleSigma_xy=SampleCov[:n,n:]\n",
        "  SampleSigma_y=SampleCov[n:,n:]\n",
        "\n",
        "  primal, dual = cert.cvxpy_solver(SampleSigma_x, SampleSigma_y, SampleSigma_xy, gamma, verbose=False)\n",
        "  primal *= epsilon"
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "Just a single resolution to check."
      ],
      "metadata": {
        "id": "UkR0MY0r5Qts"
      }
    },
    {
      "cell_type": "code",
      "execution_count": 13,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 448
        },
        "id": "8JqRqCNqp0Ke",
        "outputId": "b73501a6-5cd3-4f3d-f9b2-7144de88957c"
      },
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "<matplotlib.image.AxesImage at 0x79ffe42a24d0>"
            ]
          },
          "metadata": {},
          "execution_count": 13
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ],
            "image/png": "iVBORw0KGgoAAAANSUhEUgAAAZgAAAGdCAYAAAAv9mXmAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/bCgiHAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAWdElEQVR4nO3df4yVhb3n8S8Mzg91ZBELwhUETbMooIKgq+Ta3kj0GvTWbK+tCSZc3LVNOyjIXlNoV11jYaS3NSRiUUxruVvxx7YhWhPtsrRKaeXyS42kFmq9V6e6gN66M4LtDM6c/aPb6WWfaucAX55z8PVKzh+ePMfnkyPyzjMHzjOoUqlUAgCOsMFlDwDg2CQwAKQQGABSCAwAKQQGgBQCA0AKgQEghcAAkGLI0T5hX19fvPnmm9Ha2hqDBg062qcH4DBUKpV49913Y/To0TF48Idfoxz1wLz55psxZsyYo31aAI6gjo6OOO200z70mKMemNbW1oiIuOTEz8SQQccd7dN/oKXPrS97QsHs7XPLnlAw/S9eL3tCQePg98ueUPDzd04te0JBa2N32RMK/uWtk8ueUPCX435V9oSCvz15a9kT+r23ry+unfHP/b+Xf5ijHpg//FhsyKDjYsigxqN9+g90YmvtfRzVcHxz2RMKGk+snf9mf9D4Zy7TyzCkp6nsCQVDau8/XQze79f4QJzQ2lD2hIKBfMRRe/9nAnBMEBgAUggMACkEBoAUAgNACoEBIIXAAJBCYABIITAApBAYAFIIDAApDikw9957b4wbNy6am5vjwgsvjM2bNx/pXQDUuaoD8+ijj8bChQvj9ttvj+3bt8e5554bl19+eezduzdjHwB1qurA3H333XHDDTfE3Llz4+yzz4777rsvjj/++Pj2t7+dsQ+AOlVVYHp6emLbtm0xc+bMP/4LBg+OmTNnxnPPPfcnX9Pd3R1dXV0HPQA49lUVmLfffjt6e3tj5MiRBz0/cuTI2L179598TXt7ewwdOrT/4W6WAB8N6X+KbPHixdHZ2dn/6OjoyD4lADWgqjtannLKKdHQ0BB79uw56Pk9e/bEqaf+6VvENjU1RVNT7d3dD4BcVV3BNDY2xvnnnx/r1//x/vV9fX2xfv36uOiii474OADqV1VXMBERCxcujDlz5sS0adPiggsuiOXLl8f+/ftj7ty5GfsAqFNVB+azn/1svPXWW3HbbbfF7t2747zzzounn3668ME/AB9tVQcmImLevHkxb968I70FgGOI7yIDIIXAAJBCYABIITAApBAYAFIIDAApBAaAFAIDQAqBASCFwACQQmAASHFI30V2JHx907o4sbV2+tb2t18oe0LB3//jurInFPz4nQllTyjY+OqZZU8oOP/018ueUNDV01z2hIKP/bt9ZU8o2NhxRtkTCnr6SvutuqBnX09E/GpAx9bO7/AAHFMEBoAUAgNACoEBIIXAAJBCYABIITAApBAYAFIIDAApBAaAFAIDQAqBASCFwACQQmAASCEwAKQQGABSCAwAKQQGgBQCA0AKgQEghcAAkEJgAEghMACkEBgAUggMACkEBoAUAgNACoEBIIXAAJBCYABIITAApBAYAFIIDAApBAaAFAIDQAqBASDFkLJO/F/++T/GkBOayjp9wd9998myJxT8499cWvaEgn/44X8ve0LBf+68ruwJBac2d5U9oeCffnFG2RMKGk/sKXtCwfHNtbfp5d+MLHtCv9793QM+1hUMACkEBoAUAgNACoEBIIXAAJBCYABIITAApBAYAFIIDAApBAaAFAIDQAqBASCFwACQQmAASFFVYNrb22P69OnR2toaI0aMiKuvvjp27tyZtQ2AOlZVYJ599tloa2uLTZs2xbp16+LAgQNx2WWXxf79+7P2AVCnqrrh2NNPP33QP3/nO9+JESNGxLZt2+KSSy45osMAqG+HdUfLzs7OiIg4+eSTP/CY7u7u6O7+4x3Qurpq705/ABx5h/whf19fXyxYsCBmzJgRkyZN+sDj2tvbY+jQof2PMWPGHOopAagjhxyYtra22LFjRzzyyCMfetzixYujs7Oz/9HR0XGopwSgjhzSj8jmzZsXTz75ZGzYsCFOO+20Dz22qakpmpqaDmkcAPWrqsBUKpW48cYbY+3atfHMM8/E+PHjs3YBUOeqCkxbW1usWbMmHn/88WhtbY3du3dHRMTQoUOjpaUlZSAA9amqz2BWrlwZnZ2d8clPfjJGjRrV/3j00Uez9gFQp6r+ERkADITvIgMghcAAkEJgAEghMACkEBgAUggMACkEBoAUAgNACoEBIIXAAJBCYABIcVi3TD4c4078TTSe2FjW6Qte6R5Z9oSC//b0w2VPKPj7yZeVPaHguZ3fL3tCwfnbPlP2hIIzTt9b9oSCy0a+XPaEggdf/g9lTygYP+xfy57Q78CgngEf6woGgBQCA0AKgQEghcAAkEJgAEghMACkEBgAUggMACkEBoAUAgNACoEBIIXAAJBCYABIITAApBAYAFIIDAApBAaAFAIDQAqBASCFwACQQmAASCEwAKQQGABSCAwAKQQGgBQCA0AKgQEghcAAkEJgAEghMACkEBgAUggMACkEBoAUAgNACoEBIIXAAJBiSFknfn3/sBgSTWWdvuBXXaeUPaHg5++OKntCwYodPyh7QsGsqZ8ue0LB53+8sewJBQ//enrZEwpW77yw7AkFPb89ruwJBT19pf1WXXCgr2/Ax7qCASCFwACQQmAASCEwAKQQGABSCAwAKQQGgBQCA0AKgQEghcAAkEJgAEghMACkEBgAUggMACkOKzB33XVXDBo0KBYsWHCE5gBwrDjkwGzZsiXuv//+OOecc47kHgCOEYcUmH379sXs2bPjgQceiGHDhh3pTQAcAw4pMG1tbTFr1qyYOXPmnz22u7s7urq6DnoAcOyr+j6cjzzySGzfvj22bNkyoOPb29vjjjvuqHoYAPWtqiuYjo6OmD9/fjz00EPR3Nw8oNcsXrw4Ojs7+x8dHR2HNBSA+lLVFcy2bdti7969MXXq1P7nent7Y8OGDbFixYro7u6OhoaGg17T1NQUTU1NR2YtAHWjqsBceuml8dJLLx303Ny5c2PChAnxpS99qRAXAD66qgpMa2trTJo06aDnTjjhhBg+fHjheQA+2vxNfgBSVP2nyP5/zzzzzBGYAcCxxhUMACkEBoAUAgNACoEBIIXAAJBCYABIITAApBAYAFIIDAApBAaAFAIDQIrD/i6yQ3WgryEqfbXz9f573z2x7AkFPTX0/vzBtS9dX/aEgsuffrnsCQVrzx9X9oSCH7/yeNkTCu7+zRllTyjY+d7IsicU/OiVf1/2hH597/1uwMe6ggEghcAAkEJgAEghMACkEBgAUggMACkEBoAUAgNACoEBIIXAAJBCYABIITAApBAYAFIIDAApBAaAFAIDQAqBASCFwACQQmAASCEwAKQQGABSCAwAKQQGgBQCA0AKgQEghcAAkEJgAEghMACkEBgAUggMACkEBoAUAgNACoEBIIXAAJBCYABIITAApBhS1olPbn4vjmt+v6zTF+zraSx7QsHgQZWyJxS803V82RMKntn98bIn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K77//e+XtKg23XLLLbFo0aK49tprIyJi8uTJ8dprr0V7e/tHNjB19xlMY2NjnH/++bF+/fr+5/r6+mL9+vVx0UUXlbistlQqlZg3b16sXbs2fvSjH8X48ePLnlRzLr300njppZfihRde6H9MmzYtZs+eHS+88IK4/D8zZswo/BH3Xbt2xemnn17Sotr03nvvFW7A1dDQEH19tXcL5qOl7q5gIiIWLlwYc+bMiWnTpsUFF1wQy5cvj/3798fcuXPLnlYz2traYs2aNfH4449Ha2tr7N69OyJ+f6OglpaWktfVhtbW1sJnUieccEIMHz7cZ1X/xs033xwXX3xxLF26ND7zmc/E5s2bY9WqVbFq1aqyp9WUq666KpYsWRJjx46NiRMnxvPPPx933313XH/99WVPK0+lTt1zzz2VsWPHVhobGysXXHBBZdOmTWVPqikR8ScfDz74YNnTatonPvGJyvz588ueUXN+8IMfVCZNmlRpamqqTJgwobJq1aqyJ9Wcrq6uyvz58ytjx46tNDc3V84444zKV77ylUp3d3fZ00pTl38PBoDaV3efwQBQHwQGgBQCA0AKgQEghcAAkEJgAEghMACkEBgAUggMACkEBoAUAgNACoEBIMX/Bdzk9X0wr+OcAAAAAElFTkSuQmCC\n"
          },
          "metadata": {}
        }
      ],
      "source": [
        "import SISTA as sista\n",
        "\n",
        "Sigma_XY = cert.Sigma_xy_fun(Sigma_X, Sigma_Y,C,eps)\n",
        "\n",
        "#covariance of coupling between alpha and beta\n",
        "Cov= jnp.block( [[Sigma_X, Sigma_XY],[Sigma_XY.T, Sigma_Y]])\n",
        "u,s,vh = jnp.linalg.svd(Cov)\n",
        "Cov_sqrt = u@jnp.diag(jnp.sqrt(s))@vh\n",
        "\n",
        "# sample from coupling distribution\n",
        "N = 10000//10 #number of samples\n",
        "key = random.PRNGKey(493321)\n",
        "samples = Cov_sqrt@random.normal(key,shape=(n+n,N)) # + mean[:,None]\n",
        "samples_x = samples[:n,:]\n",
        "samples_y = samples[n:,:]\n",
        "\n",
        "plt.imshow( np.cov(samples)[:n,n:] )"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 14,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "ncBkOvfAWfZ5",
        "outputId": "3e6fc55e-c20b-4fdc-9237-6dbe145143b0"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Time taken: 6.502605199813843 seconds\n",
            "Time taken: 2.341014862060547 seconds\n"
          ]
        }
      ],
      "source": [
        "import SISTA as sista\n",
        "import time\n",
        "\n",
        "gamma = .1\n",
        "\n",
        "#create cost operator with samples\n",
        "cost = sista.Eucl_Cost(samples_x.T, samples_y.T)\n",
        "hat_pi = jnp.eye(N)/N\n",
        "\n",
        "#call bfgs\n",
        "x_init = random.normal(key,shape=(2*n*n+N,))\n",
        "start_time = time.time()\n",
        "beta, res = sista.iOT_VarPro(hat_pi,  eps, x_init, cost, gamma = gamma, niter=30)\n",
        "print(f\"Time taken: { time.time() - start_time} seconds\")\n",
        "start_time = time.time()\n",
        "beta_e, res = sista.iOT_VarPro_emp(N,  eps, x_init, cost, gamma = gamma, niter=30)\n",
        "print(f\"Time taken: { time.time() - start_time} seconds\")\n",
        "beta = np.reshape( beta, (n,n))"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 15,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 448
        },
        "id": "X3vHV3HVWjUD",
        "outputId": "3d9f895e-9d83-45ba-bb88-bf346fb2e32b"
      },
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "<matplotlib.image.AxesImage at 0x79ffe4117be0>"
            ]
          },
          "metadata": {},
          "execution_count": 15
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ],
            "image/png": "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rS3uvPPOWLp0acHratPHH38co0aNildeeSWuuuqqoufUlAMHDsT06dPjpz/9afzwhz+MSy+9NNauXVv0rJqxdOnSePXVV+O3v/1t0VNq2ty5c2P06NHxs5/9rO/YzTffHC0tLfHYY48VuKw4dXcFc/jw4di1a1fMmTOn71hjY2PMmTMnXnvttQKX1baurq6IiDjnnHMKXlJ72tvb4/rrrz/m7xT/5rnnnosZM2bELbfcEqNGjYpp06bFI488UvSsmnPllVfGli1bYs+ePRER8dZbb8W2bdviuuuuK3hZcU77m12eqk8++SR6e3tj9OjRxxwfPXp0vPvuuwWtqm2lUikWL14cs2fPjilTphQ9p6Y8+eSTsXv37ti5c2fRU2rW+++/H+vXr48lS5bE97///di5c2fcddddMXTo0Jg/f37R82rG0qVLo7u7OyZOnBhNTU3R29sbK1eujHnz5hU9rTB1Fxiq197eHu+8805s27at6Ck1pbOzMxYtWhQvvfRSDBs2rOg5NatUKsWMGTNi1apVERExbdq0eOedd2LDhg0C83eefvrpePzxx2Pz5s0xefLkePPNN2Px4sUxduzYL+zzVHeBOffcc6OpqSn27dt3zPF9+/bFeeedV9Cq2rVw4cJ4/vnnY+vWrQP6MQmDwa5du2L//v0xffr0vmO9vb2xdevWWLduXfT09ERTU1OBC2vDmDFjYtKkScccu+iii+KXv/xlQYtq0z333BNLly6NW2+9NSIipk6dGh988EF0dHR8YQNTd6/BDB06NC677LLYsmVL37FSqRRbtmyJK664osBltaVcLsfChQvjmWeeid/85jcxYcKEoifVnKuvvjrefvvtePPNN/tuM2bMiHnz5sWbb74pLv/P7NmzK37Efc+ePXH++ecXtKg2HTp0qOIDuJqamqJUqp2POz7d6u4KJiJiyZIlMX/+/JgxY0bMmjUr1q5dGwcPHowFCxYUPa1mtLe3x+bNm+PZZ5+N1tbW2Lt3b0T89YOCWlpaCl5XG1pbWytekzrrrLNi5MiRXqv6O3fffXdceeWVsWrVqvjGN74RO3bsiI0bN8bGjRuLnlZTbrjhhli5cmWMHz8+Jk+eHG+88UasWbMmbr/99qKnFadcpx588MHy+PHjy0OHDi3PmjWrvH379qIn1ZSI+Nzbo48+WvS0mvbVr361vGjRoqJn1Jxf/epX5SlTppSbm5vLEydOLG/cuLHoSTWnu7u7vGjRovL48ePLw4YNK3/5y18u/+AHPyj39PQUPa0wdfnvYACofXX3GgwA9UFgAEghMACkEBgAUggMACkEBoAUAgNACoEBIIXAAJBCYABIITAApBAYAFL8X+SukISQMhX9AAAAAElFTkSuQmCC\n"
          },
          "metadata": {}
        }
      ],
      "source": [
        "plt.imshow( beta  )"
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "Loop for several lambda, N and epsilon."
      ],
      "metadata": {
        "id": "w0Q0pu-h5NCX"
      }
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "232rsdlHFOsQ",
        "outputId": "87c21303-3047-423c-c6aa-5fdc9509cc16"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            " 82% (443 of 540) |##################    | Elapsed Time: 0:18:29 ETA:   0:03:22"
          ]
        }
      ],
      "source": [
        "import SISTA as sista\n",
        "from progressbar import ProgressBar\n",
        "\n",
        "nrep = 15 # number of replications\n",
        "niter = 50 # iteration of BFGS\n",
        "nlambda = 6 # number of tested lambda\n",
        "nsample = 3 # number of tested #samples\n",
        "neps = 2\n",
        "epsilon_list = np.linspace(1,10,neps)\n",
        "gamma_list = np.linspace(.01, .3, nlambda)\n",
        "N_list = np.linspace(200,5000,nsample).astype(np.int32)\n",
        "Err = np.zeros((nlambda,nsample,nrep,neps))\n",
        "tol = 1e-2\n",
        "\n",
        "key = random.PRNGKey(493321)\n",
        "\n",
        "k = 0\n",
        "bar = ProgressBar(max_value=nlambda*nsample*nrep*neps)\n",
        "for ieps in range(neps):\n",
        "  eps = epsilon_list[ieps]\n",
        "  Sigma_XY = cert.Sigma_xy_fun(Sigma_X, Sigma_Y,C,eps)\n",
        "  Cov= jnp.block( [[Sigma_X, Sigma_XY],[Sigma_XY.T, Sigma_Y]])\n",
        "  u,s,vh = jnp.linalg.svd(Cov)\n",
        "  Cov_sqrt = u@jnp.diag(jnp.sqrt(s))@vh\n",
        "  for isample in range(nsample):\n",
        "    # draw N sample\n",
        "    N = N_list[isample]\n",
        "    for irep in range(nrep):\n",
        "      key, subkey = random.split(key)\n",
        "      samples = Cov_sqrt@random.normal(key,shape=(n+n,N))\n",
        "      samples_x = samples[:n,:]\n",
        "      samples_y = samples[n:,:]\n",
        "      cost = sista.Eucl_Cost(samples_x.T, samples_y.T)\n",
        "      # hat_pi = jnp.eye(N)/N\n",
        "      for igamma in range(nlambda):\n",
        "        # solve SISTA\n",
        "        gamma = gamma_list[igamma]\n",
        "        x_init = random.normal(key,shape=(2*n*n+N,))\n",
        "    #    beta, res = sista.iOT_VarPro(hat_pi,  eps, x_init, cost, gamma = gamma, niter=niter)\n",
        "        beta, res = sista.iOT_VarPro_emp(N,  eps, x_init, cost, gamma = gamma, niter=niter)\n",
        "        beta = np.reshape( beta, (n,n))\n",
        "        Err[igamma,isample,irep,ieps] = jnp.sum( (jnp.abs(beta)>tol) != (jnp.abs(C)>tol) )\n",
        "        k += 1\n",
        "        bar.update(k)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "a0O7XohoLRwC"
      },
      "outputs": [],
      "source": [
        "ieps = 0 # change this to vizualize another epsilon\n",
        "s,p,q,_ = Err.shape\n",
        "fig, ax = plt.subplots()\n",
        "for i in range(p):\n",
        "    color = (i / (p - 1), 0, 1 - i / (p - 1))\n",
        "    mean_curve = np.mean(Err[:, i, :,ieps], axis=1)\n",
        "    std_curve = np.std(Err[:, i, :,ieps], axis=1)\n",
        "    lower_bound = mean_curve - 1 * std_curve\n",
        "    upper_bound = mean_curve + 1 * std_curve\n",
        "    x = np.arange(s)\n",
        "    plt.plot(x, mean_curve, color=color, label=f'$N$={N_list[i]}')\n",
        "    plt.fill_between(x, lower_bound, upper_bound, color=color, alpha=0.2)\n",
        "ax.set_ylim(0, np.max(Err[:, :, :,ieps].flatten()))\n",
        "plt.legend(loc='best')\n",
        "filename = f\"{graph_type}-perfs-eps{ieps}.pdf\"\n",
        "plt.savefig(filename, format='pdf', bbox_inches='tight')"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "LFpMKfiCmGOS"
      },
      "outputs": [],
      "source": [
        "np.savez('variables.npz', Err=Err, epsilon_list=epsilon_list, N_list=N_list)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "Qvzahj-7rcOZ"
      },
      "outputs": [],
      "source": []
    }
  ],
  "metadata": {
    "accelerator": "GPU",
    "colab": {
      "machine_shape": "hm",
      "provenance": [],
      "gpuType": "V100"
    },
    "kernelspec": {
      "display_name": "Python 3",
      "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.10.8"
    }
  },
  "nbformat": 4,
  "nbformat_minor": 0
}