{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "import sys\n",
    "sys.path.insert(0, '..')\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "from pathlib import Path\n",
    "\n",
    "import pickle, gzip, torch\n",
    "import torch.distributions as dist\n",
    "import numpy as np\n",
    "\n",
    "from rnn_utils import load_rnn_model, load_smiles_from_list, rnn_start_token_vector\n",
    "from smiles_char_dict import SmilesCharDictionary"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "test_smiles, test_prop = np.load(\"../../data/QM9/QM9_clean_smi_test_smile.npz\",allow_pickle=True).values()\n",
    "#prior = pickle.load(gzip.open('../data/prior.pkl.gz'))\n",
    "train_smiles, train_prop = np.load(\"../../data/QM9/QM9_clean_smi_train_smile.npz\",allow_pickle=True).values()\n",
    "\n",
    "model_path = '../lstm/output/QM9/2020_1_17/model_79_0.167.pt'\n",
    "model_def = Path(model_path).with_suffix('.json')\n",
    "\n",
    "# get two RNNs: one trained, one random\n",
    "trained_model = load_rnn_model(model_def, model_path, device='cpu', copy_to_cpu=True)\n",
    "random_model = load_rnn_model(model_def, model_path, device='cpu', copy_to_cpu=True)\n",
    "random_model.init_weights()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "47"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "trained_model.input_size"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Define a function to sample from the RNN-based model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "def sample_from_model(model, y, N_samples, max_len=100, greedy=False):\n",
    "    \"\"\" Draw samples from the model, and return three things:\n",
    "    \n",
    "        (1) the sampled values\n",
    "        (2) the probability corresponding to each of those sampled values\n",
    "        (3) the probability distributions used for sampling at each t = 1,…,T\n",
    "        \n",
    "        If the argument `greedy=True`, then instead of sampling at each\n",
    "        step it deterministically draws the maximum of the distribution.\n",
    "    \"\"\"\n",
    "    device = y.device\n",
    "    spot_check = True\n",
    "    model.eval()\n",
    "    \n",
    "    # hold the output here\n",
    "    actions = torch.zeros((N_samples, max_len), dtype=torch.long).to(device)\n",
    "    sampled_probs = torch.zeros((N_samples, max_len)).to(device)\n",
    "    sampling_dists = torch.zeros((N_samples, max_len, model.input_size)).to(device)\n",
    "    #sampling_dists = np.empty((batch_size, max_len), dtype=object)\n",
    "\n",
    "    # initialization for LSTM sampling\n",
    "    hidden = model.init_hidden(N_samples, device)\n",
    "    inp = rnn_start_token_vector(N_samples, device)\n",
    "    \n",
    "    # run character-by-character...\n",
    "    for char in range(max_len):\n",
    "        # one step forward\n",
    "        output, hidden = model(inp, y.expand(N_samples, -1), hidden)\n",
    "\n",
    "        prob = torch.softmax(output, dim=2)\n",
    "        if greedy:\n",
    "            # alternative, if we want to take the max instead of sampling\n",
    "            action = output.argmax(dim=2)             \n",
    "        else:\n",
    "            d = dist.Categorical(probs=prob)\n",
    "            action = d.sample()\n",
    "            \n",
    "        # store results\n",
    "        actions[:,char] = action.squeeze(1)\n",
    "        sampled_probs[:,char] = prob.squeeze(1)[torch.arange(len(action)),action.squeeze(1)]\n",
    "        sampling_dists[:,char] = prob.squeeze(1)\n",
    "\n",
    "        # double-check there is no bug\n",
    "        if spot_check:\n",
    "            j = np.random.randint(N_samples)\n",
    "            assert sampling_dists[j,char,action[j]] == sampled_probs[j,char]\n",
    "\n",
    "        # input into next timestep of LSTM\n",
    "        inp = action\n",
    "      \n",
    "\n",
    "    return actions, sampled_probs, sampling_dists\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Different ways of approximating the entropy\n",
    "\n",
    "The entropy of $p_\\theta(x | y) = \\prod_{t=1}^T p_\\theta(x_t | x_{1:t-1}, y)$ is\n",
    "\n",
    "$$H[p_\\theta(x|y)] = - E_{p_\\theta(x | y)}[\\log p_\\theta(x | y)] = -E_{p_\\theta(x | y)}\\left[\\sum_{t=1}^T \\log p_\\theta(x_t | x_{1:t-1}, y)\\right].$$\n",
    "\n",
    "The naive monte carlo estimator involves sampling trajectories $x$, given $y$, and then evaluating the log probabilities.\n",
    "\n",
    "$$\\hat H_{MC} = - \\frac{1}{S} \\sum_{s=1}^S \\sum_{t=1}^T \\log p_\\theta(x_t^s | x^s_{1:t-1}, y) $$\n",
    "\n",
    "for $x_t^s \\sim p_\\theta(x | y)$.\n",
    "\n",
    "The alternative way of approximating the entropy involves decomposing this into a sequence of other entropies. In this way, we have\n",
    "\n",
    "$$H[p_\\theta(x|y)] = H[p_\\theta(x_1 | y)] + \\sum_{t=2}^T E_{p_\\theta(x_{1:t-1}|y)}\\left[H[p_\\theta(x_t | x_{1:t-1}, y)]\\right].$$\n",
    "\n",
    "Since the entropy for each individual $x_t$ is cheap enough to compute directly in closed form, we can do so and just use sampling in order to generate the values we condition on at each step.\n",
    "This probably should be lower variance.\n",
    "Also, in the case when $x_t$ are sampled from continuous distributions, this also would be easy to directly reparameterize and use to take gradients.\n",
    "\n",
    "Let's try this for a few different number of samples, and see what the distribution of the estimators look like.\n",
    "We'll call the first one **Estimator A** and the second one **Estimator B**."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1569\n",
      "Our property vector today:\n",
      " [-0.845776   -0.45506914 -0.12361375 -0.50290503 -0.80384633  0.95706758\n",
      "  0.05098709 -0.11251635  1.04039543]\n"
     ]
    }
   ],
   "source": [
    "# pick a random properties value\n",
    "index = np.random.randint(len(test_smiles))\n",
    "print(index)\n",
    "#y = prior.transform(properties[index:index+1])\n",
    "def normalize(prop):\n",
    "    train_smiles, train_prop = np.load(\"../../data/QM9/QM9_clean_smi_train_smile.npz\",allow_pickle=True).values()\n",
    "    mean = np.mean(train_prop, axis = 0)\n",
    "    std = np.std(train_prop, axis = 0)\n",
    "    return (prop - mean) / std\n",
    "\n",
    "y = normalize(test_prop)[index:index+1]\n",
    "print(\"Our property vector today:\\n\", y.squeeze())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "CPU times: user 1.49 s, sys: 0 ns, total: 1.49 s\n",
      "Wall time: 251 ms\n"
     ]
    }
   ],
   "source": [
    "%%time\n",
    "\n",
    "with torch.no_grad():\n",
    "    samples, probs, dists = sample_from_model(trained_model, torch.FloatTensor(y), 15)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([15, 100, 47])"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "dists.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "dists[1,20, 1]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "def test_mc_estimators(model, y, N_samples, N_trials=15):\n",
    "#     mc_est = np.array([entropy_simple_mc(model, y, N_samples) for _ in range(N_trials)])\n",
    "    with torch.no_grad():\n",
    "        est_A = np.zeros((N_trials,))\n",
    "        est_B = np.zeros((N_trials,))\n",
    "        for i in range(N_trials):\n",
    "            samples, probs, dists = sample_from_model(model, torch.FloatTensor(y), N_samples)\n",
    "            est_A[i] = -probs.log().sum(-1).mean()\n",
    "            est_B[i] = -((dists+1e-45).log()*dists).sum(-1).sum(-1).mean()\n",
    "            \n",
    "    dist_A = dist.Normal(est_A.mean(), est_A.std())\n",
    "    dist_B = dist.Normal(est_B.mean(), est_B.std())\n",
    "\n",
    "    # plot them both\n",
    "    plt.subplot(121)\n",
    "    plt.hist(est_A, bins=20);\n",
    "    plt.title(\"%d samples, estimator A\" % N_samples)\n",
    "    plt.xlabel(\"mean = %0.3f, std = %0.3f\" % (dist_A.loc.item(), dist_A.scale.item()))\n",
    "\n",
    "    plt.subplot(122)\n",
    "    plt.hist(est_B, bins=20);\n",
    "    plt.title(\"%d samples, estimator B\" % N_samples)\n",
    "    plt.xlabel(\"mean = %0.3f, std = %0.3f\" % (dist_B.loc.item(), dist_B.scale.item()))\n",
    "\n",
    "\n",
    "    return est_A, est_B, dist_A, dist_B"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "CPU times: user 1min 22s, sys: 719 ms, total: 1min 22s\n",
      "Wall time: 14.7 s\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAeMAAADgCAYAAAAwlA4iAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4xLjEsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy8QZhcZAAAe4klEQVR4nO3de7wdZX3v8c+XJIDlFiRRQy5sFWiLFy5GRC01aKWAl1hBD3iBoDbWgkKPeASPBxE9FcWKIhaEggFERIFDo6QixeQAVsAkhASI2IhQQiIJBHIRBAO//vE8G4aVtfaavffae2bv9X2/XuuVWTPPmvnNZJ75zW0/jyICMzMzq85WVQdgZmbW7ZyMzczMKuZkbGZmVjEnYzMzs4o5GZuZmVXMydjMzKxiTsYjiKQeSSFpbNWxlCHpLkkzqo7DrG5cl61RVydjScdLWijpSUlzqo5nJJM0R9IXi+Mi4hURsWAIljVD0spOz7cw75D06aGYvw0N1+XOGel1WdIsSU9L2pQ/90r6WCeXMRS6OhkDq4AvAhdVHYgNnzZXI8cA64Cjhykc6wzX5S7UR13+RURsHxHbA4cDX5G07zCG1n8R0fUfUiWe06bM7sD/B9YDDwNXFKZ9A3gA2AAsAg4sTDsN+CHwXWAjsAzYEzgFWJN/d3Ch/ALgS8BteX7/CrwwT+sBAhibv+8EXAisBh7M6zGmXbwltseHgOXAo8B1wG55vICzctwb8rq8EpgN/BF4CtgE/CiXvw/4qwFuh2NzDBuBe4GP5vHbAU8Az+RlbQJ2BbYBvk46KK/Kw9vk38wAVgKfBn4HXNpivbfLyzsyr8v0qvdNf/r3cV3eYl27ri4Ds4CbG8bdBryv6v2zr0+3Xxn3xxeAnwI7A1OAbxam/RLYB3gh8D3gh5K2LUx/B3Bp/u3tpEqxFTAZOB34dsOyjiZVoknAZuDsFjHNydN3B/YFDgY+UiLeliTNBD4DvBuYCNwEXJ4nHwz8Jani7QS8F3gkIs4HLgO+Euls9B0tZt+f7bAGeDuwI6kynyVpv4j4PXAosCova/uIWAX8b+AA0v/D3sD+wGcL83sJ6f9nN9IBp5l3kw4IP8yxHdN6S9kI5ro8+utycTu8Nq/nwnZlK1X12UAdPpQ7m74EOB+YUmJ+jwJ75+HTgOsL095BOuD3nvXuQDpDHp+/LwDOKJTfi3SWOobC2TTwYuBJ4AWFskcB8/sbb0Ps/wZ8uPB9K+Bx0o7/ZuDXpIqyVcPv5gBfbBh3H88/my69HZrEdQ1wQh6eAaxsmP4b4LDC978G7iuUfwrYts26/zvw9cK2XAuMq3r/9Kdf+6/r8nPz6Mq6TLoy3gw8RroaD9IJjKreP/v6+Mq4vP9FurVzW36z8EO9EySdJGm5pPWSHiOdaU4o/PahwvATwMMR8XThO8D2hTIPFIbvB8Y1zA9ShRoHrJb0WF7ut4EXtYu3jd2AbxTmuS7PZ3JE/Aw4B/gWsEbS+ZJ2LDlf6Md2kHSopFskrctxHMaW26BoV9K26nV/HtdrbUT8odWPJU0FDiJdFUC6pbgt8La+V8lGINflUVyXs1siYnxE7EC6kn4F8I9tflMpJ+OSIuJ3EfG3EbEr8FHgnyXtLulAUmV5L7BzRIwnPdvRIBY3tTA8jfQM5+GGMg+QzqYn5J1ufETsGBGv6CveEst+gPRMZ3zh84KI+I8837Mj4jWks/w9gU/l33Ws+y9J2wBXAV8FXpy36Tye26bNlrWKdPDpNS2P69Uuvg+S6sOPJP2O9GxrW3yretRxXR71dfl5IuKhHEOrW+610NXJWNLY/DxoDDBG0rat3s6T9B5JU/LXR0k7xDOkWzKbSbc0x0o6lfRsZDA+IGkvSX9Cev5yZeGsE4CIWE16jvRPknaUtJWkl0t6U5t4kbRA0mktln0ecIqkV+SyO0l6Tx5+raTXSRoH/B74Q+88SWfKLxvkevfamvQSx1pgs6RDSc+4ej0E7CJpp8K4y4HPSpooaQJwKukFk7KOAT5Pek7V+zkcOEzSLgNeExsWrstNdWtdfp5cf/8GuGug8xgOXZ2MSS8FPAGcDHwgD3+2RdnXArdK2gTMJT3zuJf04sJPSM9f7ift1A+0mEdZl5Ke2/yOdHX2iRbljibt7HeTKumVpBdF+ooX0tn6z5vNMCL+H/Bl4PuSNgB3kl6ygHRguiAv637gEeDMPO1CYK98S+yafq5vYwwbSev8g7ys9+V16J3+K1KFvTcvb1fSs8KFwFLS252L87i2JB1AOhP/Vr4K6f3MBVaQnt9ZvbkuN+jGulzweuW/Mya9yb0W+Phg1mWoKT/wtpqQtAD4bkT8yxDNfwrwg4h4w1DM38wS12XrjxHRFJt1TkSsBFx5zUY41+XRpdtvU5uZmVXOt6nNzMwq5itjMzOzijkZm5mZVayyF7gmTJgQPT09VS3ebMRYtGjRwxExseo4+uL6bFZOq/pcWTLu6elh4cJ6t9ttVgeS7m9fqlquz2bltKrPvk1tZmZWMSdjMzOzirVNxrmN19sk3ZF7DPl8kzLbSLpC0gpJt0rqGYpgzWxwJE2VNF/S3bk+n9CkjCSdnevzUkn7VRGrWTcpc2X8JPDmiNib1Hj+Ibkt36IPA49GxO7AWaT2UM2sfjYDn4yIvUh92R4naa+GMocCe+TPbODc4Q3RrPu0TcaRbMpfx+VPY0shM4GL8/CVwFskDabbMTMbAhGxOiIW5+GNpEb0JzcUmwlckuv+LcB4SZMwsyFT6m1qSWOARcDupJ5tbm0oMpncu0lEbJa0HtiFhn47Jc0mnWkzbdq0wUVekZ6Try1V7r4z3Ce91Vt+nLQv0LI+ZyvzuNUNv++K+lymLndqPmUN9/LKGM6YRuNxuNQLXBHxdETsA0wB9pf0yoEsLCLOj4jpETF94sRa/9mk2agmaXtSh+snRsSGgczD9dmsc/r1NnVEPAbMBw5pmPQgqV9NcofeO5H6xzSzmskdyl8FXBYRVzcp8mx9zqbkcWY2RMq8TT1R0vg8/ALgrcCvGorNBY7Jw0cAPwv3QGFWO/ldjguB5RHxtRbF5gJH57eqDwDWR8TqFmXNrAPKPDOeBFycnxtvRerM+seSTgcWRsRcUuW+VNIKYB1w5JBFbGaD8Ubgg8AySUvyuM8A0wAi4jxgHnAYsAJ4HDi2gjjNukrbZBwRS0kveTSOP7Uw/AfgPZ0Nzcw6LSJuBvr8S4d8V+u44YnIzMAtcJmZmVXOydjMzKxiTsZmZmYVczI2MzOrmJOxmZlZxZyMzczMKuZkbGZmVjEnYzMzs4o5GZuZmVXMydjMzKxiTsZmZmYVczI2MzOrmJOxmZlZxZyMzczMKuZkbGZmVjEnYzMzs4o5GZuZmVWsbTKWNFXSfEl3S7pL0glNysyQtF7Skvw5dWjCNTMzG33GliizGfhkRCyWtAOwSNL1EXF3Q7mbIuLtnQ/RzMxsdGt7ZRwRqyNicR7eCCwHJg91YGZmZt2iX8+MJfUA+wK3Npn8ekl3SPo3Sa9o8fvZkhZKWrh27dp+B2tmZjYalU7GkrYHrgJOjIgNDZMXA7tFxN7AN4Frms0jIs6PiOkRMX3ixIkDjdnMzGxUKZWMJY0jJeLLIuLqxukRsSEiNuXhecA4SRM6GqmZmdkoVeZtagEXAssj4mstyrwkl0PS/nm+j3QyUDMzs9GqzNvUbwQ+CCyTtCSP+wwwDSAizgOOAD4maTPwBHBkRMQQxGtmZjbqtE3GEXEzoDZlzgHO6VRQZmZm3cQtcJmZmVXMydjMzKxiTsZmZmYVczI2MzOrmJOxWReRdJGkNZLubDHdnb6YVaDMnzaZ2egxh/SXD5f0UcadvpgNM18Zm3WRiLgRWFd1HGb2fE7GZtaobacv4I5fzDrJydjMikp1+gLu+MWsk5yMzexZ7vTFrBpOxmb2LHf6YlYNv01t1kUkXQ7MACZIWgl8DhgH7vTFrEpOxmZdJCKOajPdnb6YVcC3qc3MzCrmZGxmZlYxJ2MzM7OKORmbmZlVrG0yljRV0nxJd0u6S9IJTcpI0tmSVkhaKmm/oQnXzMxs9CnzNvVm4JMRsVjSDsAiSddHxN2FMocCe+TP64Bz879mZmbWRtsr44hYHRGL8/BGYDkwuaHYTOCSSG4Bxkua1PFozczMRqF+/Z2xpB5gX+DWhkmTgQcK31fmcasbfj8bmA0wbdq0/kVqg9Jz8rXDurz7znjbsC7PzGwkK/0Cl6TtgauAEyNiw0AW5oblzczMtlQqGUsaR0rEl0XE1U2KPAhMLXyfkseZmZlZG2XephZwIbA8Ir7Wothc4Oj8VvUBwPqIWN2irJmZmRWUeWb8RuCDwDJJS/K4zwDT4NnG5ecBhwErgMeBYzsfqpmZ2ejUNhlHxM2A2pQJ4LhOBWVmZtZN3AKXmZlZxZyMzczMKuZkbGZmVjEnYzMzs4o5GZuZmVXMydjMzKxiTsZmZmYVczI2MzOrmJOxmZlZxZyMzczMKuZkbGZmVjEnYzMzs4o5GZuZmVXMydjMzKxiTsZmZmYVczI2MzOrmJOxmZlZxdomY0kXSVoj6c4W02dIWi9pSf6c2vkwzawTStRnSTpb0gpJSyXtN9wxmnWjMlfGc4BD2pS5KSL2yZ/TBx+WmQ2ROfRdnw8F9sif2cC5wxCTWddrm4wj4kZg3TDEYmZDrER9nglcEsktwHhJk4YnOrPuNbZD83m9pDuAVcBJEXFXs0KSZpPOtpk2bVqHFm1mHTQZeKDwfWUet7qxYH/rc8/J13YkwPvOeFtH5jPaldnew70t6xhTGcMRdyde4FoM7BYRewPfBK5pVTAizo+I6RExfeLEiR1YtJlVxfXZrHMGnYwjYkNEbMrD84BxkiYMOjIzq8KDwNTC9yl5nJkNoUEnY0kvkaQ8vH+e5yODna+ZVWIucHR+q/oAYH1EbHGL2sw6q+0zY0mXAzOACZJWAp8DxgFExHnAEcDHJG0GngCOjIgYsojNbMBK1Od5wGHACuBx4NhqIjXrLm2TcUQc1Wb6OcA5HYvIzIZMifocwHHDFI6ZZW6By8zMrGJOxmZmZhVzMjYzM6uYk7GZmVnFnIzNzMwq5mRsZmZWMSdjMzOzijkZm5mZVczJ2MzMrGJOxmZmZhVzMjYzM6uYk7GZmVnFnIzNzMwq5mRsZmZWMSdjMzOzijkZm5mZVczJ2MzMrGJtk7GkiyStkXRni+mSdLakFZKWStqv82GamZmNXmWujOcAh/Qx/VBgj/yZDZw7+LDMzMy6R9tkHBE3Auv6KDITuCSSW4DxkiZ1KkAzM7PRbmwH5jEZeKDwfWUet7qxoKTZpKtnpk2b1nbGPSdfWyqA+854W6lynVpep3RqeZ1a/5GszLYc7u1Ux5jMrJ6G9QWuiDg/IqZHxPSJEycO56LNzMxqqxPJ+EFgauH7lDzOzMzMSuhEMp4LHJ3fqj4AWB8RW9yiNjMzs+baPjOWdDkwA5ggaSXwOWAcQEScB8wDDgNWAI8Dxw5VsGZmZqNR22QcEUe1mR7AcR2LyMzMrMu4BS4zM7OKORmbmZlVzMnYzMysYk7GZl1E0iGS7sltyZ/cZPosSWslLcmfj1QRp1m36UQLXGY2AkgaA3wLeCuppbxfSpobEXc3FL0iIo4f9gDNupivjM26x/7Aioi4NyKeAr5PalvezCrmZGzWPVq1I9/o8Nwd6pWSpjaZbmYd5mRsZkU/Anoi4tXA9cDFrQpKmi1poaSFa9euHbYAzUYjJ2Oz7tG2HfmIeCQinsxf/wV4TauZueMXs85xMjbrHr8E9pD0UklbA0eS2pZ/VkNf5O8Elg9jfGZdy29Tm3WJiNgs6XjgOmAMcFFE3CXpdGBhRMwFPiHpncBmYB0wq7KAzbqIk7FZF4mIeaTOXYrjTi0MnwKcMtxxmXU736Y2MzOrmJOxmZlZxZyMzczMKuZkbGZmVrFSydiNy5uZmQ2dtm9Tu3F5MzOzoVXmytiNy5uZmQ2hMsnYjcubmZkNoU69wFWqcXk3LG9mZralMsm4Y43Lu2F5MzOzLZVJxm5c3szMbAi1fZvajcubmZkNrVIdRbhxeTMzs6HjFrjMzMwq5mRsZmZWMSdjMzOzijkZm5mZVczJ2MzMrGJOxmZmZhVzMjYzM6uYk7GZmVnFnIzNzMwq5mRsZmZWMSdjMzOzijkZm5mZVczJ2MzMrGJOxmZmZhVzMjYzM6uYk7GZmVnFnIzNzMwqVioZSzpE0j2SVkg6ucn0bSRdkaffKqmn04GaWWe4PpvVT9tkLGkM8C3gUGAv4ChJezUU+zDwaETsDpwFfLnTgZrZ4Lk+m9VTmSvj/YEVEXFvRDwFfB+Y2VBmJnBxHr4SeIskdS5MM+sQ12ezGiqTjCcDDxS+r8zjmpaJiM3AemCXTgRoZh3l+mxWQ2OHc2GSZgOz89dNku7pyHzb30SbADzciWWVVSKmokHF189lDVS/YhymmBr1GWNFMTV6XowlY9ptqIIZjKGqzw22+D8d7v/HTi1PXx7+41A7g1i3IVuXTv7/lpxXR9alH3E3rc9lkvGDwNTC9yl5XLMyKyWNBXYCHmmcUUScD5xfJtpOkrQwIqYP93LLqnt84Bg7pQYxjqj6XIPt1TFel3qqy7qUuU39S2APSS+VtDVwJDC3ocxc4Jg8fATws4iIzoVpZh3i+mxWQ22vjCNis6TjgeuAMcBFEXGXpNOBhRExF7gQuFTSCmAdqYKbWc24PpvVU6lnxhExD5jXMO7UwvAfgPd0NrSOGvZb4/1U9/jAMXZK5TGOsPpc+fbqIK9LPdViXeS7T2ZmZtVyc5hmZmYVG1XJWNJFktZIurMw7oWSrpf0n/nfnWsY42mSHpS0JH8OqzjGqZLmS7pb0l2STsjja7Et+4ivNttR0raSbpN0R47x83n8S3MTkytyk5NbVxVj3YyE+lvWSKjnZdT9WNAfdT9ujKrb1JL+EtgEXBIRr8zjvgKsi4gzcju8O0fEp2sW42nApoj4alVxFUmaBEyKiMWSdgAWAe8CZlGDbdlHfO+lJtsxt1i1XURskjQOuBk4AfifwNUR8X1J5wF3RMS5VcZaFyOh/pY1Eup5GXU/FvRH3Y8bo+rKOCJuJL39WVRs2u9i0savTIsYayUiVkfE4jy8EVhOapWpFtuyj/hqI5JN+eu4/AngzaQmJqEG+2OdjIT6W9ZIqOdl1P1Y0B91P26MqmTcwosjYnUe/h3w4iqD6cPxkpbm21u1ueWj1GPPvsCt1HBbNsQHNdqOksZIWgKsAa4HfgM8lpuYhOZNUdrz1W6fG6Ta7J/9VfdjQX/U8bjRDcn4Wbnhgjrelz8XeDmwD7Aa+Kdqw0kkbQ9cBZwYERuK0+qwLZvEV6vtGBFPR8Q+pFau9gf+rMp4Rro67HODVKv9sz/qfizoj7oeN7ohGT+UnxX0PjNYU3E8W4iIh/KB+xngAtKBu1L5OedVwGURcXUeXZtt2Sy+Om5HgIh4DJgPvB4Yr9TEJDRvitKerzb73GDVdf9sp+7Hgv6o83GjG5JxsWm/Y4B/rTCWpnp36uxvgDtblR0O+eWjC4HlEfG1wqRabMtW8dVpO0qaKGl8Hn4B8FbSM6r5pCYmoab7Y83UYp/rhDrtn2XV/VjQH3U/boy2t6kvB2aQeuF4CPgccA3wA2AacD/w3oio7MWKFjHOIN0iCeA+4KOF5zHDTtJfADcBy4Bn8ujPkJ6vVL4t+4jvKGqyHSW9mvRiyxjSSe8PIuJ0SS8j9SH8QuB24AMR8WQVMdbNSKi/ZY2Eel5G3Y8F/VH348aoSsZmZmYjUTfcpjYzM6s1J2MzM7OKORmbmZlVzMnYzMysYk7GZmZmFXMyrjlJP5H0mKQfN4y/qdDLyCpJ17T4/TRJP5W0PPdW0pPHv1nSYkl3Srq4tyEKJWcr9Sq0VNJ+A4x7lqRdW0yb0bg+A1zGHEm/LWyHfVqUe7pQZm5h/FvyNlgi6WZJu+fxu0m6Ia//AklTBhurGQy+PueyO0paKemcwrijJC3L++xPJE3I4zvSI5GkEyX9SYtps4qxDFSua/cUYn1RkzK7KPW8tKlh/Xco/G6JpIclfT1P20aph7QVSj2m9Qw21qHgZFx/ZwIfbBwZEQdGxD65ucVfAFdv8cvkEuDMiPhzUssyayRtRfob2CNzjzL389wf8B8K7JE/s0lNxQ3ELKBpMu6wT/Vuh4hY0qLME4Uy7yyMPxd4f96G3wM+m8d/ldTbzquB04EvDVn01m0GW58BvgDc2Psln0h/Azgo77NLgeML5c8q7P/zBhj3iUDTZNxh7y/E2qxVrz8A/wc4qTgyIjYWfrcP6ZjWuw0/DDwaEbsDZwFfHsL4B6zrkrGkHkm/yldVv5Z0maS/kvRzpb4598/ltlNqNPw2SbdLmln4/U35imqxpDfk8TPymd2Vef6XSdJg442IG4CNfazPjqSegLY4k5a0FzA2Iq7P89oUEY8DuwBPRcSvc9HrgcPz8ExSIoqIuIXUfOOkxnkXljEmb8s785n5P0g6ApgOXJbPUl8g6ZC8XRYD7+7vdhgiAeyYh3cCVuXhvYCf5eH5pG1iNdRN9TlPfw2pU4afFkfnz3Y5xh15bl/ul7ydrlXqh/tOSf9D0idIJ9bzJc3P5Y7N2/s24I0DWdZARMTvI+JmUlJuStKewItIDXzA83uYuhJ4Syf+LzsuIrrqA/QAm4FXkU5GFgEXkXbmmcA1udw/klpHAhgP/BrYjnR2uG0evwewMA/PANaT2hveinR2+xdNlv8pYEmTz9l9xDwD+HGLaUcDV7aY9i7gx6QzxNtJZ+Vj8rreD0zP5b4BLMvDPy7GDdzQW67FMl4DXF/4Pj7/u6Aw/22BB/L2Eqnlni3WB/jTFttmSe98G8rPAe4hXQmcBWzTIsbNwELgFuBdhfEHAo+Qek+6G9gxj/8ecEIefjcpae9S9b7rT9P/226qz1vlejWFdOfpnMK0I4ANpI4ObgTG5PGnkVqVWpq3y85ttufhwAWF7zvlf+8DJuThScB/AROBrYGfF2Mp/PagFtvmP1osewGpdawlpKtf9RHnrGbLzNNOBb5a+H4nMKXw/Te961KnT9ddGWe/jYhlkRoGvwu4IdL/0jJS5QY4GDhZqQu8BaSEMo3UL+0FkpYBPyRdRfW6LSJW5vkuKczrWRFxZhRupxQ+nxjguhwFXN5i2lhSwjkJeC3wMmBWXtcjgbPyme1G4OkBLv9e4GWSvinpENIBodGfkbb5f+Zlf7fZjCLinhbbZp9InS00OiXP+7Wk5iVbdW6+W0RMB94HfF3Sy/P4fwAOi4gpwHeA3vZqTwLeJOl24E2kzhwGun1s6HVLff57YF5ErCyOVOr84GOkLgF3JSXeU/Lk/vZItAx4q6QvSzowItY3KfM6YEFErI2Ip4Arms0oIua32DZvaLHs90fEq0jHrANpcju/pCNpvQ1ra2z7IqNSsS3gZwrfn+G5bSLg8Ii4p/hDSaeR2prdm3SmWrxdUpzv0zTZvpI+Bby/SUw39rcCK72ksT+pcfNmVgJLIuLeXP4a4ADgwoj4BWmHR9LBwJ75Nw8CUwvz6LNnoYh4VNLewF8Dfwe8F/hQf9ajsD5/SouKDcxoTMjxXPuxT0r6Dg3PkQrlHsz/3itpAbCvpA3A3hHR25/pFcBPcrlV5FvpSt2tHd7iZMDqoVvq8+uBAyX9PbA9sLWkTaReiIiI3+T5/AA4OY97qDD/C0h3vlqKiF8rvbR5GPBFSTdExOn9WY/C8g4i3bFq9HizhFyopxslfY+0LS7p5zL3Jj2aW1QY3XtMW5mfr+9EuiNWK92ajMu4Dvi4pI9HREjaNyJuJ/1HroyIZyQdQ7rtW1pEnEm6XdwJR5Bud7V6fvJL0jPfiRGxlvQsaiGApBdFxBpJ25CuKP9v/s1cUkfb3yedAa/vTXqSfhURz+uTNx9AnoqIqyTdw3NXvRuBHfLwr4AeSS/PB4yjmgWbD5RN34huRtKkiFidn/+8iya9rSh1FP54RDyZY30j8BXgUWAnSXtGenbe26tS7zqty1dEp5Bu79nINuLrc0Q8m/QlzSI9BjpZ6a8W9irU8+K+PKlw0vpsj0SSJpPeDXlLcRl5Xusi4ruSHgM+kif11ueHSZ1EfEPSLqQ7Ye8B7mgS73xK1uecJMdHxMP5Sv/twL+X+W2DZncWenuY+gVpG/8s3zmpFSfj1r4AfB1YqvT28W9JO8g/A1dJOpp0JfX7oQxC0k2kW7HbS1oJfDgirsuTjwTOaCg/Hfi7iPhIRDwt6STghpywFpH66wT4lKS3k64Gzo2I3heW5pHOilcAjwPH5vlOIF1dNJoMfCdvI3ju9tgc4DxJT5DO6GcD10p6nPRixQ6NMxqAyyRNzHEtIV2ZP28bAH8OfFvSM3ldz4iIu3O5vyX9Xz5DSs69V/QzgC9JCtLzt+M6EKtVa8TX51bzjIhVkj4P3Cjpj6T3QWblyV9R+pO/Z3skyuMnkZ61N3oVcGauE38k3f4GOB/4iaRVEXFQvqPwC+AxUt0brG2A63IiHkNKxBcASHon6cTj1Pz9PtJLaltLehdwcG+dJt2Za/zzrQuBSyWtANaRtnPtuNcmKyUn7pdFxNlVx2JmgyPpeOC/ImJu28I2LJyMzczMKtatb1ObmZnVhpOxmZlZxZyMzczMKuZkbGZmVjEnYzMzs4o5GZuZmVXMydjMzKxi/w26MBbtRJSoSwAAAABJRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 576x216 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 576x216 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 576x216 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "%%time\n",
    "plt.figure(figsize=(8,3))\n",
    "tmp = test_mc_estimators(trained_model, y, 1, N_trials=15);\n",
    "trained_model_one_sample = tmp[-1]\n",
    "\n",
    "plt.figure(figsize=(8,3))\n",
    "tmp = test_mc_estimators(trained_model, y, 10, N_trials=15);\n",
    "\n",
    "plt.figure(figsize=(8,3))\n",
    "tmp = test_mc_estimators(trained_model, y, 50, N_trials=15);\n",
    "plt.savefig('entropy.jpg')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### This last, best estimator we can treat as our \"ground truth\" for purposes of the next section."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(-333.6640)"
      ]
     },
     "execution_count": 21,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "trained_model_gt = tmp[-1]\n",
    "trained_model_gt.log_prob(1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Normal(loc: 13.177141189575195, scale: 0.47150295972824097)"
      ]
     },
     "execution_count": 22,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "tmp[-1]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Let's also run this experiment again, but on a completely random (untrained) model."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.figure(figsize=(8,3))\n",
    "tmp = test_mc_estimators(random_model, y, 1, N_trials=15);\n",
    "random_model_one_sample = tmp[-1]\n",
    "\n",
    "plt.figure(figsize=(8,3))\n",
    "tmp = test_mc_estimators(random_model, y, 10, N_trials=15);\n",
    "\n",
    "plt.figure(figsize=(8,3))\n",
    "tmp = test_mc_estimators(random_model, y, 50, N_trials=15);\n",
    "\n",
    "random_model_gt = tmp[-1]\n",
    "random_model_gt"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Other approximations which allow gradient computation\n",
    "\n",
    "Possibilities:\n",
    "\n",
    "1. Deterministic greedy decoding by using the max, instead of sampling\n",
    "2. Straight-through estimator, taking a mean of the embeddings\n",
    "\n",
    "\n",
    "## Greedy decoding"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "_, probs, dists = sample_from_model(trained_model, torch.FloatTensor(y), 1, greedy=True)\n",
    "est_A_greedy = -probs.log().sum()\n",
    "est_B_greedy = -((dists + 1e-45).log()*dists).sum(-1).sum()\n",
    "\n",
    "domain = torch.linspace(trained_model_one_sample.loc.item() - 5*trained_model_one_sample.scale.item(),\n",
    "                        trained_model_one_sample.loc.item() + 5*trained_model_one_sample.scale.item(),\n",
    "                        200)\n",
    "\n",
    "plt.title(\"test on trained model\")\n",
    "plt.plot(domain, trained_model_gt.log_prob(domain).exp());\n",
    "plt.plot(domain, trained_model_one_sample.log_prob(domain).exp());\n",
    "height = plt.ylim()[1] * 0.25\n",
    "plt.plot([est_A_greedy, est_A_greedy], [0.0, height], '-')\n",
    "plt.plot([est_B_greedy, est_B_greedy], [0.0, height], '-')\n",
    "plt.legend(['MC 50 samples', 'MC one sample', 'greedy A', 'greedy B']);\n",
    "plt.xlabel(\"entropy estimate\");\n",
    "\n",
    "trained_greedy = est_B_greedy # save for later"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "_, probs, dists = sample_from_model(random_model, torch.FloatTensor(y), 1, greedy=True)\n",
    "est_A_greedy = -probs.log().sum()\n",
    "est_B_greedy = -(dists.log()*dists).sum(-1).sum()\n",
    "\n",
    "domain = torch.linspace(random_model_one_sample.loc.item() - 5*random_model_one_sample.scale.item(),\n",
    "                        random_model_one_sample.loc.item() + 5*random_model_one_sample.scale.item(),\n",
    "                        200)\n",
    "\n",
    "plt.title(\"test on random model\")\n",
    "plt.plot(domain, random_model_gt.log_prob(domain).exp());\n",
    "plt.plot(domain, random_model_one_sample.log_prob(domain).exp());\n",
    "height = plt.ylim()[1] * 0.25\n",
    "plt.plot([est_A_greedy, est_A_greedy], [0.0, height], '-')\n",
    "plt.plot([est_B_greedy, est_B_greedy], [0.0, height], '-')\n",
    "plt.legend(['MC 50 samples', 'MC one sample', 'greedy A', 'greedy B']);\n",
    "plt.xlabel(\"entropy estimate\");\n",
    "\n",
    "random_greedy = est_B_greedy # save for later"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Estimator A is pretty bad in this context. Let's look at B alone…"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "plt.title(\"test on random model, est B only\")\n",
    "plt.plot(domain, random_model_gt.log_prob(domain).exp());\n",
    "plt.plot(domain, random_model_one_sample.log_prob(domain).exp());\n",
    "height = plt.ylim()[1] * 0.25\n",
    "plt.plot([est_B_greedy, est_B_greedy], [0.0, height], '-')\n",
    "plt.legend(['MC 50 samples', 'MC one sample', 'greedy B']);\n",
    "plt.xlabel(\"entropy estimate\");"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "That's probably fine, or at least good-enough to use as a regularizer during early stages of training (i.e. when the network is random-ish).\n",
    "\n",
    "Greedy decoding for the values we condition on seems to work reasonably well for approximating the entropy.\n",
    "It might be slow-ish though, since it does require unrolling the RNN.\n",
    "\n",
    "## Straight-through estimator\n",
    "\n",
    "For the straight-through estimator, instead of evaluating the RNN on the embedding of a single input, we compute the mean of the embeddings under the distribution. This is the \"wrong\" thing to do, but has precedent."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# quick check: how to use the embedding layers directly\n",
    "\n",
    "inp = torch.nn.functional.one_hot(rnn_start_token_vector(10, 'cpu'), trained_model.input_size).float()\n",
    "v1 = inp @ trained_model.encoder.weight\n",
    "v2 = trained_model.encoder(rnn_start_token_vector(10, 'cpu'))\n",
    "assert (v1 == v2).float().mean().item() == 1.0"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def straight_through_estimate(model, y, max_len=100):\n",
    "    device = y.device\n",
    "    model.eval()\n",
    "    \n",
    "    # hold the output here\n",
    "    H_t = torch.zeros((max_len, )).to(device)\n",
    "\n",
    "    # initialization for LSTM sampling\n",
    "    hidden = model.init_hidden(1, device)\n",
    "    inp = torch.nn.functional.one_hot(rnn_start_token_vector(1, device), model.input_size).float()\n",
    "    \n",
    "    # run character-by-character...\n",
    "    for char in range(max_len):\n",
    "        # one step forward\n",
    "        # this requires reproducing the `forward` function of the model itself\n",
    "        embeds = inp @ model.encoder.weight\n",
    "        output, hidden = model.rnn(torch.cat((embeds, y.expand(1, -1).unsqueeze(1).expand(-1,1,-1)), -1), hidden)\n",
    "        output = model.decoder(output)\n",
    "        prob = torch.softmax(output, dim=2)\n",
    "        H_t[char] = -(prob.log() * prob).sum()\n",
    "        inp = prob # send probabilities instead of selecting next action\n",
    "\n",
    "    return H_t"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "with torch.no_grad():\n",
    "    H_est_s_trained = straight_through_estimate(trained_model, torch.FloatTensor(y)).sum()\n",
    "    H_est_s_random = straight_through_estimate(random_model, torch.FloatTensor(y)).sum()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "domain = torch.linspace(trained_model_one_sample.loc.item() - 5*trained_model_one_sample.scale.item(),\n",
    "                        trained_model_one_sample.loc.item() + 5*trained_model_one_sample.scale.item(),\n",
    "                        200)\n",
    "\n",
    "plt.title(\"test on trained model\")\n",
    "plt.plot(domain, trained_model_gt.log_prob(domain).exp());\n",
    "plt.plot(domain, trained_model_one_sample.log_prob(domain).exp());\n",
    "height = plt.ylim()[1] * 0.25\n",
    "plt.plot([trained_greedy, trained_greedy], [0.0, height], '-')\n",
    "plt.plot([H_est_s_trained, H_est_s_trained], [0.0, height], '-')\n",
    "plt.legend(['MC 50 samples', 'MC one sample', 'greedy', 'straight-through']);\n",
    "plt.xlabel(\"entropy estimate\");"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "trained_model_one_sample.loc.item() "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "domain = torch.linspace(random_model_one_sample.loc.item() - 5*random_model_one_sample.scale.item(),\n",
    "                        random_model_one_sample.loc.item() + 5*random_model_one_sample.scale.item(),\n",
    "                        200)\n",
    "\n",
    "plt.title(\"test on random model\")\n",
    "plt.plot(domain, random_model_gt.log_prob(domain).exp());derivative\n",
    "plt.plot(domain, random_model_one_sample.log_prob(domain).exp())\n",
    "height = plt.ylim()[1] * 0.25\n",
    "plt.plot([random_greedy, random_greedy], [0.0, height], '-')\n",
    "plt.plot([H_est_s_random, H_est_s_random], [0.0, height], '-')\n",
    "plt.legend(['MC 50 samples', 'MC one sample', 'greedy', 'straight-through']);\n",
    "plt.xlabel(\"entropy estimate\");"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Looks like the straight-through estimator is better on the random model, and the greedy estimator is better on the fully trained model!"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def greedy_entropy_est(model, y, max_len=100):\n",
    "    \"\"\" Quick re-implementation of greedy entropy estimation \"\"\"\n",
    "    H_est = 0.0\n",
    "    device = y.device\n",
    "\n",
    "    # initialization for LSTM sampling\n",
    "    hidden = model.init_hidden(1, device)\n",
    "    inp = rnn_start_token_vector(1, device)\n",
    "    \n",
    "    # run character-by-character...\n",
    "    for char in range(max_len):\n",
    "        # one step forward\n",
    "        output, hidden = model(inp, y.expand(1, -1), hidden)\n",
    "        action = output.argmax(dim=2)\n",
    "\n",
    "        log_prob = output - torch.logsumexp(output, dim=-1, keepdim=True)\n",
    "        H_est += -(log_prob.exp() * log_prob).sum()\n",
    "        inp = action\n",
    "\n",
    "    return H_est\n",
    "\n",
    "\n",
    "print(trained_greedy.item(), greedy_entropy_est(trained_model, torch.FloatTensor(y)))\n",
    "print(random_greedy.item(), greedy_entropy_est(random_model, torch.FloatTensor(y)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "max_len"
   ]
  },
  {
   "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.6.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
