{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "# MISC\n",
    "import os\n",
    "import sys\n",
    "\n",
    "sys.path.append(\"..\")\n",
    "os.environ['MUJOCO_GL']='egl'\n",
    "os.environ['CUDA_VISIBLE_DEVICES']='0'\n",
    "\n",
    "import shutup\n",
    "shutup.please()\n",
    "\n",
    "import jax\n",
    "import jax.numpy as jnp\n",
    "\n",
    "import numpy as np\n",
    "from functools import partial\n",
    "\n",
    "# VIS\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt\n",
    "import mediapy\n",
    "from rich.pretty import pprint\n",
    "from tqdm.auto import tqdm\n",
    "\n",
    "from hydra import initialize, compose\n",
    "from omegaconf import OmegaConf\n",
    "\n",
    "plt.style.use(['seaborn-v0_8-colorblind', 'seaborn-v0_8-notebook'])\n",
    "colors = plt.rcParams['axes.prop_cycle'].by_key()['color'] \n",
    "\n",
    "GLOBAL_KEY = jax.random.key(42)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Data Collection"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<table class=\"show_images\" style=\"border-spacing:0px;\"><tr><td style=\"padding:1px;\"><img width=\"256\" height=\"256\" style=\"image-rendering:auto; object-fit:cover;\" src=\"data:image/png;base64,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\"/></td><td style=\"padding:1px;\"><img width=\"256\" height=\"256\" style=\"image-rendering:auto; object-fit:cover;\" src=\"data:image/png;base64,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\"/></td><td style=\"padding:1px;\"><img width=\"256\" height=\"256\" style=\"image-rendering:auto; object-fit:cover;\" src=\"data:image/png;base64,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\"/></td><td style=\"padding:1px;\"><img width=\"256\" height=\"256\" style=\"image-rendering:auto; object-fit:cover;\" src=\"data:image/png;base64,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\"/></td></tr></table>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from envs.minigrid.doors_grid import DynamicsGeneralization_Doors, MinigridWrapper\n",
    "\n",
    "# Train dynamics\n",
    "layouts = []\n",
    "for i in [0, 1, 2, 3]:\n",
    "    env = DynamicsGeneralization_Doors(render_mode=\"rgb_array\", highlight=False, max_steps=100, task_num=0)\n",
    "    env._gen_grid = partial(env._gen_grid, layout_type=i)\n",
    "    env = MinigridWrapper(env)\n",
    "    env.reset()\n",
    "    \n",
    "    layouts.append(env.render())\n",
    "    \n",
    "mediapy.show_images(layouts, width=256, height=256)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "b787d00b803f4486bd1ca8e0aa5813fb",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/6 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\"><span style=\"font-weight: bold\">{</span>\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'actions'</span>: <span style=\"font-weight: bold\">(</span><span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">196000</span>,<span style=\"font-weight: bold\">)</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'layout_type'</span>: <span style=\"font-weight: bold\">(</span><span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">196000</span>, <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">6</span><span style=\"font-weight: bold\">)</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'next_observations'</span>: <span style=\"font-weight: bold\">(</span><span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">196000</span>, <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">2</span><span style=\"font-weight: bold\">)</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'observations'</span>: <span style=\"font-weight: bold\">(</span><span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">196000</span>, <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">2</span><span style=\"font-weight: bold\">)</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'terminals'</span>: <span style=\"font-weight: bold\">(</span><span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">196000</span>,<span style=\"font-weight: bold\">)</span>\n",
       "<span style=\"font-weight: bold\">}</span>\n",
       "</pre>\n"
      ],
      "text/plain": [
       "\u001b[1m{\u001b[0m\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'actions'\u001b[0m: \u001b[1m(\u001b[0m\u001b[1;36m196000\u001b[0m,\u001b[1m)\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'layout_type'\u001b[0m: \u001b[1m(\u001b[0m\u001b[1;36m196000\u001b[0m, \u001b[1;36m6\u001b[0m\u001b[1m)\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'next_observations'\u001b[0m: \u001b[1m(\u001b[0m\u001b[1;36m196000\u001b[0m, \u001b[1;36m2\u001b[0m\u001b[1m)\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'observations'\u001b[0m: \u001b[1m(\u001b[0m\u001b[1;36m196000\u001b[0m, \u001b[1;36m2\u001b[0m\u001b[1m)\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'terminals'\u001b[0m: \u001b[1m(\u001b[0m\u001b[1;36m196000\u001b[0m,\u001b[1m)\u001b[0m\n",
       "\u001b[1m}\u001b[0m\n"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 800x550 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from envs.minigrid.env_utils import random_exploration\n",
    "\n",
    "train_layout_data = []\n",
    "\n",
    "NUM_TRAIN_LAYOUTS = 6\n",
    "NUM_TRAIN_STEPS = 99\n",
    "NUM_TRAIN_EPISODES = 2000\n",
    "\n",
    "for i in tqdm(range(NUM_TRAIN_LAYOUTS)):\n",
    "    env = DynamicsGeneralization_Doors(render_mode=\"rgb_array\", highlight=False, max_steps=NUM_TRAIN_STEPS, agent_start_pos=None) # comment agent_start_pos for same start pos\n",
    "    env._gen_grid = partial(env._gen_grid, layout_type=i)\n",
    "    env = MinigridWrapper(env)\n",
    "    dataset, env = random_exploration(env, num_episodes=NUM_TRAIN_EPISODES, layout_type=i, num_mdp=NUM_TRAIN_LAYOUTS)\n",
    "    train_layout_data.append(dataset)\n",
    "\n",
    "pprint(jax.tree.map(lambda x: x.shape, train_layout_data[0]))\n",
    "\n",
    "coverage_map = np.zeros(shape=(env.env.unwrapped.width, env.env.unwrapped.height))\n",
    "for layout in train_layout_data:\n",
    "    for obs in layout['observations']:\n",
    "        obs = obs.astype(np.int16)\n",
    "        coverage_map[obs[1], obs[0]] += 1\n",
    "        \n",
    "plt.imshow(coverage_map, cmap='inferno', vmin=0)\n",
    "plt.colorbar()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Init FB"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\"><span style=\"font-weight: bold\">{</span>\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'seed'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">42</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'run_group'</span>: <span style=\"color: #008000; text-decoration-color: #008000\">'fb'</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'save_dir'</span>: <span style=\"color: #008000; text-decoration-color: #008000\">'experiment_logs/'</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'train_steps'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">150000</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'log_interval'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">10000</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'eval_interval'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">30000</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'eval_tasks'</span>: <span style=\"color: #800080; text-decoration-color: #800080; font-style: italic\">None</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'eval_episodes'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">20</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'eval_temperature'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">0</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'eval_gaussian'</span>: <span style=\"color: #800080; text-decoration-color: #800080; font-style: italic\">None</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'video_episodes'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">1</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'video_frame_skip'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">3</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'eval_on_cpu'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">0</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'frame_stack'</span>: <span style=\"color: #800080; text-decoration-color: #800080; font-style: italic\">None</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'p_aug'</span>: <span style=\"color: #800080; text-decoration-color: #800080; font-style: italic\">None</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'agent'</span>: <span style=\"font-weight: bold\">{</span>\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'agent_name'</span>: <span style=\"color: #008000; text-decoration-color: #008000\">'fb'</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'lr'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">0.0003</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'batch_size'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">1024</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'discrete'</span>: <span style=\"color: #00ff00; text-decoration-color: #00ff00; font-style: italic\">True</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'discount'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">0.99</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'tau'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">0.01</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'clip_by_global_norm'</span>: <span style=\"color: #00ff00; text-decoration-color: #00ff00; font-style: italic\">True</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'dataset_class'</span>: <span style=\"color: #008000; text-decoration-color: #008000\">'GCDataset'</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'value_p_curgoal'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">0.0</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'value_p_trajgoal'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">1.0</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'value_p_randomgoal'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">0.0</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'value_geom_sample'</span>: <span style=\"color: #00ff00; text-decoration-color: #00ff00; font-style: italic\">True</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'actor_p_curgoal'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">0.0</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'actor_p_trajgoal'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">0.0</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'actor_p_randomgoal'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">1.0</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'actor_geom_sample'</span>: <span style=\"color: #ff0000; text-decoration-color: #ff0000; font-style: italic\">False</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'gc_negative'</span>: <span style=\"color: #00ff00; text-decoration-color: #00ff00; font-style: italic\">True</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'p_aug'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">0.0</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'frame_stack'</span>: <span style=\"color: #800080; text-decoration-color: #800080; font-style: italic\">None</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'z_dim'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">100</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'z_mix_ratio'</span>: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">0.5</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'boltzmann'</span>: <span style=\"color: #00ff00; text-decoration-color: #00ff00; font-style: italic\">True</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'f_hidden_dims'</span>: <span style=\"font-weight: bold\">[</span><span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">1024</span>, <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">512</span>, <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">512</span><span style=\"font-weight: bold\">]</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'f_layer_norm'</span>: <span style=\"color: #00ff00; text-decoration-color: #00ff00; font-style: italic\">True</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'b_hidden_dims'</span>: <span style=\"font-weight: bold\">[</span><span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">512</span>, <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">512</span>, <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">512</span><span style=\"font-weight: bold\">]</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'b_layer_norm'</span>: <span style=\"color: #00ff00; text-decoration-color: #00ff00; font-style: italic\">True</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   │   </span><span style=\"color: #008000; text-decoration-color: #008000\">'use_context'</span>: <span style=\"color: #ff0000; text-decoration-color: #ff0000; font-style: italic\">False</span>\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"font-weight: bold\">}</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'env'</span>: <span style=\"font-weight: bold\">{</span><span style=\"color: #008000; text-decoration-color: #008000\">'env_name'</span>: <span style=\"color: #008000; text-decoration-color: #008000\">'fourrooms-vanilla'</span>, <span style=\"color: #008000; text-decoration-color: #008000\">'discrete'</span>: <span style=\"color: #00ff00; text-decoration-color: #00ff00; font-style: italic\">True</span><span style=\"font-weight: bold\">}</span>,\n",
       "<span style=\"color: #7fbf7f; text-decoration-color: #7fbf7f\">│   </span><span style=\"color: #008000; text-decoration-color: #008000\">'tags'</span>: <span style=\"font-weight: bold\">[</span><span style=\"color: #008000; text-decoration-color: #008000\">'vanilla-fb'</span>, <span style=\"color: #008000; text-decoration-color: #008000\">'4rooms'</span>, <span style=\"color: #008000; text-decoration-color: #008000\">'discrete'</span><span style=\"font-weight: bold\">]</span>\n",
       "<span style=\"font-weight: bold\">}</span>\n",
       "</pre>\n"
      ],
      "text/plain": [
       "\u001b[1m{\u001b[0m\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'seed'\u001b[0m: \u001b[1;36m42\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'run_group'\u001b[0m: \u001b[32m'fb'\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'save_dir'\u001b[0m: \u001b[32m'experiment_logs/'\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'train_steps'\u001b[0m: \u001b[1;36m150000\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'log_interval'\u001b[0m: \u001b[1;36m10000\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'eval_interval'\u001b[0m: \u001b[1;36m30000\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'eval_tasks'\u001b[0m: \u001b[3;35mNone\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'eval_episodes'\u001b[0m: \u001b[1;36m20\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'eval_temperature'\u001b[0m: \u001b[1;36m0\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'eval_gaussian'\u001b[0m: \u001b[3;35mNone\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'video_episodes'\u001b[0m: \u001b[1;36m1\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'video_frame_skip'\u001b[0m: \u001b[1;36m3\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'eval_on_cpu'\u001b[0m: \u001b[1;36m0\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'frame_stack'\u001b[0m: \u001b[3;35mNone\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'p_aug'\u001b[0m: \u001b[3;35mNone\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'agent'\u001b[0m: \u001b[1m{\u001b[0m\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'agent_name'\u001b[0m: \u001b[32m'fb'\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'lr'\u001b[0m: \u001b[1;36m0.0003\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'batch_size'\u001b[0m: \u001b[1;36m1024\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'discrete'\u001b[0m: \u001b[3;92mTrue\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'discount'\u001b[0m: \u001b[1;36m0.99\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'tau'\u001b[0m: \u001b[1;36m0.01\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'clip_by_global_norm'\u001b[0m: \u001b[3;92mTrue\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'dataset_class'\u001b[0m: \u001b[32m'GCDataset'\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'value_p_curgoal'\u001b[0m: \u001b[1;36m0.0\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'value_p_trajgoal'\u001b[0m: \u001b[1;36m1.0\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'value_p_randomgoal'\u001b[0m: \u001b[1;36m0.0\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'value_geom_sample'\u001b[0m: \u001b[3;92mTrue\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'actor_p_curgoal'\u001b[0m: \u001b[1;36m0.0\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'actor_p_trajgoal'\u001b[0m: \u001b[1;36m0.0\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'actor_p_randomgoal'\u001b[0m: \u001b[1;36m1.0\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'actor_geom_sample'\u001b[0m: \u001b[3;91mFalse\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'gc_negative'\u001b[0m: \u001b[3;92mTrue\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'p_aug'\u001b[0m: \u001b[1;36m0.0\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'frame_stack'\u001b[0m: \u001b[3;35mNone\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'z_dim'\u001b[0m: \u001b[1;36m100\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'z_mix_ratio'\u001b[0m: \u001b[1;36m0.5\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'boltzmann'\u001b[0m: \u001b[3;92mTrue\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'f_hidden_dims'\u001b[0m: \u001b[1m[\u001b[0m\u001b[1;36m1024\u001b[0m, \u001b[1;36m512\u001b[0m, \u001b[1;36m512\u001b[0m\u001b[1m]\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'f_layer_norm'\u001b[0m: \u001b[3;92mTrue\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'b_hidden_dims'\u001b[0m: \u001b[1m[\u001b[0m\u001b[1;36m512\u001b[0m, \u001b[1;36m512\u001b[0m, \u001b[1;36m512\u001b[0m\u001b[1m]\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'b_layer_norm'\u001b[0m: \u001b[3;92mTrue\u001b[0m,\n",
       "\u001b[2;32m│   │   \u001b[0m\u001b[32m'use_context'\u001b[0m: \u001b[3;91mFalse\u001b[0m\n",
       "\u001b[2;32m│   \u001b[0m\u001b[1m}\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'env'\u001b[0m: \u001b[1m{\u001b[0m\u001b[32m'env_name'\u001b[0m: \u001b[32m'fourrooms-vanilla'\u001b[0m, \u001b[32m'discrete'\u001b[0m: \u001b[3;92mTrue\u001b[0m\u001b[1m}\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'tags'\u001b[0m: \u001b[1m[\u001b[0m\u001b[32m'vanilla-fb'\u001b[0m, \u001b[32m'4rooms'\u001b[0m, \u001b[32m'discrete'\u001b[0m\u001b[1m]\u001b[0m\n",
       "\u001b[1m}\u001b[0m\n"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{'actions': (1176000,), 'layout_type': (1176000, 6), 'next_observations': (1176000, 2), 'observations': (1176000, 2), 'terminals': (1176000,)}\n"
     ]
    }
   ],
   "source": [
    "import functools\n",
    "from utils.datasets import Dataset, GCDataset\n",
    "\n",
    "with initialize(version_base=None, config_path=\"../configs/\"):\n",
    "    fb_config = compose(config_name='entry.yaml', overrides=['experiment=fb_vanilla_discrete_4rooms.yaml'])\n",
    "    fb_config = OmegaConf.to_container(fb_config, resolve=True)\n",
    "    pprint(fb_config)\n",
    "\n",
    "def concatenate_dicts(dict1, dict2):\n",
    "    return jax.tree.map(lambda x, y: jnp.concatenate([x, y]), dict1, dict2)\n",
    "\n",
    "whole_data = functools.reduce(concatenate_dicts, train_layout_data)\n",
    "\n",
    "print(jax.tree.map(lambda x: x.shape, whole_data))\n",
    "whole_dataset = Dataset.create(**jax.device_get(whole_data))\n",
    "gc_whole_dataset = GCDataset(whole_dataset, config=fb_config['agent'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "sys.path.append(\"..\")\n",
    "from agents.fb import ForwardBackwardAgent\n",
    "\n",
    "gc_whole_dataset = GCDataset(whole_dataset, config=fb_config['agent'])\n",
    "example_batch = gc_whole_dataset.sample(1)\n",
    "fb_agent = ForwardBackwardAgent.create(\n",
    "    0,\n",
    "    example_batch['observations'],\n",
    "    np.full_like(example_batch['actions'], env.action_space.n - 1),\n",
    "    config=fb_config['agent']\n",
    ")\n",
    "fb_agent, info = fb_agent.update(gc_whole_dataset.sample(512))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Train & Plot Value/Policy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "ename": "AttributeError",
     "evalue": "'DynamicsGeneralization_Doors' object has no attribute 'env'",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mAttributeError\u001b[0m                            Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[6], line 70\u001b[0m\n\u001b[1;32m     63\u001b[0m     pred_policy_img \u001b[38;5;241m=\u001b[39m policy_image_grid(env, example_batch,\n\u001b[1;32m     64\u001b[0m                                                     action_fn\u001b[38;5;241m=\u001b[39mpartial(supply_rng(fb_agent\u001b[38;5;241m.\u001b[39msample_actions,\n\u001b[1;32m     65\u001b[0m                                                                                 rng\u001b[38;5;241m=\u001b[39mjax\u001b[38;5;241m.\u001b[39mrandom\u001b[38;5;241m.\u001b[39mPRNGKey(np\u001b[38;5;241m.\u001b[39mrandom\u001b[38;5;241m.\u001b[39mrandint(\u001b[38;5;241m0\u001b[39m, \u001b[38;5;241m2\u001b[39m\u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39m\u001b[38;5;241m32\u001b[39m))), latent_z\u001b[38;5;241m=\u001b[39mlatent_z,\n\u001b[1;32m     66\u001b[0m                                                                 temperature\u001b[38;5;241m=\u001b[39m\u001b[38;5;241m0.0\u001b[39m),\n\u001b[1;32m     67\u001b[0m                                                     goal\u001b[38;5;241m=\u001b[39mgoal)\n\u001b[1;32m     68\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m pred_policy_img\n\u001b[0;32m---> 70\u001b[0m pred_policy_img \u001b[38;5;241m=\u001b[39m \u001b[43mvisualize_policy_image\u001b[49m\u001b[43m(\u001b[49m\u001b[43mlayout_type\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m1\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mtask_num\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m0\u001b[39;49m\u001b[43m)\u001b[49m\n\u001b[1;32m     71\u001b[0m plt\u001b[38;5;241m.\u001b[39mimshow(pred_policy_img)\n\u001b[1;32m     72\u001b[0m plt\u001b[38;5;241m.\u001b[39mtight_layout()\n",
      "Cell \u001b[0;32mIn[6], line 63\u001b[0m, in \u001b[0;36mvisualize_policy_image\u001b[0;34m(layout_type, task_num)\u001b[0m\n\u001b[1;32m     61\u001b[0m latent_z \u001b[38;5;241m=\u001b[39m jax\u001b[38;5;241m.\u001b[39mdevice_get(fb_agent\u001b[38;5;241m.\u001b[39minfer_z(goal)[\u001b[38;5;28;01mNone\u001b[39;00m])\n\u001b[1;32m     62\u001b[0m example_batch \u001b[38;5;241m=\u001b[39m gc_whole_dataset\u001b[38;5;241m.\u001b[39msample(\u001b[38;5;241m1\u001b[39m)\n\u001b[0;32m---> 63\u001b[0m pred_policy_img \u001b[38;5;241m=\u001b[39m \u001b[43mpolicy_image_grid\u001b[49m\u001b[43m(\u001b[49m\u001b[43menv\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mexample_batch\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m     64\u001b[0m \u001b[43m                                                \u001b[49m\u001b[43maction_fn\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mpartial\u001b[49m\u001b[43m(\u001b[49m\u001b[43msupply_rng\u001b[49m\u001b[43m(\u001b[49m\u001b[43mfb_agent\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43msample_actions\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m     65\u001b[0m \u001b[43m                                                                            \u001b[49m\u001b[43mrng\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mjax\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mrandom\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mPRNGKey\u001b[49m\u001b[43m(\u001b[49m\u001b[43mnp\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mrandom\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mrandint\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m0\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m2\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m32\u001b[39;49m\u001b[43m)\u001b[49m\u001b[43m)\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mlatent_z\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mlatent_z\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m     66\u001b[0m \u001b[43m                                                            \u001b[49m\u001b[43mtemperature\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m0.0\u001b[39;49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m     67\u001b[0m \u001b[43m                                                \u001b[49m\u001b[43mgoal\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mgoal\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     68\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m pred_policy_img\n",
      "Cell \u001b[0;32mIn[6], line 13\u001b[0m, in \u001b[0;36mpolicy_image_grid\u001b[0;34m(env, dataset, action_fn, **kwargs)\u001b[0m\n\u001b[1;32m     11\u001b[0m fig \u001b[38;5;241m=\u001b[39m plt\u001b[38;5;241m.\u001b[39mfigure(tight_layout\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mTrue\u001b[39;00m)\n\u001b[1;32m     12\u001b[0m canvas \u001b[38;5;241m=\u001b[39m FigureCanvas(fig)\n\u001b[0;32m---> 13\u001b[0m \u001b[43mplot_policy\u001b[49m\u001b[43m(\u001b[49m\u001b[43menv\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mdataset\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mfig\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mfig\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43max\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mplt\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mgca\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43maction_fn\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43maction_fn\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     14\u001b[0m image \u001b[38;5;241m=\u001b[39m get_canvas_image(canvas)\n\u001b[1;32m     15\u001b[0m plt\u001b[38;5;241m.\u001b[39mclose(fig)\n",
      "Cell \u001b[0;32mIn[6], line 25\u001b[0m, in \u001b[0;36mplot_policy\u001b[0;34m(env, dataset, fig, ax, title, action_fn, **kwargs)\u001b[0m\n\u001b[1;32m     22\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m fig \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m \u001b[38;5;129;01mor\u001b[39;00m ax \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m:\n\u001b[1;32m     23\u001b[0m     fig, ax \u001b[38;5;241m=\u001b[39m plt\u001b[38;5;241m.\u001b[39msubplots()\n\u001b[0;32m---> 25\u001b[0m grid \u001b[38;5;241m=\u001b[39m \u001b[43menv\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mget_grid_array\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     26\u001b[0m ax \u001b[38;5;241m=\u001b[39m env\u001b[38;5;241m.\u001b[39mplot_grid(ax\u001b[38;5;241m=\u001b[39max, grid\u001b[38;5;241m=\u001b[39mgrid)\n\u001b[1;32m     28\u001b[0m goal \u001b[38;5;241m=\u001b[39m kwargs\u001b[38;5;241m.\u001b[39mget(\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mgoal\u001b[39m\u001b[38;5;124m'\u001b[39m, \u001b[38;5;28;01mNone\u001b[39;00m)\n",
      "File \u001b[0;32m~/ZeroShotRL/notebooks/../envs/minigrid/doors_grid.py:196\u001b[0m, in \u001b[0;36mMinigridWrapper.get_grid_array\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m    195\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m\u001b[38;5;250m \u001b[39m\u001b[38;5;21mget_grid_array\u001b[39m(\u001b[38;5;28mself\u001b[39m):\n\u001b[0;32m--> 196\u001b[0m     grid \u001b[38;5;241m=\u001b[39m (\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43menv\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43menv\u001b[49m\u001b[38;5;241m.\u001b[39munwrapped\u001b[38;5;241m.\u001b[39mgrid\u001b[38;5;241m.\u001b[39mencode()[:, :, \u001b[38;5;241m0\u001b[39m]\u001b[38;5;241m.\u001b[39mT)\u001b[38;5;241m.\u001b[39mastype(np\u001b[38;5;241m.\u001b[39mint16)\n\u001b[1;32m    197\u001b[0m     grid \u001b[38;5;241m=\u001b[39m np\u001b[38;5;241m.\u001b[39mwhere(grid \u001b[38;5;241m==\u001b[39m \u001b[38;5;241m2\u001b[39m, \u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1000\u001b[39m, grid)\n\u001b[1;32m    198\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m grid\n",
      "\u001b[0;31mAttributeError\u001b[0m: 'DynamicsGeneralization_Doors' object has no attribute 'env'"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAxYAAAIcCAYAAABrUjh1AAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjEsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvc2/+5QAAAAlwSFlzAAAPYQAAD2EBqD+naQAAIUBJREFUeJzt3W+MleWZ+PGLGRxGIyANyzCws8tK19pWBQSZHa2hNrOdRAPLi01ZNcAS/2ClxmWyW8E/TKlbYK01JBULUl37oi60jZqmEFw7Ldu1zoYUmMRW0CBaumZmhOzKALY7MPP8XhinPypazlwMHPTzSSbm3D7Pc65jbvF8PefMGVIURREAAAAJFWd6AAAA4OwnLAAAgDRhAQAApAkLAAAgTVgAAABpwgIAAEgTFgAAQJqwAAAA0oQFAACQJiwAAIC0ksPiZz/7WcycOTPGjRsXQ4YMiWeeeeaPnrN169a4/PLLY9iwYfHxj388nnjiiQGMCgAAlKuSw+LIkSMxadKkWLNmzUkd/9prr8V1110X11xzTbS3t8c//MM/xM033xzPPvtsycMCAADlaUhRFMWATx4yJJ5++umYPXv2+x5z1113xaZNm+KXv/xl/9rf/d3fxVtvvRVbtmwZ6F0DAABlZNA/Y9HW1haNjY3HrTU1NUVbW9tg3zUAAHCaDB3sO+js7Iyamprj1mpqaqK7uzt++9vfxrnnnntS1+no6IiOjo7+2319fVFRURETJ06MIUOGnNKZAQDgw6ooijh06FCMGzcuKipO3esMgx4Wp8q6deti+fLlZ3oMAAD4UPjNb34Tf/qnf3rKrjfoYTF27Njo6uo6bq2rqytGjBhx0q9WREQsXLgwZs2a1X/70KFD8dnPfjZ+85vfxIgRI07ZvAAA8GHW3d0ddXV1MXz48FN63UEPi4aGhti8efNxa88991w0NDSUdJ3a2tqora3tv93d3R0RESNGjBAWAABQolP9cYKS31R1+PDhaG9vj/b29oh459fJtre3x759+yIiYunSpTFv3rz+42+77bbYu3dvfPnLX47du3fHI488Et/73vdi8eLFp+YRAAAAZ1zJYfGLX/wipkyZElOmTImIiObm5pgyZUosW7YsIt75kPW7kRER8Rd/8RexadOmeO6552LSpEnxjW98I7797W9HU1PTKXoIAADAmZb6Hoszqbu7O0aOHBkHDx70VigAADhJg/U8etC/xwIAAPjwExYAAECasAAAANKEBQAAkCYsAACANGEBAACkCQsAACBNWAAAAGnCAgAASBMWAABAmrAAAADShAUAAJAmLAAAgDRhAQAApAkLAAAgTVgAAABpwgIAAEgTFgAAQJqwAAAA0oQFAACQJiwAAIA0YQEAAKQJCwAAIE1YAAAAacICAABIExYAAECasAAAANKEBQAAkCYsAACANGEBAACkCQsAACBNWAAAAGnCAgAASBMWAABAmrAAAADShAUAAJAmLAAAgDRhAQAApAkLAAAgTVgAAABpwgIAAEgTFgAAQJqwAAAA0oQFAACQJiwAAIA0YQEAAKQJCwAAIE1YAAAAacICAABIExYAAECasAAAANKEBQAAkCYsAACANGEBAACkCQsAACBNWAAAAGnCAgAASBMWAABAmrAAAADShAUAAJAmLAAAgDRhAQAApAkLAAAgTVgAAABpwgIAAEgTFgAAQJqwAAAA0oQFAACQJiwAAIA0YQEAAKQJCwAAIE1YAAAAacICAABIExYAAECasAAAANKEBQAAkCYsAACANGEBAACkCQsAACBNWAAAAGnCAgAASBMWAABAmrAAAADShAUAAJAmLAAAgDRhAQAApA0oLNasWRMTJkyI6urqqK+vj23btn3g8atXr45PfOITce6550ZdXV0sXrw4fve73w1oYAAAoPyUHBYbN26M5ubmaGlpiR07dsSkSZOiqakp3nzzzRMe/+STT8aSJUuipaUldu3aFY899lhs3Lgx7r777vTwAABAeSg5LB566KG45ZZbYsGCBfGpT30q1q5dG+edd148/vjjJzz+hRdeiKuuuipuuOGGmDBhQnz+85+P66+//o++ygEAAJw9SgqLnp6e2L59ezQ2Nv7+AhUV0djYGG1tbSc858orr4zt27f3h8TevXtj8+bNce211ybGBgAAysnQUg4+cOBA9Pb2Rk1NzXHrNTU1sXv37hOec8MNN8SBAwfiM5/5TBRFEceOHYvbbrut5LdCdXR0REdHR//tw4cPl3Q+AAAweAb9t0Jt3bo1VqxYEY888kjs2LEjnnrqqdi0aVPcf//9JV1n3bp1MXXq1P6fGTNmDNLEAABAqUp6xWL06NFRWVkZXV1dx613dXXF2LFjT3jOfffdF3Pnzo2bb745IiIuvfTSOHLkSNx6661xzz33REXFybXNwoULY9asWf23Dx8+LC4AAKBMlBQWVVVVMXXq1GhtbY3Zs2dHRERfX1+0trbGl770pROe8/bbb78nHiorKyMioiiKk77v2traqK2t7b/d3d1dyugAAMAgKiksIiKam5tj/vz5MW3atJg+fXqsXr06jhw5EgsWLIiIiHnz5sX48eNj5cqVERExc+bMeOihh2LKlClRX18fe/bsifvuuy9mzpzZHxgAAMDZreSwmDNnTuzfvz+WLVsWnZ2dMXny5NiyZUv/B7r37dt33CsU9957bwwZMiTuvffeeOONN+JP/uRPYubMmfG1r33t1D0KAADgjBpSlPJ+pDLS3d0dI0eOjIMHD8aIESPO9DgAAHBWGKzn0YP+W6EAAIAPP2EBAACkCQsAACBNWAAAAGnCAgAASBMWAABAmrAAAADShAUAAJAmLAAAgDRhAQAApAkLAAAgTVgAAABpwgIAAEgTFgAAQJqwAAAA0oQFAACQJiwAAIA0YQEAAKQJCwAAIE1YAAAAacICAABIExYAAECasAAAANKEBQAAkCYsAACANGEBAACkCQsAACBNWAAAAGnCAgAASBMWAABAmrAAAADShAUAAJAmLAAAgDRhAQAApAkLAAAgTVgAAABpwgIAAEgTFgAAQJqwAAAA0oQFAACQJiwAAIA0YQEAAKQJCwAAIE1YAAAAacICAABIExYAAECasAAAANKEBQAAkCYsAACANGEBAACkCQsAACBNWAAAAGnCAgAASBMWAABAmrAAAADShAUAAJAmLAAAgDRhAQAApAkLAAAgTVgAAABpwgIAAEgTFgAAQJqwAAAA0oQFAACQJiwAAIA0YQEAAKQJCwAAIE1YAAAAacICAABIExYAAECasAAAANKEBQAAkCYsAACANGEBAACkCQsAACBNWAAAAGnCAgAASBMWAABAmrAAAADShAUAAJAmLAAAgDRhAQAApAkLAAAgTVgAAABpwgIAAEgTFgAAQNqAwmLNmjUxYcKEqK6ujvr6+ti2bdsHHv/WW2/FokWLora2NoYNGxYXXXRRbN68eUADAwAA5WdoqSds3LgxmpubY+3atVFfXx+rV6+OpqamePnll2PMmDHvOb6npyf++q//OsaMGRM/+MEPYvz48fHrX/86LrjgglMxPwAAUAaGFEVRlHJCfX19XHHFFfHwww9HRERf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vvWt6Onpib/927897jotLS3xla98JTc9AABQFkr+Hoty4XssAACgdIP1PLrkb94GAAD4Q8ICAABIExYAAECasAAAANKEBQAAkCYsAACANGEBAACkCQsAACBNWAAAAGnCAgAASBMWAABAmrAAAADShAUAAJAmLAAAgDRhAQAApAkLAAAgTVgAAABpwgIAAEgTFgAAQJqwAAAA0oQFAACQJiwAAIA0YQEAAKQJCwAAIE1YAAAAacICAABIExYAAECasAAAANKEBQAAkCYsAACANGEBAACkCQsAACBNWAAAAGnCAgAASBMWAABAmrAAAADShAUAAJAmLAAAgDRhAQAApAkLAAAgTVgAAABpwgIAAEgTFgAAQJqwAAAA0oQFAACQJiwAAIA0YQEAAKQJCwAAIE1YAAAAacICAABIExYAAECasAAAANKEBQAAkCYsAACANGEBAACkCQsAACBNWAAAAGnCAgAASBMWAABAmrAAAADShAUAAJAmLAAAgDRhAQAApAkLAAAgTVgAAABpwgIAAEgTFgAAQJqwAAAA0oQFAACQJiwAAIA0YQEAAKQJCwAAIE1YAAAAacICAABIExYAAECasAAAANKEBQAAkCYsAACANGEBAACkCQsAACBNWAAAAGnCAgAASBMWAABAmrAAAADShAUAAJAmLAAAgLQBhcWaNWtiwoQJUV1dHfX19bFt27YPPP773/9+XHzxxVFdXR2XXnppbN68eUDDAgAA5anksNi4cWM0NzdHS0tL7NixIyZNmhRNTU3x5ptvnvD4F154Ia6//vq46aabYufOnTF79uyYPXt2/PKXv0wPDwAAlIchRVEUpZxQX18fV1xxRTz88MMREdHX1xd1dXVxxx13xJIlS95z/Jw5c+LIkSPxox/9qH/tr/7qr2Ly5Mmxdu3aAQ/e3d0dI0eOjIMHD8aIESMGfB0AAPgoGazn0UNLObinpye2b98eS5cu7V+rqKiIxsbGaGtrO+E5bW1t0dzcfNxaU1NTPPPMMyUN2tHRER0dHf23Dx06FBHv/IMBAABOzrvPn0t8feGPKiksDhw4EL29vVFTU3Pcek1NTezevfuE53R2dp7w+M7OzpIGXbduXSxfvvw963V1dSVdBwAAiHj11Vfj8ssvP2XXKykszqSFCxfGrFmz+m//6le/innz5sXWrVtjypQpZ3Ayzkbt7e0xY8aM+I//+I+YPHnymR6Hs5A9RIb9Q5Y9RMbOnTvjs5/9bPT19Z3S65YUFqNHj47Kysro6uo6br2rqyvGjh17wnPGjh1b0vHvp7a2Nmpra9+zPnz4cJ+xoGTnn39+/1/tHwbCHiLD/iHLHiJj+PDhEfHORxpOpZKuVlVVFVOnTo3W1tb+tb6+vmhtbY2GhoYTntPQ0HDc8RERzz333PseDwAAnH1KfitUc3NzzJ8/P6ZNmxbTp0+P1atXx5EjR2LBggURETFv3rwYP358rFy5MiIi7rzzzpgxY0Z84xvfiOuuuy42bNgQv/jFL+LRRx89tY8EAAA4Y0oOizlz5sT+/ftj2bJl0dnZGZMnT44tW7b0f0B73759x72scuWVV8aTTz4Z9957b9x9993xl3/5l/HMM8/EJZdckhq8trY2WlpaTvj2KPhj7B+y7CEy7B+y7CEyBmv/lPw9FgAAAH/o1H5iAwAA+EgSFgAAQJqwAAAA0oQFAACQJiwAAIA0YQEAAKQJCwAAIE1YAAAAacICAABIK+uwWLNmTUyYMCGqq6ujvr4+tm3b9oHHf//734+LL744qqur49JLL43NmzefpkkpR6Xsn/Xr18fVV18do0aNilGjRkVjY+Mf3W98uJX658+7NmzYEEOGDInZs2cP7oCUvVL30FtvvRWLFi2K2traGDZsWFx00UX+O/YRVur+Wb16dXziE5+Ic889N+rq6mLx4sXxu9/97jRNS7n52c9+FjNnzoxx48bFkCFD4plnnvmj52zdujUuv/zyGDZsWHz84x+PJ554ovQ7LsrUhg0biqqqquLxxx8vfvWrXxW33HJLccEFFxRdXV0nPP7nP/95UVlZWTzwwAPFSy+9VNx7773FOeecU7z44ouneXLKQan754YbbijWrFlT7Ny5s9i1a1fx93//98XIkSOL//7v/z7Nk1MOSt0/73rttdeK8ePHF1dffXXxN3/zN6dnWMpSqXvo//7v/4pp06YV1157bfH8888Xr732WrF169aivb39NE9OOSh1/3z3u98thg0bVnz3u98tXnvtteLZZ58tamtri8WLF5/mySkXmzdvLu65557iqaeeKiKiePrppz/w+L179xbnnXde0dzcXLz00kvFN7/5zaKysrLYsmVLSfdbtmExffr0YtGiRf23e3t7i3HjxhUrV6484fFf+MIXiuuuu+64tfr6+mLhwoWDOiflqdT984eOHTtWDB8+vPjOd74zWCNSxgayf44dO1ZceeWVxbe//e1i/vz5wuIjrtQ99K1vfau48MILi56entM1ImWs1P2zaNGi4nOf+9xxa83NzcVVV101qHNydjiZsPjyl79cfPrTnz5ubc6cOUVTU1NJ91WWb4Xq6emJ7du3R2NjY/9aRUVFNDY2Rltb2wnPaWtrO+74iIimpqb3PZ4Pr4Hsnz/09ttvx9GjR+NjH/vYYI1JmRro/vnqV78aY8aMiZtuuul0jEkZG8ge+uEPfxgNDQ2xaNGiqKmpiUsuuSRWrFgRvb29p2tsysRA9s+VV14Z27dv73+71N69e2Pz5s1x7bXXnpaZOfudqufRQ0/lUKfKgQMHore3N2pqao5br6mpid27d5/wnM7OzhMe39nZOWhzUp4Gsn/+0F133RXjxo17z79kfPgNZP88//zz8dhjj0V7e/tpmJByN5A9tHfv3vjJT34SN954Y2zevDn27NkTt99+exw9ejRaWlpOx9iUiYHsnxtuuCEOHDgQn/nMZ6Ioijh27Fjcdtttcffdd5+OkfkQeL/n0d3d3fHb3/42zj333JO6Tlm+YgFn0qpVq2LDhg3x9NNPR3V19ZkehzJ36NChmDt3bqxfvz5Gjx59psfhLNXX1xdjxoyJRx99NKZOnRpz5syJe+65J9auXXumR+MssHXr1lixYkU88sgjsWPHjnjqqadi06ZNcf/995/p0fiIKctXLEaPHh2VlZXR1dV13HpXV1eMHTv2hOeMHTu2pOP58BrI/nnXgw8+GKtWrYof//jHcdlllw3mmJSpUvfPq6++Gq+//nrMnDmzf62vry8iIoYOHRovv/xyTJw4cXCHpqwM5M+g2traOOecc6KysrJ/7ZOf/GR0dnZGT09PVFVVDerMlI+B7J/77rsv5s6dGzfffHNERFx66aVx5MiRuPXWW+Oee+6Jigr/H5kP9n7Po0eMGHHSr1ZElOkrFlVVVTF16tRobW3tX+vr64vW1tZoaGg44TkNDQ3HHR8R8dxzz73v8Xx4DWT/REQ88MADcf/998eWLVti2rRpp2NUylCp++fiiy+OF198Mdrb2/t/Zs2aFddcc020t7dHXV3d6RyfMjCQP4Ouuuqq2LNnT3+URkS88sorUVtbKyo+Ygayf95+++33xMO7kfrOZ3fhg52y59Glfa789NmwYUMxbNiw4oknniheeuml4tZbby0uuOCCorOzsyiKopg7d26xZMmS/uN//vOfF0OHDi0efPDBYteuXUVLS4tfN/sRVur+WbVqVVFVVVX84Ac/KDo6Ovp/Dh06dKYeAmdQqfvnD/mtUJS6h/bt21cMHz68+NKXvlS8/PLLxY9+9KNizJgxxT//8z+fqYfAGVTq/mlpaSmGDx9e/Nu//Vuxd+/e4t///d+LiRMnFl/4whfO1EPgDDt06FCxc+fOYufOnUVEFA899FCxc+fO4te//nVRFEWxZMmSYu7cuf3Hv/vrZv/pn/6p2LVrV7FmzZoP16+bLYqi+OY3v1n82Z/9WVFVVVVMnz69+K//+q/+vzdjxoxi/vz5xx3/ve99r7jooouKqqqq4tOf/nSxadOm0zwx5aSU/fPnf/7nRUS856elpeX0D05ZKPXPn/+fsKAoSt9DL7zwQlFfX18MGzasuPDCC4uvfe1rxbFjx07z1JSLUvbP0aNHi6985SvFxIkTi+rq6qKurq64/fbbi//93/89/YNTFn7605+e8HnNu/tm/vz5xYwZM95zzuTJk4uqqqriwgsvLP71X/+15PsdUhReIwMAAHLK8jMWAADA2UVYAAAAacICAABIExYAAECasAAAANKEBQAAkCYsAACANGEBAACkCQsAACBNWAAAAGnCAgAASBMWAABA2v8DCBYVRjzfpWgAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 800x550 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from functools import partial\n",
    "from utils.evaluation import supply_rng\n",
    "from matplotlib.backends.backend_agg import FigureCanvasAgg as FigureCanvas\n",
    "\n",
    "def get_canvas_image(canvas):\n",
    "    canvas.draw() \n",
    "    out_image = np.asarray(canvas.buffer_rgba())\n",
    "    return out_image\n",
    "\n",
    "def policy_image_grid(env, dataset, action_fn=None, **kwargs):\n",
    "    fig = plt.figure(tight_layout=True)\n",
    "    canvas = FigureCanvas(fig)\n",
    "    plot_policy(env, dataset, fig=fig, ax=plt.gca(), action_fn=action_fn, **kwargs)\n",
    "    image = get_canvas_image(canvas)\n",
    "    plt.close(fig)\n",
    "    return image\n",
    "\n",
    "def plot_policy(env, dataset, fig=None, ax=None, title=None, action_fn=None, **kwargs):\n",
    "    action_names = [\n",
    "            r'$\\leftarrow$', r'$\\rightarrow$', r'$\\uparrow$', r'$\\downarrow$'\n",
    "        ]\n",
    "    if fig is None or ax is None:\n",
    "        fig, ax = plt.subplots()\n",
    "    \n",
    "    grid = env.get_grid_array()\n",
    "    ax = env.plot_grid(ax=ax, grid=grid)\n",
    "    \n",
    "    goal = kwargs.get('goal', None)\n",
    "    for (y, x), value in np.ndenumerate(grid):\n",
    "        if value == 1 or value == 4:\n",
    "            action = action_fn(np.concatenate([[x], [y]], -1)).squeeze()\n",
    "            action_name = action_names[action]\n",
    "            ax.text(x, y, action_name, ha='center', va='center', fontsize='large', color='green')\n",
    "            \n",
    "    ax.set_title('Goal: ({:.2f}, {:.2f})'.format(goal[0], goal[1])) \n",
    "    ax.scatter(goal[0], goal[1], s=80, c='black', marker='*')\n",
    "        \n",
    "    if title:\n",
    "        ax.set_title(title)\n",
    "        \n",
    "    return fig, ax\n",
    "\n",
    "def visualize_policy_image(layout_type, task_num):\n",
    "    mdp_type = np.zeros((NUM_TRAIN_LAYOUTS, ))\n",
    "    if mdp_type is not None:\n",
    "        if layout_type == 0:\n",
    "            mdp_type[0] = 1\n",
    "        elif layout_type == 1:\n",
    "            mdp_type[1] = 1\n",
    "        else:\n",
    "            mdp_type[2] = 1\n",
    "\n",
    "    env = DynamicsGeneralization_Doors(render_mode=\"rgb_array\", highlight=False,\n",
    "                                    max_steps=NUM_TRAIN_STEPS, task_num=task_num)\n",
    "    env._gen_grid = partial(env._gen_grid, layout_type=layout_type)\n",
    "    env = MinigridWrapper(env)\n",
    "    dynamics_embedding=None\n",
    "    obs, info = env.reset()\n",
    "    goal = info.get(\"goal_pos\", None)\n",
    "    \n",
    "    latent_z = jax.device_get(fb_agent.infer_z(goal)[None])\n",
    "    example_batch = gc_whole_dataset.sample(1)\n",
    "    pred_policy_img = policy_image_grid(env, example_batch,\n",
    "                                                    action_fn=partial(supply_rng(fb_agent.sample_actions,\n",
    "                                                                                rng=jax.random.PRNGKey(np.random.randint(0, 2**32))), latent_z=latent_z,\n",
    "                                                                temperature=0.0),\n",
    "                                                    goal=goal)\n",
    "    return pred_policy_img\n",
    "\n",
    "pred_policy_img = visualize_policy_image(layout_type=1, task_num=0)\n",
    "plt.imshow(pred_policy_img)\n",
    "plt.tight_layout()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<minigrid.core.grid.Grid at 0x7074100d0ac0>"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "env.env.unwrapped.grid"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.image.AxesImage at 0x7f4348146350>"
      ]
     },
     "execution_count": 16,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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1KmsW7eu3WN27drFmWeeSW5uLvn5+Zx//vnU13v3GigRERFxydB+2L67N0lLXQ6kDQ0NHHLIIdxzzz0dHv/Vr37FXXfdxe9+9zsWLVpEVlYW06ZNo7n50xnKZ555JqtXr2b27Nk899xzvP766/z4xz/u/lmIiIiIx7Uuim85cXc3hdK01OUh+xNPPJETTzyxw2PGGH7zm99w7bXXcsoppwDwyCOPUFRUxNNPP833vvc93n//fV588UXefvttJk6cCMDdd9/NN77xDX79619TWlrq4nRERETEk9rWIXVcXLfhqJc0XfXoNaQfffQRFRUVTJ06NXFfXl4ekyZNYuHChXzve99j4cKF5OfnJ8IowNSpU7Ftm0WLFvGtb31rnz7Xtm3b2LZtW+JjDfmLiIh4WU/s1KS97NNVjwbSiooKAIqKitrdX1RUlDhWUVFB//792xfh91NQUJB4zL64//77ufHGG11WLCIiIsnjdtknhdF09aWdZX/BBRdw8sknJz6ur69nypQpKaxIREREOrdnpyYXPZwWDuohTU89GkiLi4sBqKyspKSkJHF/ZWUl48ePTzymqqqq3fNisRi7du1KPH9flJSUtPsctbXeXRhaRERkv5e4htRFoNQ1pGnLxYUceysrK6O4uJg5c+Yk7qutrWXRokWUl5cDUF5eTnV1NUuXLk085tVXX8VxHCZNmtST5YiIiIiXaNkn6USXe0jr6+tZv3594uOPPvqI5cuXU1BQwODBg7nsssv4+c9/zsiRIykrK+N//ud/KC0t5Zvf/CYABx10ECeccAI/+tGP+N3vfkc0GuXiiy/me9/7nmbYi4iIpC0DjuNyUpN2akpXXQ6kS5Ys4Wtf+1ri45kzZwJwzjnn8PDDD3PVVVfR0NDAj3/8Y6qrq/nKV77Ciy++SDgcTjznscce4+KLL+a4447Dtm1OO+007rrrrh44HREREfGktoXxHRc7YWnIPm11OZAec8wxn7tVmWVZ3HTTTdx0002dPqagoIDHH3+8q59aREREvrTaht3dvITCaLr60s6yFxERkS8Xy3GwTPcXxrd0HWnaUiAVERHpJcYYLMvFzkTppKd2apK01KOz7EVERKRVQ0MLd935Eo6baybTSg/NsFcPaVpSIBUREekFixat5x//WMTHH+9IdSne4bgMpAr3aUuBVEREpBc88/QSPlxfyTvL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",
      "text/plain": [
       "<Figure size 800x550 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from mpl_toolkits.axes_grid1 import make_axes_locatable\n",
    "\n",
    "def value_image_fourrooms(env, dataset, value_fn, action_fn=None, **kwargs):\n",
    "    fig = plt.figure(tight_layout=True)\n",
    "    canvas = FigureCanvas(fig)\n",
    "    plot_value_image_grid(env, dataset, value_fn, fig=fig, ax=plt.gca(), action_fn=action_fn, **kwargs)\n",
    "    image = get_canvas_image(canvas)\n",
    "    plt.close(fig)\n",
    "    return image\n",
    "\n",
    "def plot_value_image_grid(env, dataset, value_fn, action_fn, fig=None, ax=None, title=None, **kwargs):\n",
    "    if fig is None or ax is None:\n",
    "        fig, ax = plt.subplots()\n",
    "    \n",
    "    goal = kwargs.get('goal', None)\n",
    "    grid = env.get_grid_array()\n",
    "    ax = env.plot_grid(ax=ax, grid=grid)\n",
    "\n",
    "    for (y, x), value in np.ndenumerate(grid):\n",
    "        if value == 1 or value == 4:\n",
    "            grid[y, x] = jax.device_get(value_fn(np.concatenate([[x], [y]], -1)).max(-1)[0])\n",
    "            \n",
    "    divider = make_axes_locatable(ax)\n",
    "    cax = divider.append_axes('right', size='5%', pad=0.05)\n",
    "    im = ax.imshow(grid, cmap='inferno', vmin=-1000)\n",
    "    fig.colorbar(im, cax=cax, orientation='vertical')\n",
    "    if goal is not None:\n",
    "        ax.set_title('Goal: ({:.2f}, {:.2f})'.format(goal[0], goal[1])) \n",
    "        ax.scatter(goal[0], goal[1], s=80, c='black', marker='*')\n",
    "    return fig, ax\n",
    "\n",
    "def visualize_value_image(layout_type, task_num):\n",
    "    mdp_type = None\n",
    "    env = DynamicsGeneralization_Doors(render_mode=\"rgb_array\", highlight=False,\n",
    "                                    max_steps=NUM_TRAIN_STEPS, task_num=task_num)\n",
    "    env._gen_grid = partial(env._gen_grid, layout_type=layout_type)\n",
    "    env = MinigridWrapper(env)\n",
    "    dynamics_embedding=None\n",
    "    obs, info = env.reset()\n",
    "    goal = info.get(\"goal_pos\", None)\n",
    "    \n",
    "    latent_z = jax.device_get(fb_agent.infer_z(goal)[None])\n",
    "    pred_value_img = value_image_fourrooms(env, example_batch,\n",
    "                                value_fn=partial(fb_agent.predict_q, z=latent_z),\n",
    "                                action_fn=None, goal=goal)\n",
    "    return pred_value_img\n",
    "\n",
    "pred_value_img = visualize_value_image(2, task_num=1)\n",
    "plt.imshow(pred_value_img)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "fe85a10d7dba40f58691b99e65fed429",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/100000 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "ename": "NameError",
     "evalue": "name 'clear_output' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[17], line 8\u001b[0m\n\u001b[1;32m      5\u001b[0m fb_agent, info \u001b[38;5;241m=\u001b[39m fb_agent\u001b[38;5;241m.\u001b[39mupdate(batch)\n\u001b[1;32m      7\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m update_step \u001b[38;5;241m%\u001b[39m \u001b[38;5;241m5_000\u001b[39m \u001b[38;5;241m==\u001b[39m \u001b[38;5;241m0\u001b[39m:\n\u001b[0;32m----> 8\u001b[0m     \u001b[43mclear_output\u001b[49m()\n\u001b[1;32m      9\u001b[0m     fig, ax \u001b[38;5;241m=\u001b[39m plt\u001b[38;5;241m.\u001b[39msubplots(nrows\u001b[38;5;241m=\u001b[39m\u001b[38;5;241m1\u001b[39m, ncols\u001b[38;5;241m=\u001b[39m\u001b[38;5;241m2\u001b[39m, figsize\u001b[38;5;241m=\u001b[39m(\u001b[38;5;241m20\u001b[39m, \u001b[38;5;241m10\u001b[39m))\n\u001b[1;32m     10\u001b[0m     pred_policy_img \u001b[38;5;241m=\u001b[39m visualize_policy_image(task_num\u001b[38;5;241m=\u001b[39m\u001b[38;5;241m0\u001b[39m)\n",
      "\u001b[0;31mNameError\u001b[0m: name 'clear_output' is not defined"
     ]
    }
   ],
   "source": [
    "pbar = tqdm(range(100_000))\n",
    "\n",
    "for update_step in pbar:\n",
    "    batch = gc_whole_dataset.sample(1024)\n",
    "    fb_agent, info = fb_agent.update(batch)\n",
    "    \n",
    "    if update_step % 5_000 == 0:\n",
    "        clear_output()\n",
    "        fig, ax = plt.subplots(nrows=1, ncols=2, figsize=(20, 10))\n",
    "        pred_policy_img = visualize_policy_image(task_num=0)\n",
    "        ax[0].imshow(pred_policy_img)\n",
    "        pred_value_img = visualize_value_image(task_num=0)\n",
    "        ax[1].imshow(pred_value_img)\n",
    "        \n",
    "        fig.suptitle(f\"Training step: {update_step}\")   \n",
    "        plt.tight_layout()\n",
    "        display(fig)\n",
    "        plt.close(fig)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "jax2",
   "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.10.16"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
