{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "import functools\n",
    "import itertools\n",
    "import re\n",
    "import sys\n",
    "import time\n",
    "\n",
    "from matplotlib.pyplot import *\n",
    "from tqdm import tqdm_notebook as tqdm\n",
    "\n",
    "import jax\n",
    "\n",
    "from jax import lax\n",
    "import jax.numpy as jnp\n",
    "import jax.scipy as jsp\n",
    "from jax import random\n",
    "\n",
    "import numpy as np\n",
    "import scipy as sp"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "np.random.seed(10009)\n",
    "\n",
    "num_features = 1\n",
    "num_points = 100\n",
    "\n",
    "y = np.random.randn(num_points, num_features).astype(jnp.float32)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "@jax.jit\n",
    "def llh(y, x_loc, x_scale):\n",
    "    result = jnp.sum(jsp.stats.norm.logpdf(y, loc=x_loc, scale=x_scale))\n",
    "    return result\n",
    "\n",
    "batched_llh = jax.jit(jax.vmap(llh,(1,None,None)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "def llh_avg(y, x_loc, x_log_scale):\n",
    "    x_scale = jnp.exp(x_log_scale)\n",
    "    return jnp.mean(batched_llh(y, x_loc, x_scale),0)\n",
    " \n",
    "llh_avg = jax.jit(llh_avg)\n",
    "llh_avg_val_and_grad = jax.jit(jax.value_and_grad(llh_avg, argnums=(1, 2)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/disk_c/akash/anaconda/envs/iclr/lib/python3.7/site-packages/ipykernel_launcher.py:6: TqdmDeprecationWarning: This function will be removed in tqdm==5.0.0\n",
      "Please use `tqdm.notebook.tqdm` instead of `tqdm.tqdm_notebook`\n",
      "  \n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "cafe9fcdca634e1aac4ae135dd9ddcff",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "HBox(children=(FloatProgress(value=0.0, max=1000.0), HTML(value='')))"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0\t-1338.639892578125\n",
      "10\t-1585.385986328125\n",
      "20\t-585.3890380859375\n",
      "30\t-137.86053466796875\n",
      "40\t-137.54461669921875\n",
      "50\t-137.39547729492188\n",
      "60\t-137.30780029296875\n",
      "70\t-137.24981689453125\n",
      "80\t-137.20855712890625\n",
      "90\t-137.17767333984375\n",
      "100\t-137.1536102294922\n",
      "110\t-137.13433837890625\n",
      "120\t-137.1185760498047\n",
      "130\t-137.1054229736328\n",
      "140\t-137.09426879882812\n",
      "150\t-137.08468627929688\n",
      "160\t-137.07638549804688\n",
      "170\t-137.06912231445312\n",
      "180\t-137.06268310546875\n",
      "190\t-137.0569610595703\n",
      "200\t-137.0518341064453\n",
      "210\t-137.04721069335938\n",
      "220\t-137.04302978515625\n",
      "230\t-137.03921508789062\n",
      "240\t-137.03573608398438\n",
      "250\t-137.0325469970703\n",
      "260\t-137.02960205078125\n",
      "270\t-137.02688598632812\n",
      "280\t-137.0243682861328\n",
      "290\t-137.02203369140625\n",
      "300\t-137.01983642578125\n",
      "310\t-137.01780700683594\n",
      "320\t-137.01589965820312\n",
      "330\t-137.0141143798828\n",
      "340\t-137.01242065429688\n",
      "350\t-137.01083374023438\n",
      "360\t-137.0093536376953\n",
      "370\t-137.0079345703125\n",
      "380\t-137.006591796875\n",
      "390\t-137.00534057617188\n",
      "400\t-137.00413513183594\n",
      "410\t-137.00299072265625\n",
      "420\t-137.00189208984375\n",
      "430\t-137.0008544921875\n",
      "440\t-136.9998779296875\n",
      "450\t-136.99893188476562\n",
      "460\t-136.99803161621094\n",
      "470\t-136.99716186523438\n",
      "480\t-136.996337890625\n",
      "490\t-136.99554443359375\n",
      "500\t-136.99476623535156\n",
      "510\t-136.99404907226562\n",
      "520\t-136.99334716796875\n",
      "530\t-136.99267578125\n",
      "540\t-136.99200439453125\n",
      "550\t-136.99139404296875\n",
      "560\t-136.99078369140625\n",
      "570\t-136.9901885986328\n",
      "580\t-136.9896240234375\n",
      "590\t-136.98910522460938\n",
      "600\t-136.9885711669922\n",
      "610\t-136.98805236816406\n",
      "620\t-136.987548828125\n",
      "630\t-136.98707580566406\n",
      "640\t-136.98663330078125\n",
      "650\t-136.98619079589844\n",
      "660\t-136.98573303222656\n",
      "670\t-136.98532104492188\n",
      "680\t-136.9849090576172\n",
      "690\t-136.98452758789062\n",
      "700\t-136.984130859375\n",
      "710\t-136.9837646484375\n",
      "720\t-136.9833984375\n",
      "730\t-136.9830322265625\n",
      "740\t-136.98269653320312\n",
      "750\t-136.98236083984375\n",
      "760\t-136.98204040527344\n",
      "770\t-136.9817352294922\n",
      "780\t-136.98141479492188\n",
      "790\t-136.98110961914062\n",
      "800\t-136.98081970214844\n",
      "810\t-136.98052978515625\n",
      "820\t-136.98025512695312\n",
      "830\t-136.97998046875\n",
      "840\t-136.97972106933594\n",
      "850\t-136.97946166992188\n",
      "860\t-136.97921752929688\n",
      "870\t-136.97894287109375\n",
      "880\t-136.9787139892578\n",
      "890\t-136.97848510742188\n",
      "900\t-136.97825622558594\n",
      "910\t-136.97802734375\n",
      "920\t-136.97781372070312\n",
      "930\t-136.97760009765625\n",
      "940\t-136.9773712158203\n",
      "950\t-136.9771728515625\n",
      "960\t-136.97695922851562\n",
      "970\t-136.9767608642578\n",
      "980\t-136.9765625\n",
      "990\t-136.97637939453125\n",
      "\n"
     ]
    }
   ],
   "source": [
    "x_loc = 5.\n",
    "x_log_scale = 0.\n",
    "\n",
    "step_size = 0.01\n",
    "\n",
    "for i in tqdm(range(1000)):\n",
    "\n",
    "    llh_val, (x_loc_grad, x_log_scale_grad) = llh_avg_val_and_grad(y, x_loc, x_log_scale)\n",
    "    x_loc += step_size * x_loc_grad\n",
    "    x_log_scale += step_size * x_log_scale_grad\n",
    "    if i % 10 == 0:\n",
    "        print('{}\\t{}'.format(i, llh_val))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(DeviceArray(0.098069, dtype=float32), DeviceArray(-0.06291761, dtype=float32))"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_loc, x_log_scale"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x7f0a300423d0>"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAbgAAAGpCAYAAAD/QCONAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjMuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/d3fzzAAAACXBIWXMAAAsTAAALEwEAmpwYAABJqUlEQVR4nO3dd3hUVeLG8e9JQhEBQQTFBWk/BakRoiYiEEoKnYA0g4KKChZUFKW4irLKiivSVQQFFNClCNJLIJDEoIIVARUhCGKhKIZm2vn9kbIICSRkJndm8n6eh2cy7d53JiRvzr137jHWWkRERHyNn9MBRERE3EEFJyIiPkkFJyIiPkkFJyIiPkkFJyIiPinA6QAFccUVV9iaNWs6HUNERDzItm3bDltrK599u1cVXM2aNdm6davTMURExIMYY/bldrs2UYqIiE9SwYmIiE9SwYmIiE/yqn1wuUlNTeXAgQOcPn3a6SgixU7p0qWpVq0aJUqUcDqKyDm8vuAOHDhAuXLlqFmzJsYYp+OIFBvWWo4cOcKBAweoVauW03FEzuH1myhPnz5NpUqVVG4iRcwYQ6VKlbT1RDyW1xccoHITcYh+9sST+UTBiYiInE0F5yIffPABxhh27drlWIaDBw9y2223uWRZS5YsYceOHQV+XtmyZXO93d/fn8DAQBo2bEjPnj05efJkgZablJTEvHnzCpwH4JZbbrmo52WrWbMmLVq0+Ntt2a9FRDyXCs5F5s+fz6233sp7773nsmWmpaUV6PFXX301CxcudMm6L7bg8nLJJZfwxRdfsH37dkqWLMnrr79eoOdfTMGlp6cD8NFHHxX4OWdLTk5m//79AOzcubNAOUTEGcWy4Lbt+52pG3ezbd/vLlne8ePHSUhIYObMmX8ruNjYWFq2bElUVBT169dn0KBBZGRkAJkjnccff5ymTZvStm1bDh06BEBoaCgjR46kVatWTJw4kZiYGG644QYaNWrE3XffzV9//cWnn35K48aNOX36NCdOnKBBgwZs376dpKSknFHFrFmz6NatG507d6ZWrVpMmTKF8ePHc8MNNxAcHMzRo0cBePPNN7nxxhtp0qQJPXr04OTJk3z00Ud8+OGHDBs2jMDAQH744Qd++OEHIiMjadasGS1atMgZqe7du5eQkBBuvPFG/vnPf+br/WrRogW7d+/m6NGjdOvWjcaNGxMcHMxXX30FwKZNmwgMDCQwMJAbbriB5ORkhg8fTlxcHIGBgbz66qukp6czbNgwbrzxRho3bswbb7yR8563bt2a22+/nUaNGuW815B51N+wYcNo2LAhjRo14v3338/zOWfr1atXzuPnz59P3759c+7LK8vx48dp27YtTZs2pVGjRixduhTILOvrr7+ee++9lwYNGhAeHs6pU6cAmDRpEvXr16dx48b06dMnX++niOTBWus1/5o1a2bPtmPHjnNuO5+tSUdt3adX2lrDl9u6T6+0W5OOFuj5uXnnnXfs3Xffba21NiQkxG7bts1aa+3GjRttqVKl7A8//GDT0tJsu3bt7IIFC6y11gL23XfftdZa+9xzz9kHH3zQWmttq1at7ODBg6211p46dcpWq1bNfvvtt9Zaa++44w776quvWmutHTVqlH388cftAw88YF988UVrrbV79+61DRo0sNZa+/bbb9s6derYP//80/7222+2fPny9rXXXrPWWvvoo4/mLOfw4cM5r2PUqFF20qRJ1lpr+/fvn5PVWmvbtGljv/vuO2uttVu2bLGtW7e21lrbuXNnO3v2bGuttVOmTLGXXnppru9R9u2pqam2S5cudtq0afahhx6yo0ePttZaGxMTY5s0aWKttbZTp042Pj7eWmttcnKyTU1NtRs3brQdO3bMWd4bb7xhx4wZY6219vTp07ZZs2Z2z549duPGjbZMmTJ2z54956x74cKFtl27djYtLc3+8ssvtnr16vbgwYO5PudMNWrUsN9++60NCQmx1lobGBhov/nmm5z3Oq8sqamp9tixY9Zaaw8dOmTr1KljMzIy7N69e62/v7/9/PPPrbXW9uzZ077zzjvWWmurVq1qT58+ba219vfff881j6cp6M+giKsBW20unVHsRnBb9hwhJS2DDAupaRls2XOk0MucP39+zl/bffr0Yf78+Tn33XTTTdSuXRt/f3/69u1LfHw8AH5+fvTu3RuAfv365dwO5Nz+7bffUqtWLa677joA+vfvz+bNmwF45plnWLduHVu3buXJJ5/MNVfr1q0pV64clStX5rLLLqNz584ANGrUiKSkJAC2b99OixYtaNSoEXPnzuWbb745ZznHjx/no48+omfPngQGBnL//ffz888/A5CQkJAzmrnjjjvyfI9OnTpFYGAgQUFBXHPNNdxzzz3Ex8fn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1vLMM8/wyCOPEBUVxcqVK3NmufcUOopSREQKJD09nYcffpjXXnuNe+65h9dff52AAM+rE43gREQk31JSUoiOjua1117jqaee4s033/TIcgON4EREJJ9OnDhB9+7dWbt2LS+//DJPPPGE05HOSwUnIiIXdOTIETp27Minn37KW2+9xV133eV0pAtSwYmIyHn99NNPhIeH88MPP7Bo0SK6devmdKR8UcGJiEievvvuO8LDwzl69CirV68mNDTU6Uj5poITESmkbft+Z8ueIwTXrkSzGhWdjuMyn332GZGRkQDExsbStGlThxMVjApORKQQtu37negZW0hJy6BkgB9zBwb7RMnFxsbSpUsXLr/8ctauXct1113ndKQC08cEREQKYcueI6SkZZBhITUtgy17jjgdqdCWLFlCZGQk1atXJyEhwSvLDVRwIiKFEly7EiUD/PA3UCLAj+DalZyOVChvv/02PXr0IDAwkLi4OP7xj384HemiaROliEghNKtRkbkDg31iH9zLL7/Mk08+SXh4OIsWLaJs2bJORyoUFZyISCE1q1HRq4vNWsvw4cMZN24cvXv3Zs6cOZQsWdLpWIWmghMRKcbS0tIYNGgQM2fOZPDgwUyePBl/f3+nY7mE9sGJiBRTp0+fplevXsycOZNnnnmGqVOn+ky5gUZwIiLF0p9//km3bt3YuHEjEydOZMiQIU5HcjkVnIhIMXPo0CHat2/Pl19+ybvvvkt0dLTTkdxCBSciUoz8+OOPhIWFsX//fpYuXUqHDh2cjuQ2KjgRkWJix44dhIeHc+LECdatW0fz5s2djuRWOshERKQY+Pjjj2nRogXp6els2rTJ58sNVHAiIj5v3bp1tG3blgoVKpCQkEDjxo2djlQkVHAiIj5swYIFdOzYkf/7v/8jISGB2rVrOx2pyKjgRER81Ouvv07v3r25+eabiY2N5aqrrnI6UpFSwYmI+BhrLS+88AKDBw+mY8eOrFmzhgoVKjgdq8ip4EREfEhGRgZDhw7l6aefpl+/fixevJgyZco4HcsR+piAiIiPSE1N5e677+bdd9/lkUceYfz48fj5Fd9xjApORMQHnDp1il69erF8+XL+9a9/MXLkSIwxTsdylApORMTL/fHHH3Tp0oX4+Hhee+01Bg0a5HQkj6CCExHxYr/88guRkZHs2LGD9957j169ejkdyWOo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\n",
      "text/plain": [
       "<Figure size 504x504 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "figure(figsize=(7, 7))\n",
    "plot(true_beta, beta_loc, '.', label='Approximated Posterior Means')\n",
    "plot(true_beta, beta_loc + 2*jnp.exp(beta_log_scale), 'r.', label='Approximated Posterior $2\\sigma$ Error Bars')\n",
    "plot(true_beta, beta_loc - 2*jnp.exp(beta_log_scale), 'r.')\n",
    "plot_scale = 3\n",
    "plot([-plot_scale, plot_scale], [-plot_scale, plot_scale], 'k')\n",
    "xlabel('True beta')\n",
    "ylabel('Estimated beta')\n",
    "legend(loc='best')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "a = np.random.rand(32,10)\n",
    "b = np.random.rand(32,10)\n",
    "c = np.random.rand(3,10,10)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(3, 32, 10)"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "d = a@c\n",
    "d.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(3, 10, 10)"
      ]
     },
     "execution_count": 14,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "e = b.T@d\n",
    "e.shape"
   ]
  },
  {
   "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.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
