{
  "nbformat": 4,
  "nbformat_minor": 0,
  "metadata": {
    "colab": {
      "provenance": [],
      "collapsed_sections": []
    },
    "kernelspec": {
      "name": "python3",
      "display_name": "Python 3"
    },
    "language_info": {
      "name": "python"
    },
    "accelerator": "GPU"
  },
  "cells": [
    {
      "cell_type": "code",
      "execution_count": 2,
      "metadata": {
        "id": "WfZ7lfx7PMnj"
      },
      "outputs": [],
      "source": [
        "import numpy as np\n",
        "import time\n",
        "import matplotlib.pyplot as plt\n",
        "from sklearn import linear_model\n",
        "import numpy as np\n",
        "\n",
        "\n",
        "\n",
        "%matplotlib inline"
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "\n",
        "## to make lasso even slower, try N = 400000\n",
        "N = 40000\n",
        "M = 100\n",
        "\n",
        "## this is a compressive sensing task with sparse feature\n",
        "A = np.random.normal(0.0, 1.0/ N**0.5, (M, N))\n",
        "x_0 = np.random.normal(0.0, 1.0, N)\n",
        "x_0 = np.random.binomial(1.0, 0.1, N) * x_0\n",
        "y = A @ x_0 + np.random.normal(0.0, 1e-2, M)"
      ],
      "metadata": {
        "id": "rteb8ElBPPup"
      },
      "execution_count": 65,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "\n",
        "t1 = time.time()\n",
        "lasso = linear_model.Lasso(alpha=1e-5, tol=1e-12, max_iter=1e6)\n",
        "lasso.fit(A, y)\n",
        "t2 = time.time()\n",
        "print(t2 - t1, \"sec\")"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "iul4rl97PUw7",
        "outputId": "3dc92e5f-c187-4f98-dbdf-a121897cbb84"
      },
      "execution_count": 4,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "74.54628086090088 sec\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "import torch\n",
        "import torch.nn as nn\n",
        "import torch.nn.functional as F\n",
        "import numpy as np\n",
        "import random\n",
        "import torch.optim as optim\n",
        "import matplotlib.pyplot as plt\n",
        "import os\n",
        "from torch.optim.optimizer import Optimizer, required\n",
        "## two layer neural network\n",
        "\n",
        "from scipy.linalg import svdvals\n",
        "\n",
        "import scipy as sp\n",
        "\n",
        "import sklearn.covariance as cov"
      ],
      "metadata": {
        "id": "mqCDZUlMPWah"
      },
      "execution_count": 5,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "## here is a spred linear perceptron\n",
        "class Net(nn.Module):\n",
        "    def __init__(self, dim=N):\n",
        "        super(Net, self).__init__()\n",
        "        self.weight = nn.Parameter(torch.zeros(dim).normal_(0, 1/2)) \n",
        "        self.weight2 = nn.Parameter(torch.zeros(dim).normal_(0, 1/2)) \n",
        "\n",
        "    def forward(self, x):\n",
        "\n",
        "        return self.weight * self.weight2\n",
        "\n",
        "model = Net().cuda()\n",
        "print(model(1))"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "r8hkhhaOPlZW",
        "outputId": "68060af7-e9c7-41d0-d0ea-9ae2eea3c908"
      },
      "execution_count": 6,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "tensor([-0.1329,  0.1539,  0.0716,  ..., -0.0188,  0.0804, -0.0102],\n",
            "       device='cuda:0', grad_fn=<MulBackward0>)\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "### let us train with Adam\n",
        "\n",
        "model = Net(dim=40000).cuda()\n",
        "opt = torch.optim.Adam(model.parameters(), lr=1e-2, weight_decay= 2 * 1e-5)\n",
        "losses = []\n",
        "\n",
        "Atorch = torch.tensor(A).cuda().float()\n",
        "ytorch = torch.tensor(y).cuda().float()\n",
        "\n",
        "t1 = time.time()\n",
        "for j in range(10000):\n",
        "  w = model(1)\n",
        "  loss = F.mse_loss(Atorch @ w, ytorch)\n",
        "  opt.zero_grad()\n",
        "  loss.backward()\n",
        "  opt.step()\n",
        "  losses.append(loss.item())\n",
        "t2 = time.time()\n",
        "print(t2-t1)"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "rdVHMlv2P3yi",
        "outputId": "387c4493-755f-4074-90c3-58ba1a967c76"
      },
      "execution_count": 66,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "7.456382751464844\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "## the model has roughly converged\n",
        "plt.plot(losses)\n",
        "plt.loglog()"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 287
        },
        "id": "5MWXkaFNQCSK",
        "outputId": "3591fc11-f9a6-444e-9809-b0f0a028e030"
      },
      "execution_count": 67,
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "[]"
            ]
          },
          "metadata": {},
          "execution_count": 67
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 432x288 with 1 Axes>"
            ],
            "image/png": "iVBORw0KGgoAAAANSUhEUgAAAXwAAAD8CAYAAAB0IB+mAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4yLjIsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy+WH4yJAAAgAElEQVR4nO3deZxbV5Un8N95WktLbarFdi2usst27Dix41RMFqAdskOcAGEgCcwAycQTIISGprvDANPTzTBpmqanSSfdIQxpAw3ZSAgJhCyEbJ12Jl6yeU3Ke+37IpV23fnj6WkrSfUkPUlPpfP9fPyJ65X0dOulfHR03rn3khACjDHGlj6p3ANgjDFWGhzwGWOsSnDAZ4yxKsEBnzHGqgQHfMYYqxIc8BljrEoYyz2AdIhoO4DtTqfzlrVr15Z7OIwxVlH27t07LoRoTj1Oeu7D7+3tFXv27Cn3MBhjrKIQ0V4hRG/qcS7pMMZYleCAzxhjVYIDPmOMVQkO+IwxViU44DPGWJXggM8YY1WCAz5jjFUJDviMMVYlOOAzxliVKFnAJ6JVRPQTIvpVqV6TMcZYnKqAT0T3E9EoEe1POX4lER0hoj4iuiPbOYQQx4QQNxcyWMYYY/lTu3jaTgB3A/iZcoCIDADuAXAZgH4Au4noCQAGAHemPP8mIcRowaNljDGWN1UBXwjxMhF1pRzeCqBPCHEMAIjoQQDXCiHuBHB1vgMioh0AdgBAZ2dnvqdhjDGWopAafhuA0wlf90ePpUVELiK6F8A5RPSNTI8TQtwnhOgVQvQ2Ny9Y3ZMxxlieSrYevhBiAsCtpXo9xhhjyQrJ8AcAdCR83R49VjAi2k5E983MzGhxOsYYYygs4O8GsIaIuonIDOB6AE9oMSghxJNCiB11dXVanI4xxhjUt2U+AGAXgHVE1E9ENwshQgBuA/AMgEMAHhZCHNBiUJzhM8aY9niLQ8YYW2J4i0PGGKtyugz4XNJhjDHt6TLg801bxhjTni4DPmOMMe3pMuBzSYcxxrSn6y6dDWdvFj9/8o95PbfLZYfLYdF4RIwxpn+ZunRKtrRCPo6OeXDdv+zK67n1NhN+d/sH0FZfo/GoGGOsMuk64He77Ljvpq05P28+EMLXH3kbX/7lPjz03y6AyaDLyhVjjJWULgM+EW0HsL2npwcfXJvfipmBsMDtD7yBHzz7Lu646gxtB8gYYxVIl6mvFm2Z12xagRu2duLel47ihSO89wpjjOky4Gvlr7ZvwBnLnPizh9/C8Iyv3MNhjLGyWtIB32oy4O4bt8AbCOP2B99AKBwp95AYY6xsdBnwtezD72lx4H99dCNePz6Ju55/T4PRMcZYZdJlwNd6aYXrzm3HJ85txz+90Id/f29ck3Myxlil0WXAL4a/ufZMrG524E8fehOjc1zPZ4xVn6oJ+DazEffcuAVufxBffehNhCP6nWHMGGPFUDUBHwDWLXPir685E6/2TeCfX+gr93AYY6ykqirgA8Aneztw7eYV+D9/eBevHZso93AYY6xkdD/Ttgjnxnc/dhbe7p/BVx58A0/d/oGKXmTt6f1D+LtnjiAUFjBIBIkAg0RotJvxvevOxkqXvdxDZIzphC4z/GJvgOKwGHH3jedgaj6Irz38FiIVWs//j75x3P7AmzAbJGzprMfGtjqcsawWq5ocODw8h8/e/zom3P5yD5MxphO6zPBL4cwVdfj21Rvw7cf3475XjuHWP1ld7iHl5MDgDHb8fC+6mmx4aMcFqLOZkr6/79QUbvzxa7hp5248sON82MxV+7+aMRalywy/VD7zvk58+Kxl+P4zR7D35GS5h6Pa6cl5fO5fd8NpNeKnN21dEOwBYEtnA/7phi14Z2AGX/rFPp5lzBir7oBPRPjb687GinorvvzLNzA9Hyj3kBY16Qngs/e/jkAogp/dtBXL6zKv93/ZhlZ856Mb8cKRMXzz1/uh581uGGPFV9UBHwBqrSbcc+MWjLn9+Pojb+s6KM4HQrhp524MTHvxk8/2Yk2rc9HnfPp9K3H7h3rw0J7TuP/VE8UfJGNMt6o+4APA2e31+MZV6/GHQyO6DYrBcARf+sU+vN0/jbtuOAe9XY2qn/vVy9biwtUu/OSVYzzhjLEqxgE/6vMXdeGyDa34298fwlunp8s9nAV+tuskXjgyhu98dCOuOHNZTs8lIvzn81dicMaHl98dK9IIGWN6xwE/iojw/U+cjRanFbc9sA8z3mC5h5TkkT2nsbmjHp9+38q8nn/J+lY0Ocx44PVTGo+MMVYpdBnwtVweORf1NjPuuuEcDE37cMej+qnnHxqaxeHhOXx8S1ve5zAbJXzi3A48f3gUo7O8eBxj1UiXAb/YE6+yOXdlA/78inX4/f5h/NtrJ0v++uk8/sYAjBLh6rNXFHSe68/rQDgi8Mjefo1GxhirJLoM+OV2ywdW4eJ1zfjObw9h/0BpP2WkCkcEHn9zANvWNaPRbi7oXF1NdlywyoUHd5+q2NnFjLH8ccBPQ5IIP/jkZjTazbjtl/vg9ofKNpZdRycwMuvHx85p1+R812/twOlJL149yhvBMFZtOOBn0GiX6/mnJufx3x97p2z1/F+/MQCnxYhL1rdocr4rzlyGBpsJD75+WpPzMcYqBwf8LLZ2N+Jrl63FE28N4qHdpQ+Q3kAYT+8fwofPWg6ryaDJOa0mAz6+pR3PHhzGOC+sxlhV4YC/iC9u68EH1jThr544gMPDsyV97WcPDsMTCONjBXTnpHPD1g4EwwK/eXNQ0/MyxvSNA/4iJInwD5/cjNoaE770i32YD5Sunv/rNwbQVl+DrTnMqlWjp8WJjsaailowjjFWOA74KjQ7Lfjhpzbj2LgH3378QElec2zOj1feG8e1m1dAkkjz829qr8dbp8vbgcQYKy0O+Cpd2NOEL39oDR7d149HS9DH/sRbgwhHBD52jrblHMWm9noMTHu5js9YFSlpwCeijxLRj4noISK6vJSvrYWvXLIGZ7fX4cevHCv6az3+xgA2ttWqWhEzH5s66gEAb/frb90gxlhxqA74RHQ/EY0S0f6U41cS0REi6iOiO7KdQwjxuBDiFgC3AvhUfkMuH4NE2NrViOPjnqJOXOobncM7AzOa9d6ns7GtFhIBb3JZh7GqkUuGvxPAlYkHiMgA4B4AVwHYAOAGItpARGcR0W9T/iQ2kn8r+ryK091shz8UwVAR16N5bN8ADBLhmk2FLaWQjc1sxNpWJ2f4jFUR1RudCiFeJqKulMNbAfQJIY4BABE9COBaIcSdAK5OPQcREYC/BfB7IcS+fAddTt1NdgDAiXEP2uoz7zaVr0hEbpd8f08Tmp0Wzc+f6Oz2Ojx3cARCCMj/a/IzPR+A2x+C3WxEQ4HLPzDGiqfQGn4bgMQZSf3RY5l8GcClAD5BRLemewAR7SCiPUS0Z2xMf2u3KwH/2LinKOd//cQkBqa9Ba2MqdamjnpMzQfRP+XN+xzjbj8+8L0X8P7vvYBzvvMcfvxy8e9vMMbyU9KbtkKIu4QQ5wohbhVC3JvhMfcJIXqFEL3Nzc2lHJ4qrU4rakwGHB8rTsD/9b4B2M0GXL4ht01O8rGpXb5x+2YBG778y4tH4QmE8NfXnInzVzXih8+/VxF7AzNWjQoN+AMAOhK+bo8eK0i51sNXQ5IIXU12nJjQPuD7gmE89c4Qrti4DDVmbZZSyGbdMifMRinvOv7wjA8/f+0krtvSjs9e2IW/vmYjPIFQSbqYGGO5KzTg7wawhoi6icgM4HoATxQ6qHKuh69Gd5MNx4tQ0nn+0Cjm/CF8vIjdOYlMBglnrqjNewLW3S+8ByEEbr9kDQD5DeTDZy3HzldPYNLDWT5jepNLW+YDAHYBWEdE/UR0sxAiBOA2AM8AOATgYSFEwVNR9ZzhA3Id/9TkPILhiKbnffzNAbTWWnDBapem581mU3s93hmYQSjHn2Xc7cdDu0/jk70d6Gi0xY5/5ZI18ATCePyNgj/oMcY0pjrgCyFuEEIsF0KYhBDtQoifRI8/JYRYK4RYLYT4rhaD0n+G70A4Igq62ZnOwcFZXLi6CYYiLKWQyaaOOniDYfSNuXN63u7jkwiGBa47N/nTyNpWJ9obarD7BK/Tw5je8NIKeehukjPa4+O5BclswhGB4VlfUVo9s1Fu3L6V443bvSenYDbKJaFU53U1YveJSd3sCcwYk+ky4Ou/pOMAABzTsFNndM6HcERgeb1Vs3Oq0eWyw2k14q3+3K71vlNTOKutDhbjwpvL53U1YtwdwImJea2GyRjTgC4Dvt5LOg02E+pqTJp26gxOy+WhFSXO8CWJoitnqs/w/aEw9g/M4tyVDWm/f16XfJzLOozpiy4Dvt4REbqb7Jp26gxOy0s1rKgrbcAHgDPbavHuyJzq9YH2D8wiEI5gS2d92u/3tDjQYDNh93EO+IzpiS4Dvt5LOoDcqaPl5Kt4hl/akg4gv8kEwwKTKidM7Ts5BQDY0pk+wycinLuyEXuij2OM6YMuA77eSzqAHPAHZ3zwBcOanG9w2gunxQin1aTJ+XLRWiuv2TOickG4faem0NFYg5bazG9O53U14Pi4B2NzvN4+Y3qhy4BfCWKLqGlUxx+c8ZW8fq9odsqBe1RFcBZCYO/JqYzZvWJjm/xm/e7IXOEDZIxpggN+npSAr1VZZ3DaW5ZyDhDP8EdVZPgD016Mzvkz3rBVdEYnY52a5E4dxvRClwG/Emr4XRqvmjk048PysmX4Skln8Qx/3ym5m2exDH9FfQ1MBsJJbs1kTDd0GfAroYbvsBjR4rTghAYB3xsIY9ITKPmkK4XFaECj3ayqhq/8vD0tjqyPM0iE9gYbTnOGz5hu6DLgV4oujVozh2bkDp3ldeUp6QBAi9OiqoY/NOOFy26G1bT4ap4djTacnCzOMtKMsdxxwC/AKo0CfqwHv0wZPgC01FpV1fDl0pO6N6aVjTac4pIOY7rBAb8A3U12THgCmPEGCzrPYDTDL8ekK0Wr06Kqhj807cOyWnXj7Gy0YdYX4g1RGNMJXQb8SrhpC8Rv3BZaxx+c9oIIaK0r7h622bTWWjHm9iO8yGzboRn13USdLu7UYUxPdBnwK+GmLSCXdAAUXNYZmvahyWFJuxBZqbTUWhCOiKwbl3j8Icz6Qlim8l7DymjA504dxvRBlwG/UnS6bCAqPOAPznjLWr8HgJbo5KtsnTpDM7mt99PRwBk+Y3rCAb8AFqMBbfU1BQf8gWkv2so06UoRm3w1ly3gy/ca1Gb4dosRTQ4L37hlTCc44Beo0FUzhRAYmvZheRlv2AJyDR/IPvlqKI8VPVe6uDWTMb3ggF+gVU12nBj35L270/R8EN5guOwlnSaHsrxCloAfLenkcnO5o6FG860gGWP50WXAr5QuHUDO8Of8IYy782s9jLdklrekYzZKcNnNGFmkpNPkMOd0c9nlsGS9EcwYKx1dBvxK6dIB4q2Z+ZZ19DDpSrHY5KuhmdxLT412M+YD4UWXkf77Z47gaw+/uWhbKGMsf7oM+JVkVXR/23w3NI8tq1Dmm7aAfOM2aw1/xqv6hq2iwWYGIJeuMukbdeOfX+zDY/sG8MPn38vp/Iwx9TjgF6itQV4V8vh4fp0oA9NemA0Smuzlm3SlkNfTyZ7h51p6arDJG7pkK+v84x/ehdVkwKXrW3HPC31w+0M5vQZjTB0O+AUySITORlv+Gf60D8vqrJAk0nhkuWuttWJsLv1sW7c/hDlfCMtyLOk02OUMfyrD8gr+UBhP7x/Gp87rwKff14lwROCdfv3fu2GsEnHA10B3k6OAGn75Nj5J1VJrRUQAE+6FZZ3hmfz23G1cJOAfG/MgFBE4p7MBmzrkTdHfPD2d02swxtThgK+BVc12nJiYRySPG46D096yLpqWqDXLRijKzeVlWfaxTac+WtKZylDSUbZAXNfqRKPdjJUuG948zZufM1YMHPA10OWyIxCKxFos1QqFIxiZ8+uiQwdAbFPydHX8oViGn2NJJ3rTdtKT/qbtkeE5GCWKbRm5uaOeM3zGikSXAb+S+vCBhP1tcyzrjEbr5XoJ+MryCuky/NikqxwzfJNBgtNqzFjSOTI8h1XNdpiN8q/i5o56jMz6Ve2+xRjLjS4DfiX14QNySQfIfZlkPbVkAvJsW6L0C6gpK3oqgTkXDTZz5oA/Mod1y2pjX69ulttceYVNxrSny4BfaVqcFtjMhpw3NB+I1sXLtZdtKpNBgsuevjVzaNaX983lBrs5bVum2x9C/5QX61rj++MqrzGUY3mMMbY4DvgaICJ0uXJfRG1ouvx72aZqrbWkXU9neMabczlH0WgzpZ141Tcqt7KuaXXGjikzeZWbxIwx7XDA10h3sz3nks7gtBdOqxFOq6lIo8pdi9OSdj2dOV8IdTX5jbPBlj7DH55Z+AnHbjGi1mrkDJ+xIuCAr5FVTXacnvIiEIqofs7gjE83LZmK1lpr2pu2vmAYVlN+vy4N9vQ1/AmP/Douhznp+Ir6Gs7wGSsCDvga6XLZEY4InJ5Sf7NRT5OuFC21Voy7/QiFk9+4/KEIrHluwZhpAbXJ6AqjyuQsxbI6K2f4jBUBB3yNdOfRqTM47cVyndywVbQ4LRACmEgowQgh4AuGYckzw1cmX6XW8Sc8ATgtxgXLLS+vq4m1gTLGtMMBXyO5bmjuDYQxNR/UTYeOwmk1ApA3LFeEIgIRgfwz/Njkq+SyzoQnsKCcA8h7A0x6AosuqcwYyw0HfI3U28yot5lUt2YO5rk2TbFZTXJQ9yYEWyXw5pvhZ1pAbdLjX1DOARD71MNZPmPa4oCvoe4mO46PqQv4yv6w5d7LNlVNNOAnZtf+6I1o5c0gV8ryCqkBf8IdgMuxcFloZQlmruMzpq2SBXwiWk9E9xLRr4joC6V63VLqbrLjxITKDD/ag6+3kk4sww/Eb9rGMvw8ZtkCQIM9/QJqE54AXGkyfOUTQaZNUybcfjy9fyjvfYQZq1aq/gUT0f1ENEpE+1OOX0lER4ioj4juyHYOIcQhIcStAD4J4KL8h6xf3S47hmZ8mA8svoHH4IwXRLmvTVNsxcjw62uUDD8ewCMRgckMNXyl33/GuzDg+0NhXPGPr+DWf9uHvSd5VU3GcqE2ZdsJ4MrEA0RkAHAPgKsAbABwAxFtIKKziOi3KX9aos+5BsDvADyl2U+gI/FOncVbMwenvWjOc22aYqoxy+NJW8PP86at2SjBKFHSm8isL4hwRKAxzU5ftdGAP5sm4P/x0CjGo+v1P7j7dF7jYaxaqYo2QoiXAUymHN4KoE8IcUwIEQDwIIBrhRDvCCGuTvkzGj3PE0KIqwB8OtNrEdEOItpDRHvGxsby+6nKRFk1U01ZZ2jGp7uWTCDTTVs5w8/3pi0gl4P8CZPSlLbPpjQZvt1sgEGitBn+waFZGCTCtZtX4On9w1zWYSwHhaSXbQASU6z+6LG0iGgbEd1FRD9ClgxfCHGfEKJXCNHb3NxcwPBKr8ulvjVzYNqLNp116ACZSjry3/NtywTkN5LEc05kmHQFyGsT1dWY0gb8Q0OzWNVkR+/KBrj9IQzzMsqMqWYs1QsJIV4E8GKpXq8c7BYjWmstOLZIp44QAoPTXly8rqVEI1MvftM2IeBrlOErnxQAuSUTAFwZNm+vtRox61t4L+TQ0BzOXdkQW0b52JhHd51OjOlVIRn+AICOhK/bo8cKVmkboCRS06kzPR+ELxjRzcYniayxDD8enLXK8JXzAMB4NMNPd9MWQNoMf9YXxMC0F2csd2JVLODnt3k8Y9WokIC/G8AaIuomIjOA6wE8ocWgKm0DlERqNjQfiLVk6q+kY5AIZqOkfQ3fZEjJ8OWAr/Top6qtMS24aat8clrT4kRrrQV2swFHVc57YIypb8t8AMAuAOuIqJ+IbhZChADcBuAZAIcAPCyEOKDFoCo7w7dh0hPATIYeciA+g1SvpYialHp7LMPPsy0TUG7aJtbw/ai1GjN2KaUL+ONzchmotdYCIkJ3sz3nTWcYq2aqavhCiBsyHH8KRWixFEI8CeDJ3t7eW7Q+d7F1N8mlhuMTHmy21ad9jDLpSo8lHQCwmqSkGr6SmVsLaCG1mqTYvQAAcPvDWfcBSFfSUdoxldm57fU2HOWSDmOq6asJfAmIb2ieORANznhhNkhpZ5nqQY3JkGEtnQK7dELJnxqylYhqrSbM+oJJbZdKK6dy3Zqc5qRVPRlj2eky4FdySaez0QaJkHVNncFpH5bXWyFJVMKRqZfaQhmbaVtAhm8xJmf4vmD29fXrakwIhkXSG8+42w+nxRgrLbnsFkzNBxas3c8YS0+XAb+Sb9qajRLaG2xZa8tD015d7WObqsa8MMM3SASjoZCSTm4ZfrrlFcbdyUsxNDnMEAKYTLObFmNsIV0G/Eq3WGumvNOVPuv3QLqbtpGCsnsgXYYfzprh19bIt5dmvfFe/Am3H00Jq2sqfx+f44DPmBq6DPiVXNIB4sskp5v2HwpHMDyrv71sE1nT1PALqd8r50zM8H3BSNY9ctNn+P6kDF+5eavsjcsYy06XAb+SSzqAHPA9gTDG5hYGotE5PyJCvx06QPSmbUDbDH/hfYFw1jbPujQLqE24AykZvhz8le4dxlh2ugz4la47y3aHSkvmch1OulJYUyZJ+YLZg7MayuJpyqceXzCSdX19pWVz1icH/HBEYHI+ecOUWIbv5pIOY2pwwC+CrAE/OulKbxufJKoxSwtq+IUu42w1GSAEEIh21Cz2JqKUe5QOoUlPAEIkr65ZazXCbJAwxhk+Y6roMuBXeg1/RX0NzAYpe4av5y6dNDV8LTJ8IB7AFw34xuRVO5VavlLqAeRVNRvspgU7aTHG0tNlwK/0Gr5BIqx02dIG/KFpL5xWY9ZZpuWm3LRVyi/+RcovalhSll32hyJZ2zItKRm+ck/BZk6eHO60muD2L77DGGNMpwF/KehusqcN+APTPl2Xc4B4+UUJtovdYFV1TiXDD8p1fH8oknUHLUtKhq9sG2k3Jz/HYTFiLs0yyoyxhTjgF0l3kx0nJ+YRjiS3Zg7N6HvSFRDfBEXpm1/sBqsaSobvD4UT9sjNfE6DRDAZKPbY+WiGX5MS8J1WI2f4jKmky4Bf6TV8QA74gXAkVrNX6H3SFRAPqt5Y+UW7DN8XjMSy9sXW17caDQkZfvqSjsNihJszfMZU0WXAr/QaPpC+U8cbCGNqPqj/gJ+yr+1ik6TUsCbU8GOrby7yJmIxxffB9URLOrY0JR3O8BlTR5cBfylIF/AHZ5RlkfVd0lGCu3Kj1B8KZ623q5HYpaOsi79YmciSkOHHb9qmBHwrZ/iMqcUBv0ianfKOTEkBP9aSqe8MP5aNh/ST4SslHbslpUvHYoQ7EEIksnAZC8ZYMg74RaLsyJQY8Iem9T/pCoiXdHwBuTXTp0WGn9BmGavhL/ImYjUa4E/o0iFa+KnAYTVCCGA+Yd4AYyw9DvhF1OVKDvgD014QAa21+i7pJN60DYYFhFg8OC8mcSJVPODnluHbTAYQJe8h4LDI8xm4rMPY4nQZ8JdClw4ArGqyo39qHoFo0Bqa8aLZYSl4mYJiS7xp64vV2wtfLROQy0NKEF+shp/cpROCzbJwR06HVT7m9mfeQ5gxJtNl5FkKXToA0N1sR0QApybnAcg7Xem9QweIB2dvIBzrxS80w4/ftC0gwzcvfLwz+ibAk68YW5wuA/5S0eVK7tQZnPHqvkMHSLxpG6+3a7EePhDtw1cx8QpQavjxgF+TZgzxDJ8DPmOL4YBfRIkbmgsh5ElXOu/QAeI1fF8grLr8sph0Gf5iZSKLSYqVlOYDoQUdOoDchw9wDZ8xNTjgF1G9zYxGuxnHx+cxPR+ELxjB8koo6USDszeHG6yLkSSC2SAl1/BzzPDTlXSUgD/HGT5ji+KAX2TyImpuDER78NsqoKRjNEgwGQjeoHYZPhDN2IPhWKulmhp+LMP3pw/4sU8j3JbJ2KI44BeZ0ppZKZOuFNboNodqg7MaFqMhuQ9/sbV0TAkZfjC0YB0d+ZzxVTgVwXAED75+CsPRzWYYYzIO+EW2qtmOkVk/+sbcAPS9l22iGpNBrrerXAZBDatJgj8601YiwGSgrI+3GOUMXwgBbyC8YKVM+THxVTgV3/3dIdzx2Du454W+gsfM2FLCAb/IlBu3u45OwGyU4LKbF3mGPtSYDSltmYVn+FZTPMO3GBdOokr3eCGAYFjA4w8vWAsfkN80JELSHry7T0wCAN4dmSt4zIwtJboM+Etl4hUQb83cfWISy+uskKTsQU4vlG0OlQxfm5JOtIYfUrc2jyXh5rE3GEZNmpIOEUVLRfI4hRA4NSHPezg4NBvbtYsxptOAv1QmXgFAV5MNgJyBVkJLpsJiMsAbjMQyfG1KOgb4om2Zat5AlN7/6Xl5z9p0N23lx8UnaM14g5jzh7Cq2Y45Xwj9U960z2GsGuky4C8lNrMxtsPV8gro0FHUmCT4Atq1ZQLym4Y/OvFKzfmU9tCpeXnZhEwBP3EJhpPR7P6CVS4AwOicv+BxM7ZUcMAvAaWso/dVMhPVRLNxLdsyEzN8NedTMnxPtMfeZEj/nMQMX1nG4pzOBgDAhDs54J+enMcje07zcsqsKi0sijLNdTfbsevYRMW0ZALRm7ZT8bXrtevSkSdeqVmqQcnwlWUTDBnufyifHIB4gD9jmVP+2hNIeuyHf/gK5vwhrKivwUU9Tfn9IIxVKM7wS2BVtFOnEtbRUVgTbtoaJYIxQ3adC4sxoYafQ4Y/H93e0Jgh4CufHID4Impd0Ws+mRDwpzyB2Izc/zg6nudPwVjl4oBfAr1djXBajThjWW25h6Ka1WSIzopVV29Xd87o0goqb9rGM3w5mKvJ8Gd9QVhNEhwWI5wWI8YTSjqHhmZjf993cjrvn4OxSsUlnRLY3FGPd/7nFeUeRk5qojNt5d2utMkLLNEdrHzBSF41fKOU/jlWkyH2mDlfCE6rvClKo8OMCXc8wz88LPflv7+nCScnPQtPxNgSxxk+S0u+aRtR3UKphrw2jryJuaoMP9qrHwv4GWbmyv39coYvB3w5j3HZzUklnZE5H8wGCdl87g8AABc2SURBVGe112Fo2ocw37hlVYYDPkurxmxAOCLg9oUWXdVSLavRgEAoAm8wrHLilZLhyyWdTDX8xIlXs75gPMO3m5Nu2k64A3A5zGhvqEEoIjAyy2vtsOpS0oBPRHYi2kNEV5fydVnulAx82hsseHtDRUejPAltZNafU4av3LTNWMNPaMuc84VQG83wHRZj7NMBAIy7/WhyWGLtscoKpoxVC1UBn4juJ6JRItqfcvxKIjpCRH1EdIeKU/0lgIfzGSgrLSXYTs8HCt7eUHHNphXojAZ9NTV8pe9+PqBk+Bn68I2GhJJOMFbSsVmMsTcLQAn4ZrQ45W6p1B59xpY6tf+SdwK4MvEAERkA3APgKgAbANxARBuI6Cwi+m3KnxYiugzAQQCjGo6fFUlNbFmDoGY3bc1GCV+/Yh0AdTN3TdEAr8yizdqlk9CWWRst6TgsxqStD8fnAnA5LGiMLmCX2qPP2FKnqktHCPEyEXWlHN4KoE8IcQwAiOhBANcKIe4EsKBkQ0TbANghvzl4iegpIUQk9XFMHxIDvlY3bQHg6rOW48jwLK7cuGzRxyo3aZU9cDPdtFVW4QSSb9razHLmH44ISARMeOSSToNdfkOY4oDPqkwhbZltAE4nfN0P4H2ZHiyE+CYAENHnAIxnCvZEtAPADgDo7OwsYHisENboujWBsLoWSrUkifDnV5yh6rGxgK8iww+EIrEbws6EDB8APIEQDEQIhgXqbSZYjAY4LEZMeoKF/jhFMTzjw1v90xib88NhMcJslOANhDEfDGNdqxObO+ph1vD/CaseJe/DF0LsXOT79wG4DwB6e3u5b65MEnej0jLDz4VS0lF23crYpRO9xzDhkWvy8Qxf/u+8PwzlqcqbQIPdhKl5/WT4wXAET741iF/8v1PYe3Iq62NtZgMuWOXCxWe04NL1rVhWVzkzuFl5FRLwBwB0JHzdHj1WMCLaDmB7T0+PFqdjeUjcXWqxrQiLRZKSNzfJlOEr45vyJK+qabfI/3X7QwsCfqMtuUe/nHYdncAdj72NkxPz6Glx4OuXr8X71zRjWa0V84EQ/KEIakwGmI0S3u6fwat943jp3TE8f3gU33p8P85ur8NVG5fj3JUNWFZrRafLVu4fielUIQF/N4A1RNQNOdBfD+BGLQYlhHgSwJO9vb23aHE+lruahKxeqz78fBgN8Y3MM3bpRMc355MDvtJGalcy/IROHXsswy9/wPf4Q/i7pw/jp7tOYqXLhv/7X3pxyfqWrDuBraivwZUbl0EIgb5RN549OIJnDwzje08fjj1mXasTW1Y24Oz2OpzX1YjVzfZFdxdj1UFVwCeiBwBsA9BERP0A/koI8RMiug3AMwAMAO4XQhzQYlCc4ZdfYsAvV0kHAEwSxWr4mWfayuObjS6cptS3bQkZvkLJ8J1WU2zt/HJ4/fgk/uyRN3F60ovPXdiFv7hyXdpN2jMhIqxpdWJNqxNfurgHpyfncWLCg75RN545MIzf7x/CA6+fAgA0Oy24cLULF61uwp+sa0ZrLZeAqpXaLp0bMhx/CsBTmo4InOHrgdUcz6a1vGmbK4NEsZJO5tUykzN8c7R/Xwnu8/4wlJtB8YBvjD2+lCIRgX956Sh+8OwRdDTa8NCO8/G+6GYthehotKGj0YYPrGnG5y/qhhACx8c9eP34JHYdm8CrfRP4zZuDAIANy2uxbV0zLj6jBed01GuyEiqrDLx4GkvLqpcM3yAtuh6+UurxRCdoxTJ8c7xLJxLd21ap6zstxtgnglKJRATueOxtPLynH9s3rcCdHz8r9gakNSLCqmYHVjU7cP3WTgghcHh4Di8eGcOLR0bxo5eP4Z9fPIpaqxEfXNuMyza0Ytu6FtTVmIoyHqYPugz4XNIpv6QafhkzfKOBYj32mWr4SuY/708u6cTaMv1hhKMBPzHDD0QXctNq6YhswhGBOx59G4/s7ceXP9SDr122tqR1dSLC+uW1WL+8Fl/Ythoz3iBe7RvHi0dG8cfDY/jt20MwSoQLVrtw+YZWXLZhGXf/LEG6DPhc0ik/k0GCUSKEIkLV7lTFkhjkM2b40dr+ggzfEl9eORbwrfEaPgB5cThHcX++cETgLx99G7/a24+vXLIGX71sbVFfT426GhM+fNZyfPis5YhEBN44PY1nDw7juQMj+PZvDuDbvzmATe11uPzMZbh8Qyt6Whx843cJ0GXAZ/pQYzJgzh9StTtVsZgSbtRmquErbwreaDeOUsNX2jX9IXkNfonin1yUXn23PwSXw1KcwQOYmQ/iLx59C88cGMGfXroGf3pp+YN9KkkinLuyAeeubMA3rlof7f4ZxrMHRvD9Z47g+88cQZfLFgv+53Q2ZHzzZfqmy4DPJR19sJrlgF/WDD/hhqIhQ5dOaoav3MQ1GQhEgD8Ugdsfgt1sjGWpSmlnroh1/L0nJ/HlX76B0Tk/vvWR9fivH1hVtNfSUk+LAz0tPfjith6MzPrw3MERPHtwBP/66nHc9/IxNDnMuHR9Ky5d34otKxtiaxMx/dNlwOeSjj4ogbOcGX5iVp85w0+p4RvkNygiii6sFoE3EI6VeIB4SWe2SJ06zx4Yxhd/sQ8r6mvw6BcuxKaO+qK8TrG11lrxmfNX4jPnr8SsL4iXjozh2YMj+O3bQ3hwt7yyyoo6K85sq8MZy+Q20bWtDqxqcvDyDzqky4DP9EEpf5QzwzcZFq/hK8dTa/hAwraKKbtsKSWdYmT4J8Y9uP3BN7CxrQ4/vWnrkul8qbWasH3TCmzftAKBUAR7T05h/8AM9g/OYP/ADP54eDS2i5hBInS5bFjbGn8TWNvqRHeTPen/KSstDvgsIyXglzXDT6rhpx+HEkC8aQO+nOH7U/bRVWbcJs7CTSSEwN8/ewTtDTbcsFX9In5CCHzz8XdgkiTc+5lzl0ywT2U2SrhgtQsXrI7PIfCHwjg25sG7I3PRP24cGprF0weGEb1nDpOB0N1kx5pWJ9obatBkt8DlMMPlsMBlN8PlMKPRbi5J55Qe9Y3O4bVjk/jM+SuLcn5dBnyu4euDVQ8ZfkKQz3SfUMnwleCdFPCju2Gltl/aYx084bTn3HNyCve8cBQAcOFqF1a67KrG+/ibA3i1bwLf+ejGqmtrtBgNsdbPRL5gGH2jbrw3Kr8JvDcyh3f6Z/DcgREEwulXSHdajKi3m9BgM6PeZkaDzQSn1Qi7xQi7WfmvQf6vxQCryYAakwE1ZgMa7Wa47BZd3Vg+PTkPImD/wCw8/hAGp734wXPvZnz8W6en8f3/tEnzcegy4HMNXx+UBdS02vEqH8o/WvkGbPp/wEonj7IzltmQUtIJheEPRZJ+jnTr7CR6+d2x2N//eHgUn7+oO+3jguEIjJI8thlvEN/93SFs7qjHp3P4VLDUWU0GbGyrw8a2uqTjQgi4/SFMuAOY8Pgx7g5g0hPAhFv++/R8AFPzQUzPB3Bi3AO3PwSPPxSbl5GNQSK0OC1orbWitdaCZbVW1NnMMEoEg0SQiGCQAImUvxMkiWBIOF5vM6O7yYb2BpvqyYf9U/OY8QYx6w3hJ/9+HH84NJLXNXtkbz++d93ZkDR+09JlwGf6oLQ1lmu1TCBe0smWrRmk+FaIRMmtnBajBH8wAl8wnLRWjVKucmfI8PcPzGBdqxOeQAi7T0ymDfjPHRzB7Q+8gQabCZ+7qAuv9k1g0hPAzs9v1fwf6lJERHBaTXBaTehqUvcJCpDfZOcDYXiibwDzgTC8weifQBgTbj+GZ30YnvFjdM6HY2Me7Do6UdDMaqfFCKfVGB2vEbU1JlhN8u/W84eLs4nfiQkPVjU7ND0nB3yWkZLhl3O1TKU+n6l+L38vXtIxG6SkTwKxGn4oggZbQnlIItjNhlhnT6r3Rt3Y0tmAQCiCw8NzC77v8YfwjcfeRltDDRrtZvzvpw7DIBH+5tqNCzJZpi2TQUJdjZTz/REhBMIRgbAQiESAcPTrSOyYSDgGTEY/WfRPzWPSE8SsL4g5XxBzvhBG53zwBSMwFHEy2tMHhvHFbdqWtTngs4ysJh1k+NLiGX6sD98fXtAKmFjSSX3jslmMsc6eRKFwBEMzPnQ22iAR8OzBYfiCyV0+r7w3jnF3AHddfw4uWO3C4eE5OCxGdDTyWvR6RUQwGkh10Ot02bBZRTutEAKegPzp4pX3xnBoaBbPHBjBqcnCVmP9/IXpy4iF0GXA55u2+hBvy9RDhp8l4CszbYNhNFmSJwFZTBI8nhD8ofCCNy672QBPmgx/aMaHcESgo7EGdosREQEcHXPjzBXxzP3l98bgsBhxXndjbJ0aVp2ICA6LEQ6LER/f0g4A+OZHNkAIgdePT6LTZcOPXz6OkVkffvfOkOrzJm5CpBVdBny+aasPNdElksvZIqemhp/4ZmA2pGb4Sg0/TYZvNqa9aTsw7QUAtNXb0FIrL7vw3kg84Ash8NKRMVy42sU95SwjIootff0/tm8AAPxjOAKJCIFQBC8cGUVECEzPB9FWX4PXjk/gRy8dAwD88paM24MXRJcBn+nDdVvasaK+pqztbUr2ni3DT1xyIWNJJ7hwVUyHxZi2LXPcLe+N2+y0oLvJDqNEODISr+MfG/dgYNqLL2xbnfsPxKqakiDUmA348FnLk7538Rkt+Oqla3FkeK5oM7M54LOMlPXUy0npuMm0jg6Q3Ku/MODHb9qmLvNssxjSbnM44ZaPuRxmmAwSupvs6Bt1x76vtGz+ydrmHH8axrKzmgxFXYaDP48yXVNKOtm6dBI/gaRm8RaTBF9QuWmbWsM3pq3hT7j9kAhosMn3A9a0OpIC/kvvjmFVk51v0LKKwwGf6ZoS6FXX8NOUdJQds1IzfLvFEJuslWjMHUCj3Rx7zZ4WJ05OeOALhuELhvHasQl8kLN7VoF0WdLhLh2mUIJ5thq+JBEkAiIi/U3bYFjE/p7IliXDd9nja+SvaXHEOnUmPQH4ghF8cG1T3j8TY+WiywxfCPGkEGJHXR1PYKl2ynr4xiw1/MTHpcvwY39PLelYDPAEwhDKyl5RE54AXI54e6cykeqd/hk8d3AEFqOE8zXYeJyxUtNlwGdMEbtpm6WGD8Q/ASwI+AmtmKmrftrMRoQjYsHaLONuP5oSdsHqctnQaDdj17EJPPXOMD50RkvSMg2MVQr+rWW6pqYtM/H7qWUbS9LKmQvbMgF5DZ7EWbQT7uQMn4hw8boWPLqvHwBw7ea2XH8MxnSBM3yma2omXsmPi+7OlRLUk0o6CzL8+CbnCl8wDLc/lJThA8AXtq3Gslorrtoo7+vKWCXiDJ/pmsmw+E1bIP6GkLqUsyVlM5REyiYonoTZtsqkqyZH8hINPS0OvPbfL8ll6IzpDmf4TNfUtGUCgEkJ+Gn68BWp2X8s4CfMto1NurInZ/iMLQUc8Jmuqc7wo49LXS8nW0nHnqakM+GRM3xXSobP2FKgy4BPRNuJ6L6ZmZlyD4WVmVKbX6xLR7Egw0/Z0DyRLc2uV+NzcoafWsNnbCnQZcDnPnymUDPxCgDC0clVC2/aJu9vmyjdvrbjnOGzJYxv2jJdi62ls8jEq2BECfipffjxN4CMNfxACHc9/x5C4QhmvEE4LUbus2dLEv9WM11T24cfjgb81F77rF060aA+MO2NrUPe2WhDW0NNYYNmTKd0WdJhTKF2pm0oLM+WzTrxKuV7VpMEiYD/6JuIHTs1OY8V9Rzw2dLEAZ/pmtoMPxTJUMM3JXbpJH+PiGA3G3FoaDb6XPm12jjgsyWKAz7TNaOKDVCALAE/mtVLFP+0kMhmMSAUEbCapNh+pJefyTNp2dLENXyma2o2MQfiJZ3UBdKUgG8xGkC08BxyHd+PzkYbvvWR9bh4XQve38NLH7OliQM+0zUl0C820zaa4GdcSye1JVNhi7ZmdjbaYDMbcRmvk8OWMC7pMF0zqszwFakB32QgEC28YatQ2i95u0JWDTjgM11T26WjSA3sRASLUVrwRqDwB+VJV50c8FkVKFnAJ6JtRPQKEd1LRNtK9bqsshlUzrRVpAvsFqMhY4Z/zeY22M0GnLuyIf9BMlYhVAV8IrqfiEaJaH/K8SuJ6AgR9RHRHYucRgBwA7AC6M9vuKzamGJr6agN+At/pS1GaUFLpuLm93fjwN9cibPb6/MfJGMVQu1N250A7gbwM+UAERkA3APgMsgBfDcRPQHAAODOlOffBOAVIcRLRNQK4B8AfLqwobNqoGT26Voq00mb4ZukjBk+Y9VEVcAXQrxMRF0ph7cC6BNCHAMAInoQwLVCiDsBXJ3ldFMAMi5FSEQ7AOwAgM7OTjXDY0uYKcfVMtMFdovRkLGGz1g1KSTtaQNwOuHr/uixtIjo40T0IwA/h/xpIS0hxH1CiF4hRG9zc3MBw2NLgVHleviKdL32TQ4zr37JGErYhy+EeAzAY6V6PbY0qN3xqqfFgb5Rd9rv3X3jFtVvGIwtZYUE/AEAHQlft0ePFYyItgPY3tPTo8XpWAVrsJlw6frWRbtoHv3ChZiI7kebijczYUxGQgh1D5Rr+L8VQmyMfm0E8C6ASyAH+t0AbhRCHNBqcL29vWLPnj1anY4xxqoCEe0VQvSmHlfblvkAgF0A1hFRPxHdLIQIAbgNwDMADgF4WKtgz1scMsaY9lRn+OXAGT5jjOWuoAyfMcZY5dNlwOeSDmOMaU+XAV8I8aQQYkddXV25h8IYY0uGLgM+Y4wx7eky4HNJhzHGtKfLgM8lHcYY056u2zKJaAzAyeiXdQBSU/7UY4lfNwEYL9LQ0o1Fq+dke1ym76m5NumO6fl6qX2eVtcr3XG+Xtm/V+3XK9v3y329VgohFi5GJoSoiD8A7lvsWOLXAPaUcixaPSfb4zJ9T821qbTrpfZ5Wl2vxa5PNV+vTN+r9uuV7ft6vV66LOlk8KSKY+keUwz5vI7a52R7XKbvqbk26Y7p+XqpfZ5W1yvdcb5e2b9X7dcr2/d1eb10XdIpBBHtEWlmmrH0+Hrlhq9Xbvh65aZY16uSMvxc3VfuAVQYvl654euVG75euSnK9VqyGT5jjLFkSznDZ4wxloADPmOMVQkO+IwxViWqJuATkZ2IfkpEPyaiT5d7PHpHRKuI6CdE9Ktyj6USENFHo79bDxHR5eUej94R0XoiupeIfkVEXyj3eCpBNIbtIaKr8z1HRQd8IrqfiEaJaH/K8SuJ6AgR9RHRHdHDHwfwKyHELQCuKflgdSCX6yWEOCaEuLk8I9WHHK/X49HfrVsBfKoc4y23HK/XISHErQA+CeCicoy33HKMXwDwlwAeLuQ1KzrgA9gJ4MrEA0RkAHAPgKsAbABwAxFtgLzJ+unow8IlHKOe7IT668Xyu17fin6/Gu1EDteLiK4B8DsAT5V2mLqxEyqvFxFdBuAggNFCXrCiA74Q4mUAkymHtwLoi2aoAQAPArgWQD/koA9U+M+drxyvV9XL5XqR7HsAfi+E2FfqsepBrr9fQognhBBXAajKEmuO12sbgPMB3AjgFiLKK4YZ8x+ubrUhnskDcqB/H4C7ANxNRB9B6aZ8V4K014uIXAC+C+AcIvqGEOLOsoxOfzL9fn0ZwKUA6oioRwhxbzkGp0OZfr+2QS6zWlC9GX46aa+XEOI2ACCizwEYF0JE8jn5Ugz4aQkhPAA+X+5xVAohxATkejRTQQhxF+SkgqkghHgRwItlHkbFEULsLOT5S7G0MQCgI+Hr9ugxlh5fr9zw9coNX6/cFPV6LcWAvxvAGiLqJiIzgOsBPFHmMekZX6/c8PXKDV+v3BT1elV0wCeiBwDsArCOiPqJ6GYhRAjAbQCeAXAIwMNCiAPlHKde8PXKDV+v3PD1yk05rhcvnsYYY1WiojN8xhhj6nHAZ4yxKsEBnzHGqgQHfMYYqxIc8BljrEpwwGeMsSrBAZ8xxqoEB3zGGKsSHPAZY6xK/H8IUM7YFaY+LwAAAABJRU5ErkJggg==\n"
          },
          "metadata": {
            "needs_background": "light"
          }
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "print((w.abs().detach().cpu() < 1e-5).float().mean())\n",
        "print((w.abs().detach().cpu() < 1e-6).float().mean())\n",
        "print((w.abs().detach().cpu() < 1e-10).float().mean())\n",
        "print((w.abs().detach().cpu() == 0).float().mean())"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "b7JLDaIJQYVF",
        "outputId": "50c2df5a-d18b-43a8-8a02-0755ae77a8fb"
      },
      "execution_count": 68,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "tensor(0.9976)\n",
            "tensor(0.9976)\n",
            "tensor(0.9975)\n",
            "tensor(0.9972)\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "## the agreement with lasso is very good\n",
        "torch.tensor((lasso.coef_  == 0)).float().mean()"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "PjvBg18ZQiJ8",
        "outputId": "2385b040-2ab9-436b-fbcf-6ae2ab7bbc27"
      },
      "execution_count": 48,
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "tensor(0.9977)"
            ]
          },
          "metadata": {},
          "execution_count": 48
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "## ok now lets try a much harder problem\n",
        "\n",
        "N = 400000\n",
        "M = 100\n",
        "\n",
        "## this is a compressive sensing task with sparse feature\n",
        "A = np.random.normal(0.0, 1.0/ N**0.5, (M, N))\n",
        "x_0 = np.random.normal(0.0, 1.0, N)\n",
        "x_0 = np.random.binomial(1.0, 0.1, N) * x_0\n",
        "y = A @ x_0 + np.random.normal(0.0, 1e-2, M)"
      ],
      "metadata": {
        "id": "DKfuqfu7Rkr-"
      },
      "execution_count": 72,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "### let us train with Adam\n",
        "\n",
        "model = Net(dim=N).cuda()\n",
        "opt = torch.optim.Adam(model.parameters(), lr=1e-2, weight_decay= 2 * 1e-5)\n",
        "losses = []\n",
        "\n",
        "Atorch = torch.tensor(A).cuda().float()\n",
        "ytorch = torch.tensor(y).cuda().float()\n",
        "\n",
        "t1 = time.time()\n",
        "for j in range(10000):\n",
        "  w = model(1)\n",
        "  loss = F.mse_loss(Atorch @ w, ytorch)\n",
        "  opt.zero_grad()\n",
        "  loss.backward()\n",
        "  opt.step()\n",
        "  losses.append(loss.item())\n",
        "t2 = time.time()\n",
        "print(t2-t1)"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "EC1HaZioS18W",
        "outputId": "b06c7d32-420f-4b21-8413-7b3911108ca0"
      },
      "execution_count": 73,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "10.790321826934814\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "## the model has roughly converged\n",
        "plt.plot(losses)\n",
        "plt.loglog()"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 287
        },
        "id": "rDlqbbY2TSC-",
        "outputId": "9d9bb3ac-680d-4bd3-d629-3b94be65fd2f"
      },
      "execution_count": 74,
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "[]"
            ]
          },
          "metadata": {},
          "execution_count": 74
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 432x288 with 1 Axes>"
            ],
            "image/png": "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\n"
          },
          "metadata": {
            "needs_background": "light"
          }
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "print((w.abs().detach().cpu() < 1e-5).float().mean())\n",
        "print((w.abs().detach().cpu() < 1e-6).float().mean())\n",
        "print((w.abs().detach().cpu() < 1e-10).float().mean())\n",
        "print((w.abs().detach().cpu() == 0).float().mean())"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "yPruchLFS5-m",
        "outputId": "b161bc09-6a0c-41db-b1a7-b1edc46fd471"
      },
      "execution_count": 75,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "tensor(0.9998)\n",
            "tensor(0.9998)\n",
            "tensor(0.9998)\n",
            "tensor(0.9997)\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "## now try lasso if you want to\n",
        "\n",
        "t1 = time.time()\n",
        "lasso = linear_model.Lasso(alpha=1e-5, tol=1e-12, max_iter=1e6)\n",
        "lasso.fit(A, y)\n",
        "t2 = time.time()\n",
        "print(t2 - t1, \"sec\")"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "b6VPexfcS8Y2",
        "outputId": "548eded1-aeeb-4506-ea05-bf1f191c3629"
      },
      "execution_count": 76,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "212.97227215766907 sec\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "## agreement\n",
        "torch.tensor((lasso.coef_  == 0)).float().mean()"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "n7tLeXVlTfBu",
        "outputId": "34082b00-88d5-423c-d903-e2bc5f1e85ce"
      },
      "execution_count": 77,
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "tensor(0.9998)"
            ]
          },
          "metadata": {},
          "execution_count": 77
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [],
      "metadata": {
        "id": "p6tpcQk9UX09"
      },
      "execution_count": null,
      "outputs": []
    }
  ]
}