{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import math\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import torch\n",
    "import torch.autograd as autograd\n",
    "import torch.optim as optim\n",
    "import matplotlib.pyplot as plt\n",
    "device = torch.device(3 if torch.cuda.is_available() else 'cpu')\n",
    "import time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "import time\n",
    "import seaborn as sns"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch.distributions as D\n",
    "K = 10\n",
    "torch.manual_seed(1)\n",
    "mix = D.Categorical(torch.ones(K,).to(device))\n",
    "comp = D.Independent(D.Normal(\n",
    "             torch.randn(K,2).to(device), torch.ones(K,2).to(device)*0.1), 1)\n",
    "gmm = D.MixtureSameFamily(mix, comp)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x7fec12d9e7d0>]"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAXwAAAD4CAYAAADvsV2wAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjQuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/Z1A+gAAAACXBIWXMAAAsTAAALEwEAmpwYAAAbfElEQVR4nO3df4xd9Xnn8fdnxh6ytAgmtguGMeN4cWlid7vxzMLQRLuwJVFAFk4wFGi0gSrEzS6ojborLd2o7sor7ab9o1KruE1dEjWsCCEBE1xilkDiiO0fQz0zgtiG0jheBo9xYDATksiE8fg++8c9M76+vj/nnrm/zucljebce75zz+Pje5/7Pc/5nu9RRGBmZt2vp9UBmJlZczjhm5llhBO+mVlGOOGbmWWEE76ZWUYsa3UAlaxcuTLWrl3b6jDMzDrG+Pj4mxGxqtS6tk74a9euZWxsrNVhmJl1DEmT5da5pGNmlhFO+GZmGdFwwpe0RtI+SS9KOiTpD0q0kaS/lHRY0g8kbWp0u2ZmVp80avhzwH+OiAlJFwDjkp6OiBcL2twArE9+rgb+OvltZmZN0nAPPyKOR8REsvwz4CXgsqJmW4AHIm8UuEjS6ka3bWZmtUu1hi9pLfBB4LmiVZcBRwseT3Hul8L8a2yTNCZpbHp6Os3wzMwyLbWEL+mXgUeBz0XETxf7OhGxKyKGI2J41aqSQ0nN2sb45Aw79x1mfHKm1aGYVZXKOHxJy8kn+wcjYneJJseANQWPB5LnzDrW+OQMn7x/lNm5HH3Lenjw7hGGBvtbHZZZWWmM0hHwZeCliPjzMs32AJ9KRuuMAG9HxPFGt23WSqNHTjA7lyMXcGoux+iRE60OyayiNHr4HwL+A3BA0vPJc/8NuBwgIr4E7AVuBA4DJ4HfTWG7Zi01sm4Ffct6ODWXY/myHkbWrWh1SGYVNZzwI+IfAFVpE8A9jW7LrJ0MDfbz4N0jjB45wci6FS7nWNtr67l0zNrd0GC/E711DE+tYGaWEU74ZnXwMEzrZC7pmNWo1mGY45MzrutbW3LCN6tRqWGYxQndY/OtnTnhm9WocBhmb4947Sfv8LXnXmXm5OxCb774S2H3xJR7+9Y2lB8x2Z6Gh4fDd7yydjI+OcOjE1N8c+wop07nPzsCzlue780DfPL+0fyXQm8PRDCXC/f2rWkkjUfEcKl1PmlrVoehwX4EC8keIIDZghLPg3eP8IcfvZJbhgaYy8U5V+L6xK+1iks6ZnUqdUycC+g/vw84MzZ/fHKG3RNTZ12J6xq/tZITvlmdNl56Ib2Cgk4+AmZOzi48nh+ps33zhrNq/Dv3Ha564tdsqTjhm9VhfHKGHU8cIoAe5Xv2kO/1z/fwK/XiPf+OtZITvlkdCkfhiPzPfPKf7+FXGr7p+XeslZzwzepw1tDMZBTO6Vyc1VvvP7+PHuW/Ckr14j3/jrWKE75ZHYp76MBZvfX5ks/pXNDbI7Zv3uDkbm3DCd+sTqV66PNDLufLOQFExFkncs1azQnfrAHFJ2i3b97gk7LWtpzwzRpQfIJ25uSsT8pa23LCN1uE+XH2/ef3ndOj90lZa1epJHxJXwE2A29ExMYS668FHgf+X/LU7ojYkca2zZqtVBmn8OIqs3aVVg//74AvAg9UaPN/I2JzStszW3Ll5rUvVca557orWhipWW1SSfgR8ayktWm8llk78NWy1o2aWcO/RtILwGvAf4mIQ6UaSdoGbAO4/PLLmxie2Rm+Wta6UbMS/gQwGBE/l3Qj8C1gfamGEbEL2AX5+fCbFJ/ZWar14n1i1jpRUxJ+RPy0YHmvpL+StDIi3mzG9s3q5V68daOmJHxJlwCvR0RIuor8jVdONGPbZovlXrx1m7SGZT4EXAuslDQF/AmwHCAivgTcAvxHSXPAO8Dt0c73VjQz60JpjdK5o8r6L5IftmlmZi3ie9qamWWEE75ZC/hG5tYKnkvHrMnGJ2e4429HF4Z8PvSZ8jcyL3e1r9liOOGbNdnuiSlm53IAzM7l2D0xVTKZV7ra12wxXNIxa7Li4WnlhquVutrXrBFO+GZVpF1v37ppgL5eIaCvV2zdNFCy3fzVvr3Cc/ZYKlzSMatgKcoqQ4P9PLTtmqq1+aHBfrZv3sCTB49zw8bVLudYw5zwzSqoNIlaKZVOshavq/Y6uyem+ObYUeZywf5X3uLKSy5w0reGOOGbVVDPVMiVjgaqrSv8Iphv++6p3EJ9v5YvG7NqnPDNKqhnErXdE1P84tSZ0TeFCbrckUJhcu/tETu2bGTm5Cyzc2eSvXAN39LhhG9WRS2TqI1PzvDw/lcXHucC+s/vW3hc7khh9MiJhZ78XC74428d4H98/NcX2vb29nDL0ABbNw24d28Nc8I3S8HokROczp393MzJ2YXlckcKI+tW0CM4nXTnTwccfO1tT81sS8IJ3ywFI+tWsLxXzCaZu6/Gm6YMDfbzW++/mO+8+PrCc4eOvc3WTQO+T66lTu08S/Hw8HCMjY21OgyzmoxPzvDoxBQCbq6jBFM41UKQr9mft9xX1triSBqPiOFS63zhlVlKhgb7+Z+f+HVu3jTA6JETZ12oVeniraHBfh76zAgfXr8Skb/y1lfW2lJwSccsRaWGXwJVL94aGuznc9f/KvtfeaumIaBmi+GEb5aicvPf1HLxlu+ja0vNCd8sReWGX9Z68Zbvo2tLKa172n4F2Ay8EREbS6wX8BfAjcBJ4K6ImEhj22btpFwv3T13awdp9fD/jvw9ax8os/4GYH3yczXw18lvs65TbvilE721WiqjdCLiWeCtCk22AA9E3ihwkaTVaWzbzMxq06xhmZcBRwseTyXPnUPSNkljksamp6ebEpyZWRa03Tj8iNgVEcMRMbxq1apWh2Nm1jWalfCPAWsKHg8kz5mZWZM0K+HvAT6lvBHg7Yg43qRtm5kZ6Q3LfAi4FlgpaQr4E2A5QER8CdhLfkjmYfLDMn83je1a/SrdkcnMulsqCT8i7qiyPoB70tiWLd5S3J/VzDpH2520taVT7rJ/M8sGJ/wMmb/sv1eLu2VepRkfzaz9eS6dDKlncq5yN9Z2OciscznhZ0yt92ctTu7lbsJtZp3DJR07R6nk3mg5yMxazz18O0e5KX63bhoggI2XXrhwwte9fLPO4YRv5yiu9cOZOzYt6+3hkbGjzOXCtXyzDuOEbyUV1vp37jt8VokHzr7vqhO+WWdwwreqCks8vb09EMHpXLiWb9ZhnPCtqlIlHk/PYNZ5nPCtJsXDOZ3ozTqPh2VaRb661qx7uIdvZfnqWrPu4h6+leXJ1sy6ixO+leWra826i0s6VlY9k62ZWftzwreKaplszcw6g0s6ZmYZkUrCl/QxSS9LOizpvhLr75I0Len55OfuNLZrZma1a7ikI6kX2Al8BJgC9kvaExEvFjV9OCLubXR7Zma2OGn08K8CDkfEkYiYBb4ObEnhdc3MLEVpJPzLgKMFj6eS54ptlfQDSY9IWlPuxSRtkzQmaWx6ejqF8MzMDJp30vbvgbUR8a+Ap4GvlmsYEbsiYjgihletWtWk8LqDp0Ews0rSGJZ5DCjssQ8kzy2IiMJLNO8H/iyF7VqB4mkQtm/ewMzJ2Yrj54tvVG5m3S2NhL8fWC/pfeQT/e3A7xQ2kLQ6Io4nD28CXkphu1bg0Ykp3j2VI4B3T+X4428dIKDsHDieJ8csexou6UTEHHAv8BT5RP6NiDgkaYekm5Jmvy/pkKQXgN8H7mp0u3bG+OQMj4xPEcnjAE4HFefA8Tw5ZtmTypW2EbEX2Fv03PaC5T8C/iiNbWVFPeWW0SMnmDudK7mu3Bw45W5Ubtnj0l52eGqFNlRvuWU+ef/i1NlJX8D2zRtK/q3nyTFwaS9rPLVCG6pUbikeiTPfO9u+eQOfvPrys/5DBcycnC27naHBfu657gp/wDPMpb1scQ+/zYxPzvDaT95hWY/OuVF4qZE4O544dFbvbMOlF7L98YPkckHfcpdqsq5aucalvWxxwm8jhQl9WW8Pt121hq2bBhY+qMW9sScPHj+nd3bPdVdw5SUXlP2Qu16bHbWUayqV9vxe6T5O+G2kMKGfPp3jsov+xVkftOLe2A0bV7P/lbfO6Z2Vm9LY9dps2V0wVHe+Q1DufI6H7WaDE34bqXZ4Xao3Vqk3X6xUvdYf4u40PjnDN8eOLgzV7e2tr1zzaI1fFtZZnPDbSC0jZ4p7Y/XcoMT12uwYPXKCuVw+3Qu4ZWig5vdJ8XUdvT3ye6VLOOG3maW8w5SHYmZH8Zf71k0DVf9mvmb/2k/eWbiuQ8Ctw2sYGux3Tb8LKCKqt2qR4eHhGBsba3UYZh2pngR91oCBHoHE6dP5L4sH7x4BcE2/Q0gaj4jhUuvcw28j7kFZmuo5WjxrwEAuuO2q/HyIAl7+8c9Kjgjze7TzOOG3CY+KsFYqLgFtvPTChWs8cgVFgB6Vn67D2p8TfptYzAgaHxFYWorP7xQO6ZzXA3zoipV87vpf9futQznht4l6R9D4iMDSNv/+2T0xxcP7Xz0n2fct73Gy73BO+G2i3hE0HlNvafvac6+y/fGDC8M5IV/D/8gHLuY31lzkI8ku4ITfRjym3lplfHKmZLI/b3kPv/fv/mWmE33apdNWlmKd8DuUx9RbmkaPnOB0QbLv7RG3/Zuz53LKorRLp60uxTrhd7Byc6D4S8DqNbJuBect72H2VI6eHrFjy0Z+5+rLWx1Wy6VdOm11KdYJv4u0uvdgnctHjHnFHaa0S6etLsWmkvAlfQz4C6AXuD8ivlC0/jzgAWAIOAHcFhGvpLFtO6PVvQfrbEs5rUcnKNdhWswXYbkj7VZ/sTac8CX1AjuBjwBTwH5JeyLixYJmnwZmIuIKSbcDfwrc1ui27WyVeg8u9ZhVVq7DVO8XYbUj7VZ+sabRw78KOBwRRwAkfR3YAhQm/C3Af0+WHwG+KEnRzhP5dKByvQeXesyqq6fcUqkD1c5H2mkk/MuAowWPp4Cry7WJiDlJbwMrgDeLX0zSNmAbwOWX+6RRvUr1Htr5DWjWLmottxRPNHfr8BpuLhjN1Oo6fSVtd9I2InYBuyA/W2aLw+kK7fwGNGsntZRbCjtQs6eDrz33Ko9OTLF98wZmTs4ysm5F254ATyPhHwPWFDweSJ4r1WZK0jLgQvInb60JWn2iyKybzHeg5ucaCmB2Lsf2xw+Si1gom95z3RWtDvUcPSm8xn5gvaT3SeoDbgf2FLXZA9yZLN8CfM/1++YaGuznnuuucLI3a9B8B+qOqy+nb1kPvYIeidO5OKtsWs345Aw79x1mfHKmCVHnNdzDT2ry9wJPkR+W+ZWIOCRpBzAWEXuALwP/W9Jh4C3yXwpmZh1pvvSzddMAo0dO0H9+HzueONT2kx+mUsOPiL3A3qLnthcs/wK4NY1tWXkeemnWXMU1/ycPHueGjavbdvLDtjtpa7UrTPDgW9CZNVPx52/+hjH7X3mLKy+5oOLnr1UDKZzwO1TxIeHNmwY89NKsSYo/f1vr/PxVGkixlEfqTvgdqviQUOChl2ZNUvz5C+r//JWb/HApj9Sd8DtU8SHhzZsGuDk5geQavtnSKv78bd00sHACdz7Z79x3uO7P4lLX9p3wO1S5Q0InerOlV+nz10gvfalr+074HSzrsxuatVK5z18jvfSlvkjSCd/MLEWN9tKXsiPnhN9FPA7frPXaeSoTJ/wuUa1u6C8Ds+Zp13KrE36XqFQ39Hz4ZgbpTJ5mbWC+btgrzqkblvoyMLPscQ+/S1SqG3o+fDMDUDvPUjw8PBxjY2OtDqMruIZvlg2SxiNiuNQ69/Azol1PIplZ87iGb2aWEU74ZmYZ4YRvZpYRTvhmZhnRUMKX9F5JT0v6YfK75FlBSaclPZ/8FN/g3MzMmqDRHv59wHcjYj3w3eRxKe9ExL9Ofm5qcJtmLTc+OcPOfYcZn5xpdShmNWt0WOYW4Npk+avA94H/2uBrmrU1T1VhnarRHv7FEXE8Wf4xcHGZdu+RNCZpVNLHK72gpG1J27Hp6ekGwzOrrt7euqeqsE5VtYcv6RngkhKrPl/4ICJCUrnLdgcj4pikdcD3JB2IiB+VahgRu4BdkL/Stlp8ZrD4K4kX01v3VBXWqaom/Ii4vtw6Sa9LWh0RxyWtBt4o8xrHkt9HJH0f+CBQMuGb1auREsti7k7UzvOdm1XSaElnD3Bnsnwn8HhxA0n9ks5LllcCHwJebHC7ZgsaKbFUmmW0kqHBfu657gone+sojZ60/QLwDUmfBiaB3waQNAx8NiLuBt4P/I2kHPkvmC9EhBO+paaREot765Ylni3TukKrZgP1LKTWbjxbpnW9VswG6uGZ1mk8tYJlUhoXTnl4pnUa9/Atc8YnZ7jjb0cXav4PfWZxPXMPz7RO05UJ33VVq2T3xBSzczkAZudy7J6YWtT7xCd8rdN0XcJ3XdWqKR6m0MiwBd9JzDpJ19XwXVe1arZuGqCvVwjo6xVbNw20OiSzpui6Hn65uqrLPDZvaLCfh7Zd4/eDZU5XjsMvTu4u85hZVmRuHH5xXXUx86WYmXWbrqvhl7LY+VLMzLpJV/bwi3n4nJlZRhI+ePicmVkmSjpmZuaEb2aWGU74ZmYZ4YRvZpYRTvhmZhnhhG9mlhENJXxJt0o6JCmX3Me2XLuPSXpZ0mFJ9zWyTTMzW5xGe/gHgZuBZ8s1kNQL7ARuAD4A3CHpAw1u18zM6tTQhVcR8RKApErNrgIOR8SRpO3XgS3Ai41s28zM6tOMGv5lwNGCx1PJcyVJ2iZpTNLY9PT0kgdnZpYVVXv4kp4BLimx6vMR8XjaAUXELmAX5KdHTvv1zcyyqmrCj4jrG9zGMWBNweOB5DkzM2uiZpR09gPrJb1PUh9wO7CnCds1M7MCjQ7L/ISkKeAa4NuSnkqev1TSXoCImAPuBZ4CXgK+ERGHGgvbzMzq1egonceAx0o8/xpwY8HjvcDeRrZlZmaN8ZW2ZmYZ4YRvZpYRTvhmZhnhhG9mlhFO+GZmGeGEb2aWEU74ZmYZ4YRvZpYRTvhmZhnhhG9mlhFO+GZmGeGEb2aWEU74ZmYZ4YRvZpYRTvhmZhnhhG9mlhFO+GZmGeGEb2aWEY3e0/ZWSYck5SQNV2j3iqQDkp6XNNbINq1x45Mz7Nx3mPHJmVaHYmZN1NA9bYGDwM3A39TQ9rqIeLPB7VkNxidnGD1ygpF1Kxga7D9n3SfvH2V2Lkffsh4evHuEocH+in9jZt2h0ZuYvwQgKZ1orGHlEvq80SMnmJ3LkQs4NZdj9MgJgIp/Y2bdoVk1/AC+I2lc0rZKDSVtkzQmaWx6erpJ4XWmUqWZcgl93si6FfQt66FXsHxZDyPrVlT9GzPrDlV7+JKeAS4pserzEfF4jdv5cEQck/QrwNOS/ikini3VMCJ2AbsAhoeHo8bXz5xyPfn5hH5qLreQ0AsNDfbz4N0j55RvKv2NmXWHqgk/Iq5vdCMRcSz5/Yakx4CrgJIJ32pTqlc+NNhfNqHPK1Wrn/+bRyemcHHOrHs1etK2Kkm/BPRExM+S5Y8CO5Z6u92uUk9+PvEXq1bf3z0xxexcjkcnplzHN+tCjQ7L/ISkKeAa4NuSnkqev1TS3qTZxcA/SHoB+Efg2xHxfxrZrp3plf/hR6+sOTlXqtW7jm/W/RodpfMY8FiJ518DbkyWjwC/0ch2rLRyPflyKh0VVKv9m1nnU0T7nhcdHh6OsTFfp5WmamP0PRbfrLNJGo+IkhfCLnkN39pL4VFBcYKvdsTgLwSzzuaEn1HVTuA22t7M2o8nT8uoek/S+qSuWedzws+I4qtyi6+47T+/r+KEaqWu0DWzzuKSTgaUK8fMX6DVf34fO544VLFcU+2CLjNrf+7hZ0C5cszQYD/3XHcFMydnayrXzLd3sjfrTE74GVCtHONyjVk2eBx+RlQbUukhl2bdwePwreoY+3qv2jWzzuOSjplZRjjhm5llhBO+mVlGOOGbmWWEE76ZWUY44ZuZZURbj8OXNA1MpvBSK4E3U3idZnCsS8Oxpq9T4oRsxToYEatKrWjrhJ8WSWPlLkRoN451aTjW9HVKnOBY57mkY2aWEU74ZmYZkZWEv6vVAdTBsS4Nx5q+TokTHCuQkRq+mZllp4dvZpZ5TvhmZhnRlQlf0q2SDknKSSo7vEnSK5IOSHpeUksm3q8j1o9JelnSYUn3NTPGghjeK+lpST9MfpecT1nS6WSfPi9pTxPjq7iPJJ0n6eFk/XOS1jYrthKxVIv1LknTBfvx7lbEmcTyFUlvSDpYZr0k/WXyb/mBpE3NjjGJo1qc10p6u2Cfbm92jAWxrJG0T9KLyef/D0q0SX+/RkTX/QDvB64Evg8MV2j3CrCy3WMFeoEfAeuAPuAF4AMtiPXPgPuS5fuAPy3T7uctiK3qPgL+E/ClZPl24OEW/Z/XEutdwBdbEV+JeP8tsAk4WGb9jcCTgIAR4Lk2jfNa4IlW788kltXApmT5AuCfS7wHUt+vXdnDj4iXIuLlVsdRixpjvQo4HBFHImIW+DqwZemjO8cW4KvJ8leBj7cghnJq2UeF8T8C/JYkNTHGee3y/1mTiHgWeKtCky3AA5E3ClwkaXVzojujhjjbRkQcj4iJZPlnwEvAZUXNUt+vXZnw6xDAdySNS9rW6mAquAw4WvB4inPfHM1wcUQcT5Z/DFxcpt17JI1JGpX08eaEVtM+WmgTEXPA20ArbuBb6//n1uRQ/hFJa5oT2qK0y/uzFtdIekHSk5I2tDoYgKS0+EHguaJVqe/Xjr3FoaRngEtKrPp8RDxe48t8OCKOSfoV4GlJ/5T0ElKVUqxNUSnWwgcREZLKjekdTPbrOuB7kg5ExI/SjrXL/T3wUES8K+n3yB+Z/PsWx9TpJsi/N38u6UbgW8D6VgYk6ZeBR4HPRcRPl3p7HZvwI+L6FF7jWPL7DUmPkT/UTj3hpxDrMaCwhzeQPJe6SrFKel3S6og4nhxavlHmNeb36xFJ3yffe1nqhF/LPppvMyVpGXAhcGKJ4yqlaqwRURjX/eTPn7Srpr0/G1GYUCNir6S/krQyIloyqZqk5eST/YMRsbtEk9T3a2ZLOpJ+SdIF88vAR4GSZ/fbwH5gvaT3Seojf8KxaaNfCuwB7kyW7wTOOTqR1C/pvGR5JfAh4MUmxFbLPiqM/xbge5GcHWuyqrEW1WpvIl/jbVd7gE8lo0pGgLcLSn9tQ9Il8+dsJF1FPv+14gufJI4vAy9FxJ+XaZb+fm312eql+AE+Qb7e9S7wOvBU8vylwN5keR350REvAIfIl1faMtY4c8b+n8n3lFsV6wrgu8APgWeA9ybPDwP3J8u/CRxI9usB4NNNjO+cfQTsAG5Klt8DfBM4DPwjsK6F79Fqsf6v5H35ArAP+LUWxvoQcBw4lbxXPw18Fvhssl7AzuTfcoAKI+NaHOe9Bft0FPjNFu7TD5M/h/gD4Pnk58al3q+eWsHMLCMyW9IxM8saJ3wzs4xwwjczywgnfDOzjHDCNzPLCCd8M7OMcMI3M8uI/w+dUBzQxzyTSwAAAABJRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "sample = gmm.sample((100,)).cpu()\n",
    "plt.plot(sample[:,0],sample[:,1],'.')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def divergence_bf(dx, y):\n",
    "    sum_diag = 0.\n",
    "    for i in range(y.shape[1]):\n",
    "        sum_diag += torch.autograd.grad(dx[:, i].sum(), y, create_graph=True)[0].contiguous()[:, i].contiguous()\n",
    "    return sum_diag.contiguous()\n",
    "\n",
    "def sample_rademacher_like(y):\n",
    "    return torch.randint(low=0, high=2, size=y.shape).to(y) * 2 - 1\n",
    "\n",
    "\n",
    "def sample_gaussian_like(y):\n",
    "    return torch.randn_like(y)\n",
    "\n",
    "def divergence_approx(f, y, e=None):\n",
    "    if e is None:\n",
    "        e = sample_rademacher_like(y)\n",
    "    e_dzdx = torch.autograd.grad(f, y, e, create_graph=True)[0]\n",
    "    e_dzdx_e = e_dzdx * e\n",
    "    approx_tr_dzdx = e_dzdx_e.view(y.shape[0], -1).sum(dim=1)\n",
    "    return approx_tr_dzdx\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "    \n",
    "\n",
    "    \n",
    "class PFGfast:\n",
    "  def __init__(self, P, net, optimizer1,optimizer2):\n",
    "    self.P = P\n",
    "    self.net = net\n",
    "    self.optim1 = optimizer1\n",
    "    self.optim2 = optimizer2\n",
    "\n",
    "  def phi(self, X):\n",
    "    #self.net.eval()\n",
    "    #X = X.detach().requires_grad_(True)\n",
    "    \n",
    "    #log_prob = self.P.log_prob(X)\n",
    "    #score_func = torch.autograd.grad(log_prob.sum(), X)[0]\n",
    "    \n",
    "    phi = self.net(X) / X.size(0)\n",
    "\n",
    "    return -phi\n",
    "\n",
    "  def step(self, X):\n",
    "    self.optim1.zero_grad()\n",
    "    X.grad = self.phi(X)\n",
    "    self.optim1.step()\n",
    "\n",
    "  def score_step(self, X):\n",
    "    H = torch.std(X,0)\n",
    "    X = X.detach().requires_grad_(True)\n",
    "    \n",
    "    log_prob = self.P.log_prob(X)\n",
    "    score_func = torch.autograd.grad(log_prob.sum(), X)[0].detach()\n",
    "    \n",
    "    self.net.train()\n",
    "    \n",
    "    X = X.detach().requires_grad_(True)\n",
    "    S = self.net(X)\n",
    "\n",
    "\n",
    "    H = torch.eye(X.shape[1])\n",
    "\n",
    "    self.optim2.zero_grad()\n",
    "    \n",
    "    loss = (-torch.sum(score_func*S) - torch.sum(divergence_approx(S,X)) + 0.5*torch.trace(H.matmul(S.T).matmul(S)))/S.shape[0]\n",
    "    loss.backward()\n",
    "    self.optim2.step()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "from torch import nn\n",
    "\n",
    "n = 1000\n",
    "X_init = (torch.randn(n, 2)).to(device)\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "X = X_init.clone()\n",
    "\n",
    "h = 32\n",
    "net = nn.Sequential(nn.Linear(2,h),nn.Tanh(),nn.Linear(h,h),nn.Tanh(),nn.Linear(h,2)).to(device)\n",
    "\n",
    "s = lambda x: net(x)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "27.072798013687134\n"
     ]
    }
   ],
   "source": [
    "pfg = PFGfast(gmm, net,optim.SGD([X], lr=1e-1,momentum=0.),optim.SGD(net.parameters(), lr=1e-3, momentum=0.9,nesterov=1))\n",
    "\n",
    "\n",
    "#optim.SGD(net.parameters(), lr=1e-2,momentum=0.)\n",
    "t1 = time.time()\n",
    "\n",
    "for i in range(100):\n",
    "    pfg.score_step(X)\n",
    "\n",
    "for i in range(2000):\n",
    "    \n",
    "    for j in range(5):\n",
    "        pfg.score_step(X)\n",
    "     \n",
    "\n",
    "    pfg.step(X)\n",
    "print(time.time()-t1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(-2.0, 2.0)"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.plot(X.cpu().numpy()[:,0],X.cpu().numpy()[:,1],'x')\n",
    "plt.xlim([-2,2])\n",
    "plt.ylim([-2,2])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "class RBF(torch.nn.Module):\n",
    "  def __init__(self, sigma=None):\n",
    "    super(RBF, self).__init__()\n",
    "\n",
    "    self.sigma = sigma\n",
    "\n",
    "  def forward(self, X, Y):\n",
    "    XX = X.matmul(X.t())\n",
    "    XY = X.matmul(Y.t())\n",
    "    YY = Y.matmul(Y.t())\n",
    "\n",
    "    dnorm2 = -2 * XY + XX.diag().unsqueeze(1) + YY.diag().unsqueeze(0)\n",
    "\n",
    "    # Apply the median heuristic (PyTorch does not give true median)\n",
    "    if self.sigma is None:\n",
    "        np_dnorm2 = dnorm2.detach()\n",
    "        h = torch.median(np_dnorm2)/(2 * np.log(X.size(0) + 1))\n",
    "        sigma = torch.sqrt(h).item()\n",
    "    else:\n",
    "        sigma = self.sigma\n",
    "\n",
    "    gamma = 1.0 / (1e-8 + 2 * sigma ** 2)\n",
    "    K_XY = (-gamma * dnorm2).exp()\n",
    "\n",
    "    return K_XY\n",
    "\n",
    "  def dK(self, X, Y):\n",
    "    XX = X.matmul(X.t())\n",
    "    XY = X.matmul(Y.t())\n",
    "    YY = Y.matmul(Y.t())\n",
    "\n",
    "    dnorm2 = -2 * XY + XX.diag().unsqueeze(1) + YY.diag().unsqueeze(0)\n",
    "\n",
    "    # Apply the median heuristic (PyTorch does not give true median)\n",
    "    if self.sigma is None:\n",
    "        np_dnorm2 = dnorm2.detach()\n",
    "        h = torch.median(np_dnorm2)/(2 * np.log(X.size(0) + 1))\n",
    "        sigma = torch.sqrt(h).item()\n",
    "    else:\n",
    "        sigma = self.sigma\n",
    "\n",
    "    gamma = 1.0 / (1e-8 + 2 * sigma ** 2)\n",
    "    K_XY = (-gamma * dnorm2).exp() #bX x bY\n",
    "    \n",
    "    X_Y = X.view(X.shape[0],1,X.shape[1])-Y.view(1,Y.shape[0],Y.shape[1]) # bX x bY x d\n",
    "    #print(X_Y.shape,K_XY.shape)\n",
    "    dKXY = torch.sum(K_XY.view(K_XY.shape[0],K_XY.shape[1],1) * (-gamma) * X_Y,0)\n",
    "    return dKXY\n",
    "  \n",
    "# Let us initialize a reusable instance right away.\n",
    "Ker = RBF()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "class SVGD:\n",
    "  def __init__(self, P, K, optimizer):\n",
    "    self.P = P\n",
    "    self.K = K\n",
    "    self.optim = optimizer\n",
    "\n",
    "  def phi(self, X):\n",
    "    X = X.detach().requires_grad_(True)\n",
    "\n",
    "    log_prob = self.P.log_prob(X)\n",
    "    score_func = autograd.grad(log_prob.sum(), X)[0]\n",
    "\n",
    "    K_XX = self.K(X, X.detach())\n",
    "    grad_K = self.K.dK(X.detach(), X)\n",
    "    #-autograd.grad(K_XX.sum(), X)[0]\n",
    "\n",
    "    phi = (K_XX.detach().matmul(score_func) + grad_K) / X.size(0)\n",
    "\n",
    "    return phi\n",
    "\n",
    "  def step(self, X):\n",
    "    self.optim.zero_grad()\n",
    "    X.grad = -self.phi(X)\n",
    "    self.optim.step()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "61.39349818229675\n"
     ]
    }
   ],
   "source": [
    "X = X_init.clone()\n",
    "svgd = SVGD(gmm, Ker, optim.SGD([X], lr=1e-2,momentum=0.))\n",
    "t1 = time.time()\n",
    "for i in range(2000):\n",
    "    #print(i)\n",
    "    svgd.step(X)\n",
    "    \n",
    "print(time.time()-t1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(-2.0, 2.0)"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.plot(X.cpu().numpy()[:,0],X.cpu().numpy()[:,1],'x')\n",
    "plt.xlim([-2,2])\n",
    "plt.ylim([-2,2])"
   ]
  },
  {
   "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.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
