{
 "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",
    "import numpy as np\n",
    "\n",
    "# VIZ\n",
    "import matplotlib.pyplot as plt\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)\n",
    "START_POS = (1,1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAxYAAAEhCAYAAADiXjabAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjEsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvc2/+5QAAAAlwSFlzAAAPYQAAD2EBqD+naQAAKZxJREFUeJzt3X+wXHV9//HXPWzPZbObQExSNphMzAhLIG6JotAL8k1svTggCgNF0FpA2tiStiqKgTpThc50dJQWHX+UtNS0KANaoBSwiiik2tTqWAa8cE3TSMCkYWPAGHaXm7tZzuf7x/Zs9mYvJJvP57O7n5vnYybjdbN5nc8978++97xzNpchY4wRAAAAAFiI+r0AAAAAAOFjsAAAAABgjcECAAAAgDUGCwAAAADWGCwAAAAAWGOwAAAAAGCNwQIAAACANQYLAAAAANYYLAAAAABYY7A4AmzYsEFDQ0PasGGDVc55552n1atXu1mUA5dddpne9a539XsZQHBmak+4/vrrdcYZZ/R7GUCQ6AtwgcGiR7Zu3ao/+ZM/UbFY1KxZszRr1iydcsop+uM//mP95Cc/6ffyDmrjxo369re/reuuu27K488++6ze//73a+nSpcpms3rta1+rD3/4w3r++ecP+1h79uzR2rVrdeKJJyqbzWrJkiX6/d//ff385z+f8rzrrrtOd999tx5//PHDPhbQLzOxJ2zatElr167VihUrNHv2bC1cuFBvf/vb9eMf/9jqWHv37tUnP/lJnXLKKZo1a5Ze/epX65JLLtGTTz455Xkf+tCH9Pjjj+u+++6zOh7QLzOxL0jurxW2bdumG2+8Uaeffrrmzp2r+fPna9WqVfrOd77T8Vz6Qm8NGWNMvxcx0z3wwAO69NJLlclk9Lu/+7s69dRTFUWRNm3apHvuuUfPPPOMtm7dqiVLlng5/oYNG/SWt7xFjzzyiFatWnVYGRdeeKEmJib04IMPth6rVqt63etep1qtpjVr1mjx4sV6/PHHtW7dOi1fvlz/9V//pSjqbnZNkkS/+Zu/qfHxca1Zs0bFYlFbtmzRl770Jc2ZM0c//elPNXv27NbzzzjjDJ100km67bbbDuv7AvphpvaEa6+9Vn//93+viy++WKeffrr27NmjdevW6emnn9a3vvUtvfWtbz2sY1188cW67777tHr1ar3hDW/Qjh079MUvflETExMaGxubcp4uvfRSPfvss/re9753WMcC+mWm9gUf1wpf+MIXtHbtWl144YU666yz1Gg0dNttt+nRRx/Vl7/8Zb3vfe+b8nz6Qg8ZeLVlyxaTy+XMySefbHbs2NHx+/v27TOf+9znzM9//nNva3jkkUeMJPPII48c1p/fuXOnyWQy5tZbb53y+O23324kmQceeGDK4x//+MeNJPPoo492fayNGzcaSeYLX/jClMe//OUvG0nmnnvumfL4TTfdZHK5nKlUKl0fC+iHmdwTfvzjH3e8Fp977jmzYMECc9ZZZx3WsbZv324kmWuvvXbK4w8//LCRZP76r/96yuN33XWXGRoaMj/72c8O63hAP8zkvuDjWuGJJ54wu3btmvLY3r17zbJly8yiRYs6nk9f6B0+CuXZpz/9adVqNa1fv14LFy7s+P1MJqMPfOADWrx48ZTHH374YZ199tnK5XI69thjdcEFF+inP/3plOc888wzWrNmjU466SRls1nNmzdPl1xyiZ5++umDruvFF1/Upk2b9Nxzzx30ud/4xjfUaDQ6/rbxhRdekCQdd9xxUx5Pv89sNnvQ7AN1mzk6OqparaaHHnqo62MB/TCTe8Jpp52mfD4/5bF58+bp7LPP7ljroapUKpIOvSeka/qXf/mXwzoe0A8zuS/4uFZYvny55s+fP+Wx4eFhnXfeedq+fXurb6ToCz3U78lmpjv++OPNCSec0NWfeeihh0wmkzHFYtF8+tOfNjfeeKOZP3++mTt3rtm6dWvref/0T/9kTj31VPPxj3/c/O3f/q352Mc+ZubOnWuWLFliarVa63nT/S1E+tgnPvGJg67nD/7gD8y8efM6Hn/yySdNFEXmzDPPND/4wQ/Mtm3bzDe+8Q2zaNEic+GFF3b1Pad27dplcrmcWbZsmfnud79rtm/fbjZs2GBKpZJ505veZPbt2zfl+fv27TPZbNZ85CMfOazjAb02k3vCyznzzDNNsVg85Oe3q9frZtGiRaZQKJj77rvPbNu2zfzwhz80K1euNEuXLjW7d+/u+DMnnHCCufjiiw/reEA/zOS+4ONa4eW85z3vMbNmzTKNRqPj9+gLvcFg4dGePXuMpGlfOLt37za7du1q/XrxxRdbv7dixQrz67/+6+b5559vPfb444+bKIrM5Zdf3nqs/c+kfvCDHxhJ5rbbbms9Ztss3vzmN5vTTjtt2t+79dZbzbHHHmsktX5dccUVHQNANx544AGzcOHCKZlve9vbXvbjTsVi0Zx77rmHfTygV46EnnCg733ve2ZoaMj8+Z//+SE9fzo//OEPzWtf+9opPeG0004zzz777LTPP+ecc8zJJ5982McDeulI6As+rhUO9D//8z/m6KOPNr/3e7837e/TF3qDj0J5lN7+O/CjAZK0atUqLViwoPXri1/8oqTmT0547LHHdOWVV+pVr3pV6/m/8Ru/odHRUf3rv/5r67H224f79u3T888/rxNOOEHHHnusHn300Vdc26pVq2SM0Q033HDQ7+P555/X3Llzp/29V7/61Tr99NP12c9+Vv/8z/+sD3/4w7r99tt1/fXXHzT35SxYsECvf/3r9Zd/+Ze69957dcMNN+j73/9+xz/GSs2dO/eQbtMC/XYk9IR2v/jFL/Se97xHS5cu1dq1aw/6/Jczd+5crVixQtdff73uvfde3XTTTXr66ad1ySWXaO/evdM+n56AUBwJfcHHtUK7F198UZdccomy2aw+9alPTfsc+kJvZPq9gJks/elF1Wq14/fWrVunSqWinTt36r3vfW/r8WeeeUaSdNJJJ3X8mZNPPlkPPvigarWacrmcJiYm9MlPflLr16/X//7v/8q0/YCvPXv2OP1ezDQ/PGzjxo06//zz9Z//+Z964xvfKKn5EyHmzJmjG2+8UVdddZVOOeWUro7z1FNP6S1veYtuu+02XXzxxZKkCy64QK95zWt05ZVX6pvf/KbOPffcjrUNDQ0d5ncG9M5M7wntarWazj//fFUqFf37v//7tBdNh2LPnj06++yz9dGPflQf+chHWo+/8Y1v1KpVq7R+/XpdffXVHWujJyAUM70v+LhWaPfSSy/psssu0/j4uL75zW/q+OOPf9m10Rf8Y7Dw6JhjjtHChQv1xBNPdPxe+h9rOZR/PPVy/vRP/1Tr16/Xhz70IY2MjOiYY47R0NCQLrvsMiVJcti5B5o3b552797d8fi6det03HHHtRpF6p3vfKduuOEG/cd//EfXzeIf/uEftHfvXp1//vkdmVKzQR04WOzevVsnnnhiV8cB+mGm94RUvV7XRRddpJ/85Cd68MEH9brXve6wj3X33Xdr586drR6QWrlypebMmaONGzd2DBa7d+/u+IedwKCa6X3Bx7VCu9WrV+uBBx7Q7bffrt/6rd962efRF3qDwcKzt7/97br11lv1ox/9SKeffvpBn5/+fOr//u//7vi9TZs2af78+crlcpKku+66S1dccYX+6q/+qvWcvXv36le/+pWbxf+fZcuW6e677+54fOfOnXrppZc6Ht+3b58kqdFodH2snTt3yhjTkftymY1GQ9u2beu46AAG1UzuCVLzv0Vz+eWX67vf/a6+/vWva+XKlVbH2rlzpyR19IS0T0zXZ7Zu3apTTz3V6rhAL83kvuDjWiH10Y9+VOvXr9dnP/tZvfvd737F59IXeoN/Y+HZ2rVrNWvWLF111VWtN8h2B942XLhwoVasWKF//Md/nPKif+KJJ/Ttb39b5513Xuuxo446quPPf/7zn5/2BXygbn6E3MjIiHbv3q2nnnpqyuPFYlE7d+7Uhg0bpjx+xx13SJJe//rXHzT7QMViUcYYff3rXz+kzPHxce3du1dnnnlm18cC+mEm9wSp+bejX/va1/SlL31JF1100UGzDqZYLEqS7rzzzimP33fffarVah09Yc+ePfrZz35GT0BQZnJf8HGtIEmf+cxndNNNN+ljH/uYPvjBD77ic+kLPdTTfyp+hLr33ntNNps1xxxzjFmzZo1Zt26dueWWW8x1111nFi9ebKIoMnfccUfr+emPkFu2bJn5zGc+Y/7iL/7CLFiwwMydO9c89dRTreddfvnl5qijjjIf/OAHzbp168yVV15pFi1aZObNm2euuOKK1vNsf9JDuVw2mUzGrFu3bsrjmzZtMrlczuTzefNnf/Zn5pZbbjHvfve7jSQzOjo65bnr1683ksz69etf8VjPPfecKRQKJo5j84EPfMCsW7fO/OEf/qE56qijzPLly83k5OSU5990001m1qxZ5oUXXjjo9wEMipnaE26++WYjyYyMjJivfOUrHb+q1WrXx5ucnDTLly83Q0ND5sorrzS33HKLufbaa83RRx9tFi5c2PEfybrrrruMJLNly5aDfh/AIJmpfcHHtcI999xjJJkTTzxx2l5TLpenPJ++0DsMFj2yZcsWc/XVV5sTTjjBHH300SabzZply5aZP/qjPzKPPfZYx/O/853vmLPOOstks1kzZ84c8453vMOMj49Pec7u3bvN+973PjN//nyTz+fN2972NrNp0yazZMkSp83CGGPe+c53mt/+7d/ueHzTpk3md37n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",
      "text/plain": [
       "<Figure size 800x550 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from envs.custom_mazes.darkroom import FourRoomsMazeEnv, Maze\n",
    "\n",
    "test = FourRoomsMazeEnv(Maze(seed=0))\n",
    "test.reset()\n",
    "img = test.render(return_img=True)\n",
    "\n",
    "fig, ax = plt.subplots(nrows=1, ncols=3)\n",
    "for i, cur_ax in enumerate(ax.flat, start=0):\n",
    "    test.setup_goals(seed=None, task_num=i, start_pos=START_POS)\n",
    "    test.render(ax=cur_ax)\n",
    "    cur_ax.set_title(f\"Goal: {test.goal}\")\n",
    "plt.tight_layout()\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Data Collection"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "cfd7275178164b898c44edf97bdf9d82",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/1 [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\">300000</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\">300000</span>, <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">1</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\">300000</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\">300000</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\">300000</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;36m300000\u001b[0m,\u001b[1m)\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'layout_type'\u001b[0m: \u001b[1m(\u001b[0m\u001b[1;36m300000\u001b[0m, \u001b[1;36m1\u001b[0m\u001b[1m)\u001b[0m,\n",
       "\u001b[2;32m│   \u001b[0m\u001b[32m'next_observations'\u001b[0m: \u001b[1m(\u001b[0m\u001b[1;36m300000\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;36m300000\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;36m300000\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_fourrooms, q_learning\n",
    "\n",
    "train_layout_data = []\n",
    "\n",
    "NUM_TRAIN_LAYOUTS = 1\n",
    "NUM_TRAIN_STEPS = 100\n",
    "NUM_TRAIN_EPISODES = 3000\n",
    "seeds = np.arange(0, NUM_TRAIN_LAYOUTS)\n",
    "\n",
    "if NUM_TRAIN_LAYOUTS > 1:\n",
    "    for i in tqdm(range(NUM_TRAIN_LAYOUTS)):\n",
    "        env = FourRoomsMazeEnv(Maze(seed=seeds[i], maze_type='fourrooms_random_layouts'), max_steps=NUM_TRAIN_STEPS)\n",
    "        dataset, env = random_exploration_fourrooms(env, num_episodes=NUM_TRAIN_EPISODES, layout_type=i, num_mdp=NUM_TRAIN_LAYOUTS)\n",
    "        train_layout_data.append(dataset)\n",
    "else:\n",
    "    for i in tqdm(range(1)):\n",
    "        env = FourRoomsMazeEnv(Maze(seed=seeds[i]), max_steps=NUM_TRAIN_STEPS)\n",
    "        dataset, env = random_exploration_fourrooms(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.maze.size)\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": [
    "# Initialize Vanilla-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\">150</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;36m150\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': (300000,), 'layout_type': (300000, 1), 'next_observations': (300000, 2), 'observations': (300000, 2), 'terminals': (300000,)}\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",
    "                                                            f'agent.z_dim=150',\n",
    "                                                            f'agent.z_mix_ratio=0.5'])\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": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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VnfffbeaNWvmuG/q1Kny9vbWY489pl27dunVV1+Vu7u7XFxcdOLECU2aNEmrVq3SrFmz1KBBA02YMMHx2DfeeEOtWrXSgAED5Obmpi+//FL33nuv7Ha7xo0b51TDrl27NGjQIN11110aPny43nnnHY0YMUIdO3ZUq1atJEkZGRm68sor9ccff2jUqFG67LLLlJKSoi+++EIHDx5UUFCQ7Ha7BgwYoJ9//lljxoxRixYttHnzZr388svasWPHBedcz5o1SyNHjlTnzp01depUJSUl6T//+Y9++eUXx/fkn//8p5o1a6Y333zTMb2pUaNGF/z+p6WlOQUgSQoKCnK6nZOT49jn9OnT2rhxo6ZNm6arrrqq0DSqsyUlJalbt27KysrS3//+dwUGBurdd9/VgAEDNG/ePN14441O+z/33HNycXHRQw89pLS0ND3//PO6/fbbtXr16gu+lgsparxK+e+vj4+Pxo8fLx8fHy1dulQTJkxQenq6XnjhhQse96OPPtLJkyc1duxY2Ww2Pf/887rpppu0Z8+eEp/lmDt3rrKysnTPPfcoMDBQa9as0auvvqqDBw9q7ty5kqRBgwZp3Lhx+vDDD9WhQwenx3/44Yfq0aOHwsPDHa/tQmOnuIKDg/XGG2/onnvu0Y033qibbrpJktS2bVtJ0s0336wtW7bob3/7m6KiopScnKzFixcrISHhvE3vixcv1p49ezRy5EiFhoZqy5YtevPNN7VlyxatWrVKNptNknT48GF16dJFqampGjNmjJo3b65Dhw5p3rx5ysrKcjpDeu+99yo4OFgTJkxQZmamJGnSpEmaPHmyYmNjdc8992j79u164403tHbtWv3yyy9yd3dXdna2evfurTNnzuhvf/ubQkNDdejQIX311VdKTU2Vn5+ftmzZouuvv15t27bV008/LU9PT+3atUu//PLLed+/999/X6NHj1aXLl00ZswYSXL8jK5du1YrVqzQkCFDVK9ePe3bt09vvPGGevTooa1btzrOrE2aNElTp051HCc9PV3r1q3Thg0b1KtXryKfd+3aterdu7c6deqkzz//XN7e3kpOTta1116r4OBgPfbYY/L399e+ffs0f/78Cw0DoHIyAMrUzJkzjSTz/fffm6NHj5oDBw6YTz75xAQGBhpvb29z8OBBs2/fPuPq6mqeffZZp8du3rzZuLm5OW3v3r27kWTi4+MLPdfw4cNNzZo1nbZt2rTJSDKjR4922v7QQw8ZSWbp0qWObfXr1zeSzKJFi5z2XbZsmZFkWrdubbKzsx3bhw4damw2m+nTp4/T/jExMaZ+/fpO27KysgrV27t3b9OwYUOnbQU1/Pjjj45tycnJxtPT0zz44IOObRMmTDCSzPz58wsd1263G2OMef/9942Li4v56aefnO6Pj483kswvv/xS6LEFsrOzTZ06dUzr1q3NqVOnHNu/+uorI8lMmDDBsa3ge7x27dpzHu/sfYv6Kup9OPvr8ssvNykpKRd8ngceeMBIcnrtJ0+eNA0aNDBRUVEmLy/PGPO/722LFi3MmTNnHPv+5z//MZLM5s2bL/hcBcaNG1fodZxvvBY1JsaOHWtq1KhhTp8+7dg2fPhwp/G0d+9eI8kEBgaa48ePO7Z//vnnRpL58ssvz1tnwWtetmzZeWuZOnWqsdlsZv/+/Y5tQ4cONXXr1nW8f8YYs2HDBiPJzJw50xhTsrHTvXt3071790LPffZrPnr0qJFkJk6c6LTfiRMnjCTzwgsvnPc1F6Wo1/zxxx8X+vkbNmyYcXFxKXJ8F/ysFYzrK664wuTm5jruT05ONh4eHubaa691es/++9//GknmnXfeMcYYs3HjRiPJzJ0795z1vvzyy0aSOXr0aIlfa82aNc3w4cMLbS/qPVi5cqWRZN577z3Htnbt2pl+/fqd9zn++vv3559/Nr6+vqZfv35OY3nBggXF/l0BVAVMhQLKSWxsrIKDgxUREaEhQ4bIx8dHCxYsUHh4uObPny+73a5bb71VKSkpjq/Q0FA1adJEy5YtczqWp6enRo4cWaznXbhwoSRp/PjxTtsLmoLPnhrToEED9e7du8hjDRs2zOkvwdHR0TLGFDqNHx0drQMHDig3N9ex7a99GgV/re/evbv27NnjNNVBklq2bKkrr7zScTs4OFjNmjXTnj17HNs+/fRTtWvXrtBf3iU5/tI6d+5ctWjRQs2bN3d6XwumoZ39vv7VunXrlJycrHvvvVdeXl6O7f369VPz5s0LvW8l9dprr2nx4sVOX2eLjo523PfVV1/p2Wef1ZYtWzRgwACnKTpFWbhwobp06aIrrrjCsc3Hx0djxozRvn37tHXrVqf9R44c6fSX54L3/6/v+cU613j965g4efKkUlJSdOWVVyorK0vbtm274HEHDx6s2rVrl0nNf60lMzNTKSkp6tatm4wxTlOfhg0bpsOHDzuNnQ8//FDe3t66+eabJZX/2Dm7bg8PDy1fvlwnTpwo8WMLnD59WikpKerataskacOGDZIku92uzz77TP379y+yJ6jgZ63A3XffLVdXV8ft77//XtnZ2XrggQfk4uLitJ+vr6/jvfDz85Mkffvtt8rKyiqy3oKzPJ9//nmZNfP/9T3IycnRsWPH1LhxY/n7+zveg4Ln3rJli3bu3HnBYy5btky9e/dWz549NX/+fHl6ehZ6DV999ZVycnLK5DUAFRnBAignBR8kly1bpq1bt2rPnj2OD/A7d+6UMUZNmjRRcHCw09cff/zhaA4uEB4eXuwG7f3798vFxUWNGzd22h4aGip/f3/t37/fafv5pthERkY63S74MBAREVFou91udwoMv/zyi2JjY1WzZk35+/srODjYMTf67GBx9vNIUu3atZ0+OO3evbvQKkpn27lzp7Zs2VLoPW3atKkkFXpf/6rgffnrVLACzZs3L/S+lVSXLl0UGxvr9HW2oKAgx339+vXTE088obfeeksrVqxwTHc7X/1F1V6wQtPZ9Z/9nhd8YC/ph9WinGu8btmyRTfeeKP8/Pzk6+ur4OBgR+P32WOiKGVZc0JCgkaMGKGAgAD5+PgoODhY3bt3L1RLr169FBYWpg8//FBS/gfvjz/+WDfccINq1aolqfzHzl95enrq3//+t7755huFhIToqquu0vPPP68jR45c8LHHjx/X/fffr5CQEHl7eys4ONjx81/wmo8ePar09PQL/qwVOPv3x7neCw8PDzVs2NBxf4MGDTR+/Hi99dZbCgoKUu/evfXaa685vfeDBw/W5ZdfrtGjRyskJERDhgzRnDlzShUyTp06pQkTJigiIkKenp4KCgpScHCwUlNTnZ776aefVmpqqpo2bao2bdro4Ycf1m+//VboeKdPn1a/fv3UoUMHzZkzp9C47969u26++WZNnjxZQUFBuuGGGzRz5sxi93wBlQ09FkA56dKlyzlXAbLb7bLZbPrmm2+c/tpX4OyLLl3MKk1n/2XxXM537KJqO992Y4yk/BDQs2dPNW/eXNOmTVNERIQ8PDy0cOFCvfzyy4U+GFzoeMVlt9vVpk0bTZs2rcj7zw5ElUHPnj0lST/++KP+9re/ldlxy+o9L0pRYyo1NVXdu3eXr6+vnn76aTVq1EheXl7asGGDHn300WJ9WCyrmvPy8tSrVy8dP35cjz76qJo3b66aNWvq0KFDGjFihFMtrq6uuu222zRjxgy9/vrr+uWXX3T48OELroR1Ljabrch6z25gP58HHnhA/fv312effaZvv/1WTz31lKZOnaqlS5cW6gX5q1tvvVUrVqzQww8/rPbt28vHx0d2u13XXXfdRX9YL80Kci+99JJGjBihzz//XN99953+/ve/a+rUqVq1apXq1asnb29v/fjjj1q2bJm+/vprLVq0SLNnz9Y111yj77777pzj4Xz+9re/aebMmXrggQcUExMjPz8/2Ww2DRkyxOk9uOqqq7R7925HbW+99ZZefvllxcfHa/To0Y79PD091bdvX33++edatGiRrr/+eqfns9lsmjdvnlatWqUvv/xS3377rUaNGqWXXnpJq1at4gJ7qHIIFoAFGjVqJGOMGjRo4PhrelmpX7++7Ha7du7c6XQ9gaSkJKWmpqp+/fpl+nxF+fLLL3XmzBl98cUXTn9lPt9UpAtp1KiRfv/99wvu8+uvv6pnz57FDlYFCt6X7du3O6ZOFdi+ffsled+KUjC9rGCFmXOpX7++tm/fXmh7wRQjq+ovsHz5ch07dkzz58/XVVdd5dj+11XSLpXNmzdrx44devfddzVs2DDH9nOt+DNs2DC99NJL+vLLL/XNN98oODjYafpgScZO7dq1i5y6dfZZjQuN30aNGunBBx/Ugw8+qJ07d6p9+/Z66aWX9MEHHxS5/4kTJ7RkyRJNnjzZaZGFs6f6BAcHy9fX94I/a+fy1/eiYcOGju3Z2dnau3dvoTN1bdq0UZs2bfTkk09qxYoVuvzyyxUfH69//etfkiQXFxf17NlTPXv21LRp0zRlyhT985//1LJly857DZlzvX/z5s3T8OHD9dJLLzm2nT59utCKVZIUEBCgkSNHauTIkcrIyNBVV12lSZMmOQULm82mDz/8UDfccINuueUWffPNN0Wu+tW1a1d17dpVzz77rD766CPdfvvt+uSTT5yOBVQFTIUCLHDTTTfJ1dVVkydPLvTXS2NMkcs/Flffvn0lSa+88orT9oK/4vfr1++ij11cBX9J/OtrS0tL08yZMy/6mDfffLN+/fXXQqta/fV5br31Vh06dEgzZswotM+pU6ccq9YUpVOnTqpTp47i4+Odpil88803+uOPPy7J+1aUL7/8UpLUrl278+7Xt29frVmzRitXrnRsy8zM1JtvvqmoqCi1bNmyXOu8kKLGRHZ2tl5//fUKUYsxRv/5z3+K3L9t27Zq27at3nrrLX366acaMmSI0/K/JRk7jRo10rZt23T06FHHtl9//bXQSkcFqxOd/YE3KytLp0+fdtrWqFEj1apV67zTa4p6zVLh3xMuLi4aOHCgvvzyS61bt67QcS50dig2NlYeHh76v//7P6d93377baWlpTnei/T0dKeeLCk/ZLi4uDheR1FXnG/fvr0kXXAqUc2aNYsMC66uroVew6uvvlrojNHZv4N9fHzUuHHjIp/Xw8ND8+fPV+fOndW/f3+ni1meOHGi0PMV9zUAlRFnLAALNGrUSP/617/0+OOPa9++fRo4cKBq1aqlvXv3asGCBRozZoweeuihizp2u3btNHz4cL355puO6Sdr1qzRu+++q4EDB+rqq68u41dT2LXXXisPDw/1799fY8eOVUZGhmbMmKE6deooMTHxoo758MMPa968ebrllls0atQodezYUcePH9cXX3yh+Ph4tWvXTnfeeafmzJmjuLg4LVu2TJdffrny8vK0bds2zZkzx3G9jqK4u7vr3//+t0aOHKnu3btr6NChjiVDo6Ki9I9//KM0b0mxHDp0yPEX5+zsbP3666+aPn26goKCLjgN6rHHHtPHH3+sPn366O9//7sCAgL07rvvau/evfr000+dGmmt0K1bN9WuXVvDhw/X3//+d9lsNr3//vtlMvWqpJo3b65GjRrpoYce0qFDh+Tr66tPP/30vL0aw4YNc/xMnj0NqiRjZ9SoUZo2bZp69+6tu+66S8nJyYqPj1erVq2Unp7u2M/b21stW7bU7Nmz1bRpUwUEBKh169bKzc1Vz549deutt6ply5Zyc3PTggULlJSUpCFDhpyzfl9fX0c/Rk5OjsLDw/Xdd98VecZoypQp+u6779S9e3fHss2JiYmaO3eufv755/MunRscHKzHH39ckydP1nXXXacBAwZo+/btev3119W5c2fHe7d06VLdd999uuWWW9S0aVPl5ubq/fffl6urq6Mp/umnn9aPP/6ofv36qX79+kpOTtbrr7+uevXqOS1SUJSOHTvq+++/17Rp01S3bl01aNBA0dHRuv766/X+++/Lz89PLVu21MqVK/X99987lgEv0LJlS/Xo0UMdO3ZUQECA1q1bp3nz5um+++4r8vm8vb311Vdf6ZprrlGfPn30ww8/qHXr1nr33Xf1+uuv68Ybb1SjRo108uRJzZgxQ76+vo4/AgFVyqVcggqoDkqyFOmnn35qrrjiClOzZk1Ts2ZN07x5czNu3Dizfft2xz7du3c3rVq1KvLxRS03a4wxOTk5ZvLkyaZBgwbG3d3dREREmMcff9xpGURj8pc4LWpJxYLlOc9eCvJcr23ixImFloX84osvTNu2bY2Xl5eJiooy//73v80777xjJJm9e/desIailuU8duyYue+++0x4eLjx8PAw9erVM8OHD3dajjU7O9v8+9//Nq1atTKenp6mdu3apmPHjmby5MkmLS2t8Jt4ltmzZ5sOHToYT09PExAQYG6//XZz8ODBYr0PRSnuvmcvN+vi4mLq1Kljhg4danbt2nXB5zHGmN27d5tBgwYZf39/4+XlZbp06WK++uorp33O9b0tWNK1YAnV4jjXcrPnGq+//PKL6dq1q/H29jZ169Y1jzzyiPn2228LLQV7ruVmi1piVUUsyXq2opab3bp1q4mNjTU+Pj4mKCjI3H333ebXX38953uQmJhoXF1dTdOmTc/5PMUZO8YY88EHH5iGDRsaDw8P0759e/Ptt98Wes3GGLNixQrTsWNH4+Hh4XidKSkpZty4caZ58+amZs2axs/Pz0RHR5s5c+ac9z0wxpiDBw+aG2+80fj7+xs/Pz9zyy23mMOHDxf5Hu7fv98MGzbMBAcHG09PT9OwYUMzbtw4xxLFFxrX//3vf03z5s2Nu7u7CQkJMffcc485ceKE4/49e/aYUaNGmUaNGhkvLy8TEBBgrr76avP999879lmyZIm54YYbTN26dY2Hh4epW7euGTp0qNmxY8cFX+u2bdvMVVddZby9vY0kx9KzJ06cMCNHjjRBQUHGx8fH9O7d22zbts3Ur1/faXnaf/3rX6ZLly7G39/feHt7m+bNm5tnn33Wafnton7/pqSkmJYtW5rQ0FCzc+dOs2HDBjN06FATGRlpPD09TZ06dcz1119v1q1bd8HXAFRGNmMs+HMRAACVSEpKisLCwjRhwgQ99dRTVpcDABUSPRYAAFzArFmzlJeXpzvvvNPqUgCgwqLHAgCAc1i6dKm2bt2qZ599VgMHDlRUVJTVJQFAhcVUKAAAzqFHjx6OZVA/+OADhYeHW10SAFRYBAsAAAAApUaPBQAAAIBSo8fiT3a7XYcPH1atWrVKfMVeAAAA4FIyxujkyZOqW7eu5dcqKkCw+NPhw4cVERFhdRkAAABAsR04cED16tWzugxJBAuHWrVqScr/5vj6+lpcDQAAAHBu6enpioiIcHyGrQgIFn8qmP7k6+tLsAAAAEClUJGm8FeMCVkAAAAAKjWCBQAAAIBSI1gAAAAAKDV6LAAAAFCmjDHKyclRXl6e1aVUSq6urnJ3d69Q/RPFwRkLAAAAlJnc3FylpKQoOzvb6lIqrezsbKWkpCg3N9fqUkqEMxYAAAAoE8YYnThxQkFBQZXur+0VTc2aNZWSklKp3kvOWAAAAKBM5OTkyNvbu9J8EK7IbDabvL29lZOTY3UpxUawAAAAQJnIy8uTq6ur1WVUGa6urpWqT4VgAQAAAKDUCBYAAAAASo1gAQAAAKDUCBYAAABAOYiKitIrr7xidRmXDMECAAAA1ZrNZjvv16RJky7quGvXrtWYMWPKttgKjOtYAAAAoFpLTEx0/Hv27NmaMGGCtm/f7tjm4+Pj+LcxRnl5eXJzu/DH6ODg4LIttIKzPFj8+OOPeuGFF7R+/XolJiZqwYIFGjhw4Hkfs3z5co0fP15btmxRRESEnnzySY0YMeKS1AsAAIDiyziTqy1HTlry3K1Ca8nH88Ifd0NDQx3/9vPzk81mc2xbvny5rr76ai1cuFBPPvmkNm/erO+++04REREaP368Vq1apczMTLVo0UJTp05VbGys41hRUVF64IEH9MADD0jKPzMyY8YMff311/r2228VHh6ul156SQMGDCjbF24Ry4NFZmam2rVrp1GjRummm2664P579+5Vv379FBcXpw8//FBLlizR6NGjFRYWpt69e1+CigEAAFBcW46cVNf/+9mS51719ysUXb92mRzrscce04svvqiGDRuqdu3aOnDggPr27atnn31Wnp6eeu+999S/f39t375dkZGR5zzO5MmT9fzzz+uFF17Qq6++qttvv1379+9XQEBAmdRpJcuDRZ8+fdSnT59i7x8fH68GDRropZdekiS1aNFCP//8s15++WWCBVBCe45l6pONh5WTZ7e6FACo8Ho2CdIVDQOtLgMWefrpp9WrVy/H7YCAALVr185x+5lnntGCBQv0xRdf6L777jvncUaMGKGhQ4dKkqZMmaL/+7//05o1a3TdddeVX/GXiOXBoqRWrlzpdIpJknr37u04xVRciYmJTvPpMjIyyqI8oNIwxmjgzLXanGjN6WkAqGw83VwIFhehVWgtrfr7FZY9d1np1KmT0+2MjAxNmjRJX3/9tRITE5Wbm6tTp04pISHhvMdp27at4981a9aUr6+vkpOTy6xOK1W6YHHkyBGFhIQ4bQsJCVF6erpOnTolb2/vYh1n+vTpmjx5cnmUCFQKP+057ggVNT1c5eZis7giAKjYPN1YTPNi+Hi6ldl0JCvVrFnT6fZDDz2kxYsX68UXX1Tjxo3l7e2tQYMGKTs7+7zHcXd3d7pts9lkt1eNmQOVLliUlbFjxzo1ymRkZKh79+4WVgRcWvEr90uS/LzcdGhCL9UsRnMbAADI98svv2jEiBG68cYbJeV/lty3b5+1RVms0n2SCA0NVVJSktO2pKQk+fr6FvtshSSFhYUpLCzMcTs9Pb3MagQquqMZZzTvt8OSpGGdIggVAACUUJMmTTR//nz1799fNptNTz31VJU583CxKt05vZiYGC1ZssRp2+LFixUTE2NRRUDlM2vtAeXkGUnS2Jj6FlcDAEDlM23aNNWuXVvdunVT//791bt3b1122WVWl2UpmzHGWFlARkaGdu3aJUnq0KGDpk2bpquvvloBAQGKjIzU448/rkOHDum9996TlL/cbOvWrTVu3DiNGjVKS5cu1d///nd9/fXXpVoVKj09XX5+fkpLS5Ovr2+ZvDagIrLbjZo+t1S7j2XpigYB+um+y60uCQBQRZw6dUqSSjSLBOd2vvezIn52tfyMxbp169ShQwd16NBBkjR+/Hh16NBBEyZMkJS/etNfu+sbNGigr7/+WosXL1a7du300ksv6a233mKpWaCYluxM0e5jWZKkOM5WAACAMmL5GYuKoiKmPqA83DxrreZvPqLAGu46OKGXvNxdrS4JAFBFcMaibHHGAkCFdTjttD7fkr/4wcgukYQKAABQZggWQDXy9poE5dnzT1KO6RppcTUAAKAqIVgA1USe3WjGqvxrV8Q2CVKTYB+LKwIAVDVubm4XvEAcii87O1tubpVnSfjKUymAUvlmW7IOpJ6WJMV1o2kbAFD23N3d5eLiouPHj8vT01MuLvwN+2LY7XadOXNG7u7uha7UXZERLIBqIn7FPklSaC1PDWgVam0xAIAqq1atWrLb7crJyan2F4y7WG5ubvL29q50wYxgAVQD+49naeG2ZEnSXdGRcnetXL+oAACVi4uLizw9Pa0uA5cYny6AamDG6gQZI9ls0t3RNG0DAICyR7AAqricPLveWp1/kcm+zeuofkANiysCAABVEcECqOI+//2Ikk6ekSTFdYuythgAAFBlESyAKi5+Zf4SsxH+XurTvI7F1QAAgKqKYAFUYTuPZmjJzhRJ0piu9eXqYrO4IgAAUFURLIAq7M1V+b0Vri423UXTNgAAKEcEC6CKOp2Tp5lr8oPFDa1CFObrZXFFAACgKiNYAFXUp78l6lhWjiQpLibK2mIAAECVR7AAqqiCpu1GgTXUs0mQxdUAAICqjmABVEG/J6br573HJUljY+rLhaZtAABQzggWQBU0/c+zFR6uLhrROcLiagAAQHVAsACqmMwzuXpv/UFJ0qC2YQr28bS4IgAAUB0QLIAqZvamw0o/nStJiutW3+JqAABAdUGwAKqYgqbtFiE+uqJBgMXVAACA6oJgAVQh6w+kau2BVElSXEx92Ww0bQMAgEuDYAFUIdNX5Z+t8HZ30Z0d61lcDQAAqE4IFkAVkXYqRx9tOCRJGtI+XLVreFhcEQAAqE4IFkAV8eGGQ8rMzpNE0zYAALj0CBZAFWCMUfzKfZKkDuG+6hzhb2k9AACg+iFYAFXAqv0ntDnxpCQpLiaKpm0AAHDJESyAKqBgidlanm4a2iHc4moAAEB1RLAAKrnjWdmavemwJOn2y8JVy8vN4ooAAEB1RLAAKrl31x7QmVy7JJq2AQCAdQgWQCWW37SdPw2qa/3aalfXz+KKAABAdUWwACqx5buPacfRTEn5V9oGAACwCsECqMTiV+SfrfD3dtet7etaXA0AAKjOCBZAJZV08owW/J4oSRrRuZ683V0trggAAFRnBAugkpq5JkE5eUaSNLYr06AAAIC1CBZAJWS3G01flT8NqnujQDUPqWVxRQAAoLojWACV0Hc7jmrf8VOSaNoGAAAVA8ECqITiV+yTJAX7eOjGNqHWFgMAACCCBVDpHEw9pS+3JkmSRnWOlKcbTdsAAMB6BAugknlrdYLs+T3bGhMTaW0xAAAAfyJYAJVIbp5dM1YlSJJ6NwtWw8CaFlcEAACQj2ABVCJf/5Gsw+mnJdG0DQAAKhaCBVCJxK/cJ0mq6+ul61uGWFsMAADAXxAsgEpiz7FMfbv9qCRpdHSk3Fz58QUAABUHn0yASmLGqgQZI7nY8oMFAABARUKwACqB7Fy73l6T37R9fcsQRdT2trgiAAAAZwQLoBJYsDlRRzOyJdG0DQAAKiaCBVAJxK/cL0mKCvDWtc3qWFwNAABAYQQLoILblnRSy3cfkySN6Vpfri42iysCAAAojGABVHBv/nlBPDcXm0Z1oWkbAABUTAQLoAI7lZOnWWsPSJJubBOqkFqeFlcEAABQNIIFUIHN/fWwTpzKkSTFxURZWwwAAMB5ECyACix+RX7TdtPgmrq6caDF1QAAAJwbwQKooH49nKaV+09IksbG1JfNRtM2AACouAgWQAU1/c8lZj3dXDS8U4TF1QAAAJwfwQKogDLO5OqD9YckSbe2q6vAmh4WVwQAAHB+BAugAvp44yGdPJMriSttAwCAyoFgAVQwxhi9sWKfJKl1aC3FRNW2tiAAAIBiIFgAFcy6A2naeChdUv7ZCpq2AQBAZUCwACqY+JX7JEk1PFx1R8d61hYDAABQTAQLoAJJPZWjjzfmN23f1iFcft7uFlcEAABQPAQLoAJ5f91BncqxS6JpGwAAVC4EC6CCMMY4pkF1ivBTxwh/S+sBAAAoCYIFUEH8sve4tiZlSJLiYqKsLQYAAKCECBZABRH/55W2fb3cNKR9XYurAQAAKBmCBVABpGSc0dxfEyVJd3asp5qebhZXBAAAUDIEC6ACmLX2oLLz8pu2x9K0DQAAKiGCBWAxu91o+qr8aVCXR9VWmzBfiysCAAAoOYIFYLGlu1K0KyVTkhTXLcraYgAAAC4SwQKwWEHTdkANdw1qG2ZxNQAAABeHYAFYKDH9tD7//YgkaWTnCHm5u1pcEQAAwMUhWAAWemdNgnLtRpI0hqZtAABQiVWIYPHaa68pKipKXl5eio6O1po1a867/yuvvKJmzZrJ29tbERER+sc//qHTp09fomqBspFnN3pzVYIk6ZrGQWoa7GNxRQAAABfP8mAxe/ZsjR8/XhMnTtSGDRvUrl079e7dW8nJyUXu/9FHH+mxxx7TxIkT9ccff+jtt9/W7Nmz9cQTT1ziyoHSWbQtWQknTkmS4rpxtgIAAFRull+Fa9q0abr77rs1cuRISVJ8fLy+/vprvfPOO3rssccK7b9ixQpdfvnluu222yRJUVFRGjp0qFavXl2i501MTFRiYqLjdkZGRileBVByBU3bIbU8dUOrUIurAQAAKB1Lz1hkZ2dr/fr1io2NdWxzcXFRbGysVq5cWeRjunXrpvXr1zumS+3Zs0cLFy5U3759S/Tc06dPV8eOHR1f3bt3v/gXApTQ/uNZ+vqPJEnSXV0i5OFm+clDAACAUrH0jEVKSory8vIUEhLitD0kJETbtm0r8jG33XabUlJSdMUVV8gYo9zcXMXFxZV4KtTYsWM1YMAAx+2MjAzCBS6Zt1YnyBjJZpPu7so0KAAAUPlZPhWqpJYvX64pU6bo9ddfV3R0tHbt2qX7779fzzzzjJ566qliHycsLExhYf+7ZkB6enp5lAsUkpNn11ur85u2+zSvo6iAGhZXBAAAUHqWBougoCC5uroqKSnJaXtSUpJCQ4uec/7UU0/pzjvv1OjRoyVJbdq0UWZmpsaMGaN//vOfcnFhSgkqti+3JOnIyTOSpDiWmAUAAFWEpZ/CPTw81LFjRy1ZssSxzW63a8mSJYqJiSnyMVlZWYXCg6tr/kXFjDHlVyxQRuJX7pMk1fPzUp/mdawtBgAAoIxYPhVq/PjxGj58uDp16qQuXbrolVdeUWZmpmOVqGHDhik8PFxTp06VJPXv31/Tpk1Thw4dHFOhnnrqKfXv398RMICKaldKphbvSJGU31vh5soZNgAAUDVYHiwGDx6so0ePasKECTpy5Ijat2+vRYsWORq6ExISnM5QPPnkk7LZbHryySd16NAhBQcHq3///nr22WeteglAsb355xKzri423RUdYXE1AAAAZcdmmD8kKb9528/PT2lpafL19bW6HFRBZ3LzFD55sY5l5ejGNqGaP6Kz1SUBAIBKqiJ+dmUeBnCJfPpboo5l5UiiaRsAAFQ9BAvgEim40nbDwBqKbRJscTUAAABli2ABXAJbj5zUT3uOS5LGdq0vFxebxRUBAACULYIFcAlMX5V/tsLd1aaRXWjaBgAAVQ/BAihnWdm5enftAUnSzW3CFOzjaXFFAAAAZY9gAZSz2ZsOK+10riQprhtN2wAAoGoiWADlrKBpu3kdH13VMNDiagAAAMoHwQIoRxsOpmpNQqqk/CVmbTaatgEAQNVEsADK0fQ/z1Z4ubloWKd6FlcDAABQfggWQDk5eTpXH208JEka0iFctWt4WFwRAABA+SFYAOXkww0HlXEmTxJX2gYAAFUfwQIoB8YYR9N2u7q+6hLpb21BAAAA5YxgAZSD1Qmp+vVwuiSatgEAQPVAsADKQfyKfZIkH09X3X4ZTdsAAKDqI1gAZex4VrZmbzosSbr9snqq5eVmcUUAAADlj2ABlLH31h3U6Vy7JGlsV5q2AQBA9UCwAMqQMcYxDSo60l8d6vlZWxAAAMAlQrAAytCPe45p+9FMSVJcTJS1xQAAAFxCBAugDMWvyF9i1t/bXbe2D7O4GgAAgEuHYAGUkeSTZ/Tp5kRJ0rBO9VTDg6ZtAABQfRAsgDIyc+0B5eQZSTRtAwCA6odgAZQBu91o+p9X2r6qYYBahtayuCIAAIBLi2ABlIHFO45q7/EsSTRtAwCA6olgAZSB+D/PVgTV9NBNbUMtrgYAAODSI1gApXQo7ZS+3JokSRrVJUKebq4WVwQAAHDpESyAUnp79QHl2fObtsfQtA0AAKopggVQCrl5ds1YlT8NqlfTIDUKqmlxRQAAANYgWAClsPCPZB1MOy2Jpm0AAFC9ESyAUiho2g7z9VT/ViEWVwMAAGAdggVwkfYey9Ki7cmSpNHRkXJ35ccJAABUX3wSAi7SjNX7ZYzkYssPFgAAANUZwQK4CNm5dr29OkGS1K9FiCJr17C4IgAAAGsRLICL8PmWI0rOyJYkxXVjiVkAAACCBXAR4lfkN21H1vZW72Z1LK4GAADAegQLoIS2J2do6a4USdKYrpFydbFZXBEAAID1CBZACb355wXx3FxsGtWFpm0AAACJYAGUyKmcPM1ae0CSNLB1qMJ8vSyuCAAAoGIgWAAlMO/XwzqelSNJiouhaRsAAKAAwQIogYIrbTcJqqmrGwdZXA0AAEDFQbAAimlzYrpW7DshSRobU18uNG0DAAA4ECyAYpr+59kKTzcXDe9Uz+JqAAAAKhaCBVAMGWdy9d66g5KkQW3DFOTjaXFFAAAAFQvBAiiGTzYe0skzuZJo2gYAACgKwQIohoKm7VahtXR5gwCLqwEAAKh4CBbABaw7kKr1B9Mk5Z+tsNlo2gYAADgbwQK4gPgV+Wcrani46s6ONG0DAAAUhWABnEfaqRx9vOmQJGlo+3D5ebtbXBEAAEDFRLAAzuOD9QeVlZ0nSYrrRtM2AADAuRAsgHMwxjiati+r56dOEf7WFgQAAFCBESyAc1ix74R+P3JSEkvMAgAAXAjBAjiH+JX7JEm1PN00tEO4tcUAAABUcAQLoAgpGWc099dESdKdHevJx9PN4ooAAAAqNoIFUIR31x3UmVy7JGks06AAAAAuiGABnMUYo+l/Nm13i6qttnV9La4IAACg4iNYAGdZtuuYdqZkSqJpGwAAoLgIFsBZCpq2A2q4a1C7utYWAwAAUEkQLIC/OJJ+Wgs2H5EkDe8UIW93V4srAgAAqBwIFsBfvLPmgHLtRhJN2wAAACVBsAD+lGc3enNVftP21Y0D1ayOj8UVAQAAVB4EC+BP325P1v4TpyRJcTFR1hYDAABQyRAsgD/Fr8g/W1HHx0MDW4daXA0AAEDlQrAAJB04cUpf/5EkSborOlIebvxoAAAAlASfngBJb61OkN1INpt0dzRN2wAAACVFsEC1l5Nn14zV+dOgejcLVoPAGhZXBAAAUPkQLFDtfbU1SYnpZyTRtA0AAHCxCBao9gqatsP9vNSvRR2LqwEAAKicCBao1nanZOq7HUclSXdHR8rNlR8JAACAi8GnKFRrBRfEc3WxaXTXSIurAQAAqLwIFqi2zuTm6Z01ByRJ/VuGKNzP2+KKAAAAKi+CBaqtBZuPKCUzW5IUF8MSswAAAKVRIYLFa6+9pqioKHl5eSk6Olpr1qw57/6pqakaN26cwsLC5OnpqaZNm2rhwoWXqFpUFfEr86dBNQiooV5Ngy2uBgAAoHJzs7qA2bNna/z48YqPj1d0dLReeeUV9e7dW9u3b1edOoVX6MnOzlavXr1Up04dzZs3T+Hh4dq/f7/8/f0vffGotP5IOqkfdh+TJI3pGikXF5vFFQEAAFRulgeLadOm6e6779bIkSMlSfHx8fr666/1zjvv6LHHHiu0/zvvvKPjx49rxYoVcnd3lyRFRUVdypJRBUz/82yFu6tNI7vQtA0AAFBalk6Fys7O1vr16xUbG+vY5uLiotjYWK1cubLIx3zxxReKiYnRuHHjFBISotatW2vKlCnKy8sr0XMnJiZqw4YNjq9NmzaV5qWgEsnKztW76w5Kkm5qE6aQWp4WVwQAAFD5WXrGIiUlRXl5eQoJCXHaHhISom3bthX5mD179mjp0qW6/fbbtXDhQu3atUv33nuvcnJyNHHixGI/9/Tp0zV58uRS1Y/Kac6mRKWeypFE0zYAAEBZsXwqVEnZ7XbVqVNHb775plxdXdWxY0cdOnRIL7zwQomCxdixYzVgwADH7YyMDHXv3r08SkYFE79ynySpWXBNdW8UaG0xAAAAVYSlwSIoKEiurq5KSkpy2p6UlKTQ0NAiHxMWFiZ3d3e5uro6trVo0UJHjhxRdna2PDw8ivXcYWFhCgsLc9xOT0+/iFeAymbToTStTkiVJMV1i5LNRtM2AABAWbC0x8LDw0MdO3bUkiVLHNvsdruWLFmimJiYIh9z+eWXa9euXbLb7Y5tO3bsUFhYWLFDBaqvgqZtLzcXDetUz+JqAAAAqg7Lr2Mxfvx4zZgxQ++++67++OMP3XPPPcrMzHSsEjVs2DA9/vjjjv3vueceHT9+XPfff7927Nihr7/+WlOmTNG4ceOsegmoJE6eztUHG/Kbtm9tX1cBNQiiAAAAZcXyHovBgwfr6NGjmjBhgo4cOaL27dtr0aJFjobuhIQEubj8L/9ERETo22+/1T/+8Q+1bdtW4eHhuv/++/Xoo49a9RJQSXy08aAyzuSvHkbTNgAAQNmyGWOM1UVUBOnp6fLz81NaWpp8fX2tLgdlzBijDtN+1K+H09U2zFebHryK/goAAFBpVcTPrpZPhQIuhTUJqfr1cH6Dfly3+oQKAACAMkawQLUQ/2fTdk0PV91+WbjF1QAAAFQ9JQ4W8fHxLM2KSuVEVrZmbzokSbr9snD5erlbXBEAAEDVU+JgMX78eIWFhWnYsGH64YcfyqMmoEy9v/6gTuXkL08cFxNlbTEAAABVVImDxeHDh/X8889r69atuvrqq9W4cWNNmTJFhw4dKo/6gFIxxjimQXWO8FeHen4WVwQAAFA1lThY+Pv7a9y4cVq3bp02bdqk66+/Xq+88oqioqLUr18/ffrpp8rJySmPWoES+2nPcf2RlCGJJWYBAADKU6mat9u2batXXnlFmzZt0uWXX65vvvlGt9xyi8LDwzVx4kSdOnWqrOoELkrB2Qo/LzcNbl/X4moAAACqrosOFsYYffPNNxo0aJAaNmyobdu26eGHH9aKFSsUFxenV199VXfccUdZ1gqUSPLJM5r322FJ0rBOEarpafn1IAEAAKqsEn/S2r17t9555x299957Onz4sHr16qUPP/xQN9xwg9zc8g/XtWtXderUSUOGDCnzgoHimrX2gHLy8q//OJZpUAAAAOWqxMGiSZMmCg8P18iRI3XXXXepfv2iP7A1b95c0dHRpS4QuBh2u9H0VfnToK5sGKBWobUsrggAAKBqK3Gw+OKLL9S3b1+5uJx/FlXTpk21bNmyiy4MKI0lO1O051iWJJq2AQAALoUSB4vrr7++POoAylT8yn2SpKCaHrq5bZi1xQAAAFQDJW7eHjVqlAYPHlzkfUOGDNGYMWNKXRRQGofTTuvzLUmSpBGdI+Tp5mpxRQAAAFVfiYPF4sWLddNNNxV5380336xvv/221EUBpfH2mgTl2fObtsd0jbS4GgAAgOqhxMHi6NGjCg4OLvK+wMBAJSUllboo4GLl5tn15p/XrohtEqQmwT4WVwQAAFA9lDhYhIeHa/Xq1UXet3r1aoWFMZ8d1vlmW7IOpp2WJMV1o2kbAADgUilxsBg6dKieffZZzZkzx2n73LlzNWXKFN12221lVhxQUgVX2g6t5akBrUItrgYAAKD6KHGwmDBhgnr06KEhQ4aoVq1aatq0qWrVqqUhQ4aoe/fumjhxYnnUCVzQ/uNZ+mZbsiRpdHSk3F0v+sLyAAAAKKESLzfr4eGhr776SosXL9bSpUt17NgxBQYGKjY2Vj179iyPGoFimbE6QcZILjbpbpq2AQAALqkSB4sCvXr1Uq9evcqyFuCi5eTZ9dbqBElSn+Z1FFm7hsUVAQAAVC8XHSwkKSsrS6dPny60PSAgoDSHBUrs89+PKOnkGUlSXLcoa4sBAACohkocLIwx+te//qXp06crMTGxyH3y8vJKXRhQEgVN2xH+XurTvI7F1QAAAFQ/Je5uffnllzVt2jSNGzdOxhj985//1IQJE9S0aVNFRUVpxowZ5VEncE47jmZoyc4USdKYrvXl6mKzuCIAAIDqp8TB4u2339bkyZP1yCOPSJIGDhyoiRMnasuWLWrRooV27dpV5kUC51NwQTxXF5vuiqZpGwAAwAolDhb79u1T+/bt5erqKnd3d6WmpuYfyMVF9957r2bNmlXGJQLndjonT7PWHpAkDWwdqjBfL4srAgAAqJ5KHCwCAwOVkZEhSYqMjNSGDRsc96WkpCgrK6vsqgMu4NPfEnUsK0eSFBfDlbYBAACsUuLm7csvv1xr165V3759ddttt2nSpEk6cuSI3N3dNWPGDK5lgUuqoGm7UWANXdM4yOJqAAAAqq8SB4tJkybp0KFDkqQnnnhCqamp+vjjj3Xq1Cn16tVLr776apkXCRTl98R0/bz3uCRpbEx9udC0DQAAYBmbMcYUd2djjE6cOKGaNWvK09OzPOu65NLT0+Xn56e0tDT5+vpaXQ6K4W/zN+u/v+yTh6uLDk6IVbBP1RqTAAAA51IRP7uWqMciJydHderU0ffff19e9QDFknkmV++tPyhJGtQ2jFABAABgsRIFCw8PD9WrV48L4MFyn2w6rPTTuZKkuG40bQMAAFitxKtCjRs3TtOmTdPp06fLox6gWOJX7pMktQzx0RUNAqwtBgAAACVv3k5ISNCOHTsUGRmpHj16KCQkRDbb/5pmbTab/vOf/5RpkcBfrT+QqnUH0iRJcTFRTuMPAAAA1ihxsPjqq6/k6ekpT09PrV27ttD9BAuUt+mr8peY9XZ30Z2d6llcDQAAAKSLCBZ79+4tjzqAYkk7laOPNuQvdzykfbj8vd0trggAAADSRfRYAFb6cMMhZWbnLx5A0zYAAEDFUeIzFu+9994F9xk2bNhFFQOcjzHG0bTdIdxXnSP8La0HAAAA/1PiYDFixIgit/+1gZZggfKwct8JbU48KYmmbQAAgIqmxMHixIkTRW779ttv9d///lcfffRRmRQGnC1+ZX7Tdi1PNw3tEG5xNQAAAPirEgcLPz+/IreNHTtWp0+f1iOPPKJvvvmmTIoDChzPytacXw9Lku7oGK5aXiUeugAAAChHZdq83apVK/30009leUhAkvTu2gM6k2uXJI2NoWkbAACgoimzYJGVlaUZM2YoPJwpKihb+U3b+dOgutavrXZ1C581AwAAgLVKPJ+kTZs2hZpms7OzdfDgQZ06dapYq0YBJbF89zHtOJopSYrjbAUAAECFVOJg0bFjx0LBwsvLS/Xq1dNNN92kFi1alFlxgCTFr8g/W+Hv7a5b29e1uBoAAAAUpcTBYtasWeVQBlC0pJNnNH9zoiRpROd68nZ3tbgiAAAAFKXEPRYnT55UYmJikfclJiYqIyOj1EUBBd5Zk6Bcu5Ekje3KNCgAAICKqsRnLEaPHq1atWrprbfeKnTfxIkTlZGRwbUsUCbsdqM3V+VPg+rRKFDNQ2pZXBEAAADOpcRnLH788Uf169evyPv69u2rH374odRFAZL03Y6j2nf8lCSatgEAACq6EgeLEydOqFatov9yXLNmTR07dqzURQGSFL9inyQp2MdDN7YJs7YYAAAAnFeJg0XDhg31/fffF3nfkiVLFBUVVdqaAB1MPaUvtyZJkkZ1jpSHW5leyxEAAABlrMSf1kaPHq1p06bp+eefV0pKiiQpJSVFL7zwgl5++WXdfffdZV4kqp+3Vifoz55tjYmJtLYYAAAAXFCJm7f/8Y9/aPfu3Xr88cf1+OOPy83NTbm5uZKkuLg4Pfjgg2VeJKqX3Dy7ZqxKkCT1bhashoE1La4IAAAAF1LiYGGz2fTaa6/pgQce0NKlS3Xs2DEFBgbqmmuuUZMmTcqjRlQzX21N0uH005Jo2gYAAKgsShwsCjRp0oQggXIRvzJ/idm6vl66vmWIxdUAAACgOErcYzF79my98MILRd734osvau7cuaUuCtXXnmOZ+m7HUUnS3V0j5eZK0zYAAEBlUOJPbc8995w8PT2LvM/b21vPPfdcqYtC9TVjVYKMkVxs0uhomrYBAAAqixIHix07dqh169ZF3teyZUvt2LGj1EWhesrOtevtNflN29e3DFE9f2+LKwIAAEBxlThYeHl5KSkpqcj7EhMT5eZ20W0bqOYWbE7U0YxsSTRtAwAAVDYlDhbdu3fXc889p8zMTKftmZmZev7559WjR4+yqg3VTEHTdlSAt65tVsfiagAAAFASJT69MGXKFMXExKhRo0YaNGiQ6tatq8OHD2vevHk6c+aMPvnkk/KoE1XctqSTWr77mCRpTNf6cnWxWVwRAAAASqLEwaJ58+Zau3atJk6cqE8//dRxHYtevXpp0qRJcnFhFR+U3PRV+Wcr3FxsGtWFpm0AAIDK5qIaIho3bqwPP/zQcfvo0aOaM2eOhg0bplWrVikvL6/MCkTVdyonT++uPShJuqlNmEJqFb3qGAAAACqui+60zsrK0oIFC/TRRx/p+++/V25urtq3b6+XX365LOtDNTD318M6cSpHkhTXjaZtAACAyqhEwSIvL0+LFi3SRx99pC+++EKZmZkKCwtTbm6uPv74Y916663lVSeqsPgV+dOgmgbXVI9GgRZXAwAAgItRrGDxyy+/6KOPPtLcuXOVkpKiwMBA3XHHHbrtttvUunVrBQYGKjQ0tLxrRRX06+E0rdx/QpI0Nqa+bDaatgEAACqjYgWLK6+8UjabTVdffbXGjx+va6+91nG9irS0tHItEFXb9D+XmPV0c9HwThEWVwMAAICLVaxg0aZNG23evFk//PCDXF1dlZKSohtvvFG1atUq7/pQhZ08nav31+c3bd/arq4Ca3pYXBEAAAAuVrHWhv3111/1+++/6+GHH9bOnTs1YsQIhYaG6tZbb9Xnn3/O9BVclI83HlLGmfwVxLjSNgAAQOVW7ItOtGzZUlOmTNGePXv0008/acSIEfrhhx80YsQISdJ//vMf/fjjj+VVJ6oYY4ziV+6TJLUJq6WYqNrWFgQAAIBSuair2V1++eV67bXXdPjwYX311Ve67bbbtHjxYl199dVq2LBhWdeIKmjdgTRtPJQuSYqLieKsFwAAQCVXqstku7q6qm/fvnr//feVlJSkDz74QK1bty6r2lCFFZytqOnhqjs6hltbDAAAAEqtVMHir7y9vTV06FB98cUXF/X41157TVFRUfLy8lJ0dLTWrFlTrMd98sknstlsGjhw4EU9Ly691FM5+njjIUnS0A7h8vVyt7giAAAAlFaZBYvSmD17tsaPH6+JEydqw4YNateunXr37q3k5OTzPm7fvn166KGHdOWVV16iSlEW3l93UKdy7JJo2gYAAKgqKkSwmDZtmu6++26NHDlSLVu2VHx8vGrUqKF33nnnnI/Jy8vT7bffrsmTJ19UX0diYqI2bNjg+Nq0aVMpXgGK669N250i/NQxwt/SegAAAFA2LA8W2dnZWr9+vWJjYx3bXFxcFBsbq5UrV57zcU8//bTq1Kmju+6666Ked/r06erYsaPjq3v37hd1HJTMz3uPa2tShqT8pm0AAABUDcW6QF55SklJUV5enkJCQpy2h4SEaNu2bUU+5ueff9bbb79dqrMMY8eO1YABAxy3MzIyCBeXQPyK/Ctt+3q5aUj7uhZXAwAAgLJiebAoqZMnT+rOO+/UjBkzFBQUdNHHCQsLU1hYmON2enp6WZSH80jJOKN5vyVKkoZ1rKeanpVu+AEAAOAcLP9kFxQUJFdXVyUlJTltT0pKUmhoaKH9d+/erX379ql///6ObXZ7fiOwm5ubtm/frkaNGpVv0bgos9YeVHZe/vdqLE3bAAAAVYrlPRYeHh7q2LGjlixZ4thmt9u1ZMkSxcTEFNq/efPm2rx5szZt2uT4GjBggK6++mpt2rRJERERl7J8FJPdbjR9Vf40qMujaqt1mK/FFQEAAKAsWX7GQpLGjx+v4cOHq1OnTurSpYteeeUVZWZmauTIkZKkYcOGKTw8XFOnTpWXl1ehi/D5+/tLEhfnq8CW7krRrpRMSVJctyhriwEAAECZqxDBYvDgwTp69KgmTJigI0eOqH379lq0aJGjoTshIUEuLpafXEEpxK/MP1sRUMNdg9qGXWBvAAAAVDY2Y4yxuoiKID09XX5+fkpLS5OvL9N0ytLhtNOK/Nf3yrMbPdi9oV4c0MrqkgAAACq1ivjZldMAKHfvrElQnj0/v46haRsAAKBKIligXOXZjWasTpAk9WwSpKbBPhZXBAAAgPJAsEC5WrQtWQknTkmS4jhbAQAAUGURLFCuCpq2Q2t56obWha9LAgAAgKqBYIFys/94lr7+I//Ch6O6RMjdleEGAABQVfFJD+XmrdUJMkay2aS7uzINCgAAoCojWKBc5OTZ9dafTdt9mtdRVEANiysCAABAeSJYoFx8seWIjpw8I4mmbQAAgOqAYIFyEb8iv2k7wt9LfVuEWFwNAAAAyhvBAmVuV0qmvt+ZIim/t8LVxWZxRQAAAChvBAuUuTf/XGLW1cWmu7pEWlwNAAAALgWCBcrUmdw8vbMmv2l7QKsQ1fXzsrgiAAAAXAoEC5SpT39L1LGsHEk0bQMAAFQnBAuUqYIrbTcMrKHYJsEWVwMAAIBLhWCBMrPlyEn9tOe4JGls1/pyoWkbAACg2iBYoMxM//NshburTSO7RFhcDQAAAC4lggXKRFZ2rt5bd0CSNKhtXQX7eFpcEQAAAC4lggXKxOxNh5V2OlcSTdsAAADVEcECZaKgabtFiI+ubBhgcTUAAAC41AgWKLUNB1O1JiFVUn7Tts1G0zYAAEB1Q7BAqRU0bXu5uWhYp3oWVwMAAAArECxQKumnc/ThhkOSpCEdwlW7hofFFQEAAMAKBAuUyocbDikzO08STdsAAADVGcECF80Yo/gV+dOg2tf1VZdIf2sLAgAAgGUIFrhoqxNS9VtiuiQprhtN2wAAANUZwQIXLX7FPkmSj6erbutA0zYAAEB1RrDARTmela3Zmw5Lkm6/rJ5qeblZXBEAAACsRLDARXlv3UGdzrVLyr92BQAAAKo3ggVKLL9pe58kKTrSXx3q+VlbEAAAACxHsECJ/bD7mLYfzZQkxcVEWVsMAAAAKgSCBUos/s8rbft7u+vW9mEWVwMAAICKgGCBEkk+eUbzNydKkoZ3qqcaHjRtAwAAgGCBEpq59oBy8owkaSxX2gYAAMCfCBYoNrvdaPqf06CuahigFiG1LK4IAAAAFQXBAsW2eMdR7T2eJYmmbQAAADgjWKDYCpq2g2p66Ka2oRZXAwAAgIqEYIFiOZh6Sl9uTZIkjeoSIU83V4srAgAAQEVCsECxvL06QXn2/KbtMVxpGwAAAGchWOCCcvPsemt1giTp2qbBahRU0+KKAAAAUNEQLHBBC/9I1sG005KkuG6crQAAAEBhBAtcUEHTdl1fL13fMsTiagAAAFARESxwXnuPZWnR9mRJ0l3REXJ3ZcgAAACgMD4l4rxmrN4vYyQXmzQ6OtLqcgAAAFBBESxwTtm5dr39Z9N2vxYhiqxdw+KKAAAAUFERLHBOn/1+RMkZ2ZJo2gYAAMD5ESxwTvEr90mS6tf2Vu9mdawtBgAAABUawQJF2p6coWW7jknKvyCeq4vN4ooAAABQkREsUKQ3V+UvMevmYtOoLhEWVwMAAICKjmCBQk7l5GnW2gOSpIGtQxXq62VxRQAAAKjoCBYoZN6vh3U8K0eSFBdD0zYAAAAujGCBQgqutN0kqKaubhxkcTUAAACoDAgWcPLb4XSt2HdCkjQ2pr5caNoGAABAMRAs4GT6n2crPN1cNLxTPYurAQAAQGVBsIBDxplcvb/+oCTplnZhCvLxtLgiAAAAVBYECzh8svGQTp7JlSTFxURZWwwAAAAqFYIFHAqatluF1lK3qNoWVwMAAIDKhGABSdK6A6lafzBNUv4SszYbTdsAAAAoPoIFJEnxK/LPVtTwcNWdHWnaBgAAQMkQLKDUUzn6eNMhSdLQ9uHy83a3uCIAAABUNgQL6IP1B5WVnSdJiuvGlbYBAABQcgSLas4Y47h2Rcd6fuoU4W9tQQAAAKiUCBbV3Ip9J/T7kZOS8pu2AQAAgItBsKjm4lfukyT5erlpSIdwa4sBAABApUWwqMZSMs5o7q+JkqQ7LqsnH083iysCAABAZUWwqMbeXXdQZ3LtkqSxTIMCAABAKRAsqim7/X9N292iaqttXV+LKwIAAEBlRrCoppbtStHOlExJNG0DAACg9AgW1VT8n2crAmq4a1C7uhZXAwAAgMqOYFENHUk/rc9+PyJJGtE5Qt7urhZXBAAAgMqOYFENvbPmgHLtRpI0pivToAAAAFB6FSZYvPbaa4qKipKXl5eio6O1Zs2ac+47Y8YMXXnllapdu7Zq166t2NjY8+6P/8mzG725Kn8a1NWNA9Wsjo/FFQEAAKAqqBDBYvbs2Ro/frwmTpyoDRs2qF27durdu7eSk5OL3H/58uUaOnSoli1bppUrVyoiIkLXXnutDh06dIkrr3y+3Z6s/SdOSZLiYqKsLQYAAABVhs0YY6wuIjo6Wp07d9Z///tfSZLdbldERIT+9re/6bHHHrvg4/Py8lS7dm3997//1bBhwy6qhvT0dPn5+SktLU2+vlV36dUBb6/Rl1uTVMfHQwee6iUPtwqRLQEAAFACFfGzq+WXWs7Oztb69ev1+OOPO7a5uLgoNjZWK1euLNYxsrKylJOTo4CAgGI/b2JiohITEx23MzIyil90JZVwIktf/5EkSborOpJQAQAAgDJjebBISUlRXl6eQkJCnLaHhIRo27ZtxTrGo48+qrp16yo2NrbYzzt9+nRNnjy5RLVWdm+tTpDdSDabdHc0TdsAAAAoO5YHi9J67rnn9Mknn2j58uXy8vIq9uPGjh2rAQMGOG5nZGSoe/fu5VFihZCTZ9dbqxMkSdc1q6MGgTUsrggAAABVieXBIigoSK6urkpKSnLanpSUpNDQ0PM+9sUXX9Rzzz2n77//Xm3bti3R84aFhSksLMxxOz09vUSPr2y+2pqkxPQzkrjSNgAAAMqe5ZPsPTw81LFjRy1ZssSxzW63a8mSJYqJiTnn455//nk988wzWrRokTp16nQpSq3U4lfkLzFbz89LfVvUsbgaAAAAVDWWn7GQpPHjx2v48OHq1KmTunTpoldeeUWZmZkaOXKkJGnYsGEKDw/X1KlTJUn//ve/NWHCBH300UeKiorSkSP5V5H28fGRjw/XZTjb7pRMfbfjqCRpdHSk3Fwtz5MAAACoYipEsBg8eLCOHj2qCRMm6MiRI2rfvr0WLVrkaOhOSEiQi8v/Pgy/8cYbys7O1qBBg5yOM3HiRE2aNOlSll4pFFwQz9XFptFdIy2uBgAAAFVRhbiORUVQEdcCLgtncvNU7+nvlZKZrYGtQ7VgZGerSwIAAEApVcTPrsyJqeLm/3ZEKZnZkmjaBgAAQPkhWFRx8Sv3SZIaBNRQr6bB1hYDAACAKotgUYX9kXRSP+45LkkaG1NfLi42iysCAABAVUWwqMKmr8xv2nZ3tWlk5wiLqwEAAEBVRrCoorKyc/XuuoOSpJvahKlOLU+LKwIAAEBVRrCoouZsSlTqqRxJNG0DAACg/BEsqqiCpu1mwTXVvVGgtcUAAACgyiNYVEEbD6ZpdUKqJCmuW5RsNpq2AQAAUL4IFlXQ9D+vtO3l5qJhnepZXA0AAACqA4JFFXPydK4+3JDftD24fV0F1PCwuCIAAABUBwSLKuajjQeVcSZPUv40KAAAAOBSIFhUIcYYvbEifxpU2zBfRUf6W1sQAAAAqg2CRRWyJiFVvx5OlyTFdatP0zYAAAAuGYJFFRL/55W2a3q46vbLwi2uBgAAANUJwaKKOJGVrU82HpIk3X5ZuHy93C2uCAAAANUJwaKKeG/dQZ3OtUuS4mKirC0GAAAA1Q7BogowxjiuXdEl0l8d6vlZXBEAAACqG4JFFfDTnuP6IylDkhQXU9/iagAAAFAdESyqgIKmbT8vNw1uX9fiagAAAFAdESwqueSTZzTvt8OSpGGdIlTDw83iigAAAFAdESwquVlrDygnz0iSxjINCgAAABYhWFRidvv/mravbBigVqG1LK4IAAAA1RXBohL7fudR7TmWJYmmbQAAAFiLYFGJFTRtB9X00M1twyyuBgAAANUZwaKSOpx2Wl9sSZIkjewcIU83V4srAgAAQHVGsKik3l6ToDx7ftP2GKZBAQAAwGIEi0ooN8+uN/+cBhXbJEiNg2paXBEAAACqO4JFJfTNtmQdTDstSYrrxtkKAAAAWI9gUQkVNG2H1vLUgFahFlcDAAAAECwqnX3Hs/TNtmRJ0ujoSLm78i0EAACA9fhUWsnMWLVfxkguNunurpFWlwMAAABIIlhUKjl5dr295oAkqW+LEEXWrmFxRQAAAEA+gkUl8vnvR5R08owkrrQNAACAioVgUYkUNG1H1vbWdc3rWFwNAAAA8D8Ei0pix9EMLdmZIkm6OzpSri42iysCAAAA/odgUUkUXBDP1cWmu6Jp2gYAAEDFQrCoBE7n5Gnm2vym7YGtQxXm62VxRQAAAIAzgkUlMO+3RB3PypFE0zYAAAAqJoJFJRC/Yp8kqXFQTV3TOMjaYgAAAIAiECwquN8T0/XLvhOSpLFd68uFpm0AAABUQASLCm76n03bHq4uGtG5nsXVAAAAAEUjWFRgmWdy9d76g5KkQW3DFOTjaXFFAAAAQNEIFhXYJ5sOK/10riQprhtN2wAAAKi4CBYVWPzKfZKkliE+uqJBgLXFAAAAAOdBsKig1h1I1boDaZKkuJgo2Ww0bQMAAKDiIlhUUAVN297uLrqzE03bAAAAqNgIFhVQ2qkcfbzxkCRpaIdw+Xu7W1wRAAAAcH4Eiwroww2HlJmdJyl/GhQAAABQ0REsKhhjjKNpu0O4rzpF+FlbEAAAAFAMBIsKZuW+E9qceFISTdsAAACoPAgWFUz8n03btTzdNLRDuMXVAAAAAMVDsKhAjmVma86vhyVJd3QMVy0vN4srAgAAAIqHYFGBvLvugM7k2iVJY2O40jYAAAAqD4JFBWGMcVy7IqZ+bbWrS9M2AAAAKg+CRQWxfPcx7TiaKUmK68bZCgAAAFQuBIsKIn5F/tmK2t7uuqVdXYurAQAAAEqGYFEBJJ08o/mbEyVJwzvXk7e7q8UVAQAAACVDsKgA3lmToFy7kSSN7co0KAAAAFQ+BAuL5dmN3lyVPw2qR6NANQ+pZXFFAAAAQMkRLCz23fZk7Tt+SpIUxxKzAAAAqKQIFhYruNJ2sI+HbmwTZnE1AAAAwMUhWFjoYOopfbU1SZJ0V5dIebjx7QAAAEDlxCdZC721OkF2I9ls0t1dI60uBwAAALhoBAuL5ObZNWNVgiTp2qbBahhY0+KKAAAAgItHsLDIV1uTdDj9tCSatgEAAFD5ESwsUtC0XdfXS9e3DLG4GgAAAKB0CBYW2HMsU99uPyopv7fCzZVvAwAAACo3PtFa4M2V+b0VLjZpdDRN2wAAAKj8CBaXWHauXe+szQ8W/VuGqJ6/t8UVAQAAAKVHsLjEFmxO1NGMbElSXLcoa4sBAAAAygjB4hIraNqOCvDWtU2DLa4GAAAAKBsVJli89tprioqKkpeXl6Kjo7VmzZrz7j937lw1b95cXl5eatOmjRYuXHiJKr1425JOavnuY5KkMV3ry8XFZnFFAAAAQNmoEMFi9uzZGj9+vCZOnKgNGzaoXbt26t27t5KTk4vcf8WKFRo6dKjuuusubdy4UQMHDtTAgQP1+++/X+LKS2b6qvyzFW4uNo3qQtM2AAAAqg6bMcZYXUR0dLQ6d+6s//73v5Iku92uiIgI/e1vf9Njjz1WaP/BgwcrMzNTX331lWNb165d1b59e8XHxxfrORMTE5WYmOi4nZGRoe7duystLU2+vr6lfEWFncrJU93Ji5V6Kke3tqur2cM6lvlzAAAAoHpIT0+Xn59fuX12vRiWn7HIzs7W+vXrFRsb69jm4uKi2NhYrVy5ssjHrFy50ml/Serdu/c59y/K9OnT1bFjR8dX9+7dL+4FFNOcTYeVeipHkhTXjSttAwAAoGpxs7qAlJQU5eXlKSTE+erTISEh2rZtW5GPOXLkSJH7HzlypNjPO3bsWA0YMMBxu+CMRXnpFOGvu7tGasPBNPVoFFhuzwMAAABYwfJgYZWwsDCFhYU5bqenp5fr87UKraU3b2knu93IZqNpGwAAAFWL5VOhgoKC5OrqqqSkJKftSUlJCg0NLfIxoaGhJdq/ImElKAAAAFRFlgcLDw8PdezYUUuWLHFss9vtWrJkiWJiYop8TExMjNP+krR48eJz7g8AAACgfFWIqVDjx4/X8OHD1alTJ3Xp0kWvvPKKMjMzNXLkSEnSsGHDFB4erqlTp0qS7r//fnXv3l0vvfSS+vXrp08++UTr1q3Tm2++aeXLAAAAAKqtChEsBg8erKNHj2rChAk6cuSI2rdvr0WLFjkatBMSEuTi8r+TK926ddNHH32kJ598Uk888YSaNGmizz77TK1bt7bqJQAAAADVWoW4jkVFUBHXAgYAAACKUhE/u1reYwEAAACg8iNYAAAAACg1ggUAAACAUiNYAAAAACg1ggUAAACAUiNYAAAAACg1ggUAAACAUiNYAAAAACg1ggUAAACAUiNYAAAAACg1ggUAAACAUiNYAAAAACg1N6sLqCiMMZKk9PR0iysBAAAAzq/gM2vBZ9iKgGDxp5MnT0qSIiIiLK4EAAAAKJ7du3frsssus7oMSQQLh7p16+rAgQOqVauWbDZbuTzHpk2b1L17d/3www9q3759uTwHUBKMSVREjEtUNIxJVEQbN25Ujx49ZLfbrS7FgWDxJxcXF9WrV69cn8PHx8fxX19f33J9LqA4GJOoiBiXqGgYk6iIatWqJSn/M2xFUXEqAQAAAFBpESwAAAAAlBrB4hIKCwvTxIkTFRYWZnUpgCTGJComxiUqGsYkKqKKOC5tpiKtUQUAAACgUuKMBQAAAIBSI1gAAAAAKDWCBQAAAIBSI1gAAAAAKDWCBQAAAIBSI1gAAAAAKDWCBQAAAIBSI1gAAAAAKDWCBQAAAIBSI1hcIq+99pqioqLk5eWl6OhorVmzxuqSUElNnTpVnTt3Vq1atVSnTh0NHDhQ27dvd9qnR48estlsTl9xcXFO+yQkJKhfv36qUaOG6tSpo4cffli5ublO+yxfvlyXXXaZPD091bhxY82aNatQPYxtTJo0qdB4a968ueP+06dPa9y4cQoMDJSPj49uvvlmJSUlOR2D8YiyFhUVVWhc2mw2jRs3ThK/J1H+fvzxR/Xv319169aVzWbTZ5995nS/MUYTJkxQWFiYvL29FRsbq507dzrtc/z4cd1+++3y9fWVv7+/7rrrLmVkZDjt89tvv+nKK6+Ul5eXIiIi9PzzzxeqZe7cuWrevLm8vLzUpk0bLVy4sMS1FItBufvkk0+Mh4eHeeedd8yWLVvM3Xffbfz9/U1SUpLVpaES6t27t5k5c6b5/fffzaZNm0zfvn1NZGSkycjIcOzTvXt3c/fdd5vExETHV1pamuP+3Nxc07p1axMbG2s2btxoFi5caIKCgszjjz/u2GfPnj2mRo0aZvz48Wbr1q3m1VdfNa6urmbRokWOfRjbMMaYiRMnmlatWjmNt6NHjzruj4uLMxEREWbJkiVm3bp1pmvXrqZbt26O+xmPKA/JyclOY3Lx4sVGklm2bJkxht+TKH8LFy40//znP838+fONJLNgwQKn+5977jnj5+dnPvvsM/Prr7+aAQMGmAYNGphTp0459rnuuutMu3btzKpVq8xPP/1kGjdubIYOHeq4Py0tzYSEhJjbb7/d/P777+bjjz823t7eZvr06Y59fvnlF+Pq6mqef/55s3XrVvPkk08ad3d3s3nz5hLVUhwEi0ugS5cuZty4cY7beXl5pm7dumbq1KkWVoWqIjk52UgyP/zwg2Nb9+7dzf3333/OxyxcuNC4uLiYI0eOOLa98cYbxtfX15w5c8YYY8wjjzxiWrVq5fS4wYMHm969eztuM7ZhTH6waNeuXZH3paamGnd3dzN37lzHtj/++MNIMitXrjTGMB5xadx///2mUaNGxm63G2P4PYlL6+xgYbfbTWhoqHnhhRcc21JTU42np6f5+OOPjTHGbN261Ugya9eudezzzTffGJvNZg4dOmSMMeb11183tWvXdoxJY4x59NFHTbNmzRy3b731VtOvXz+neqKjo83YsWOLXUtxMRWqnGVnZ2v9+vWKjY11bHNxcVFsbKxWrlxpYWWoKtLS0iRJAQEBTts//PBDBQUFqXXr1nr88ceVlZXluG/lypVq06aNQkJCHNt69+6t9PR0bdmyxbHPX8dtwT4F45axjb/auXOn6tatq4YNG+r2229XQkKCJGn9+vXKyclxGifNmzdXZGSkY5wwHlHesrOz9cEHH2jUqFGy2WyO7fyehFX27t2rI0eOOI0NPz8/RUdHO/1u9Pf3V6dOnRz7xMbGysXFRatXr3bsc9VVV8nDw8OxT+/evbV9+3adOHHCsc/5xmlxaikutxLtjRJLSUlRXl6e0y8mSQoJCdG2bdssqgpVhd1u1wMPPKDLL79crVu3dmy/7bbbVL9+fdWtW1e//fabHn30UW3fvl3z58+XJB05cqTIMVlw3/n2SU9P16lTp3TixAnGNiRJ0dHRmjVrlpo1a6bExERNnjxZV155pX7//XcdOXJEHh4e8vf3d3pMSEjIBcdawX3n24fxiOL47LPPlJqaqhEjRji28XsSVioYQ0WNjb+Orzp16jjd7+bmpoCAAKd9GjRoUOgYBffVrl37nOP0r8e4UC3FRbAAKrFx48bp999/188//+y0fcyYMY5/t2nTRmFhYerZs6d2796tRo0aXeoyUcX16dPH8e+2bdsqOjpa9evX15w5c+Tt7W1hZUC+t99+W3369FHdunUd2/g9CZQ9pkKVs6CgILm6uhZaASUpKUmhoaEWVYWq4L777tNXX32lZcuWqV69eufdNzo6WpK0a9cuSVJoaGiRY7LgvvPt4+vrK29vb8Y2zsnf319NmzbVrl27FBoaquzsbKWmpjrt89dxwnhEedq/f7++//57jR49+rz78XsSl1LB9/98YyM0NFTJyclO9+fm5ur48eNl8vvzr/dfqJbiIliUMw8PD3Xs2FFLlixxbLPb7VqyZIliYmIsrAyVlTFG9913nxYsWKClS5cWOgValE2bNkmSwsLCJEkxMTHavHmz0y+sxYsXy9fXVy1btnTs89dxW7BPwbhlbONcMjIytHv3boWFhaljx45yd3d3Gifbt29XQkKCY5wwHlGeZs6cqTp16qhfv37n3Y/fk7iUGjRooNDQUKexkZ6ertWrVzv9bkxNTdX69esd+yxdulR2u90RhGNiYvTjjz8qJyfHsc/ixYvVrFkz1a5d27HP+cZpcWopthK1euOifPLJJ8bT09PMmjXLbN261YwZM8b4+/s7rTQBFNc999xj/Pz8zPLly52WSczKyjLGGLNr1y7z9NNPm3Xr1pm9e/eazz//3DRs2NBcddVVjmMULKN47bXXmk2bNplFixaZ4ODgIpdRfPjhh80ff/xhXnvttSKXUWRs48EHHzTLly83e/fuNb/88ouJjY01QUFBJjk52RiTv9xsZGSkWbp0qVm3bp2JiYkxMTExjsczHlFe8vLyTGRkpHn00UedtvN7EpfCyZMnzcaNG83GjRuNJDNt2jSzceNGs3//fmNM/hKv/v7+5vPPPze//fabueGGG4pcbrZDhw5m9erV5ueffzZNmjRxWm42NTXVhISEmDvvvNP8/vvv5pNPPjE1atQotNysm5ubefHFF80ff/xhJk6cWORysxeqpTgIFpfIq6++aiIjI42Hh4fp0qWLWbVqldUloZKSVOTXzJkzjTHGJCQkmKuuusoEBAQYT09P07hxY/Pwww87rc9ujDH79u0zffr0Md7e3iYoKMg8+OCDJicnx2mfZcuWmfbt2xsPDw/TsGFDx3P8FWMbgwcPNmFhYcbDw8OEh4ebwYMHm127djnuP3XqlLn33ntN7dq1TY0aNcyNN95oEhMTnY7BeER5+Pbbb40ks337dqft/J7EpbBs2bIi/389fPhwY0z+Mq9PPfWUCQkJMZ6enqZnz56FxuqxY8fM0KFDjY+Pj/H19TUjR440J0+edNrn119/NVdccYXx9PQ04eHh5rnnnitUy5w5c0zTpk2Nh4eHadWqlfn666+d7i9OLcVhM8aYkp3jAAAAAABn9FgAAAAAKDWCBQAAAIBSI1gAAAAAKDWCBQAAAIBSI1gAAAAAKDWCBQAAAIBSI1gAAAAAKDWCBQAAAIBSI1gAQBU0adIk+fj4SJJSU1M1adIkbd261ZJaXnnlFS1cuLDQ9qioKN13330WVAQAKA8ECwCo4lJTUzV58uQKFywWLFighx56yIKKAADlwc3qAgAAlYsxRtnZ2fL09CzVcTp06FBGFQEAKgLOWABAFbZv3z41aNBAknTLLbfIZrPJZrNp3759kqQzZ87oiSeeUP369eXp6akWLVroo48+cjrGiBEj1Lp1ay1cuFDt2rWTp6envvzyS2VmZuq+++5Ts2bNVKNGDUVFRSkuLk5paWmOx0ZFRWn//v167bXXHM89a9Ysx31nT4WaP3++2rdvLy8vL9WtW1fjx4/X6dOnHfcvX75cNptNixcv1m233aZatWqpfv36ev75552Os2XLFvXt21eBgYGqUaOGmjVrVmgfAEDZ4owFAFRhYWFhmj9/vm666SZNmTJFV199tWO7JN166636+eefNXHiRLVo0UILFy7UHXfcodq1a6tPnz6O4xw+fFh///vf9eSTTyoyMlKRkZHKyspSXl6enn32WQUHB+vAgQN69tlnNXDgQC1btkxS/nSnvn376oorrtCDDz4oSWrUqFGRtX7xxRcaNGiQhgwZoueee07btm3TE088oYSEBM2bN89p37i4ON15551asGCBPvvsMz366KNq27atrrvuOklS//79FRISorffflt+fn7atWuXDh48WLZvLgDACcECAKowT09Px5SjJk2aqGvXro77li1bpi+++ELffvutrr32WklSr169lJiYqIkTJzoFixMnTuibb75RdHS00/HfeOMNx79zc3PVoEEDXXHFFdqxY4eaNm2qDh06yNPTUyEhIU7PXZRJkyapa9eujjMm1113nWrUqKGxY8dq8+bNatOmjWPfm2++WZMmTZIk9ezZU19//bXmzZun6667TikpKdq7d6/+85//qH///pLkCFQAgPLDVCgAqKa+++47BQQE6JprrlFubq7jq1evXtq4caPy8vIc+wYGBhYKFZL0/vvvq0OHDvLx8ZG7u7uuuOIKSdKOHTtKVEtGRoY2bdqkQYMGOW0fPHiwJOnnn3922l4QhCTJZrOpRYsWjjMSgYGBql+/vh5//HG9++67nKkAgEuEYAEA1VRKSoqOHz8ud3d3p6/Ro0crNzdXiYmJjn1DQkIKPX7BggUaNmyYunTpojlz5mjVqlVasGCBJDn1RRRHamqqjDGFnsfPz0+enp46fvy403Z/f3+n2x4eHo7ntNls+u6779SiRQuNGzdOERER6tSpk3788ccS1QQAKBmmQgFANRUQEKDg4OAil4KVpDp16jj+bbPZCt0/d+5ctW/fXtOnT3ds++GHHy6qFn9/f9lsNiUnJzttT0tL05kzZxQQEFCi4zVt2lRz585VTk6OVqxYoSeeeEL9+/fXoUOHHNf3AACULc5YAEAV5+HhIanwWYTY2FgdPXpUHh4e6tSpU6Gvgsedy6lTpwrt8+GHHxb5/Bc6g+Hj46P27dsXatKeM2eOJDmmWJWUu7u7unfvrscee0zp6ek6fPjwRR0HAHBhnLEAgCouNDRU/v7++vjjj9WgQQN5enqqbdu26tWrl/r376/rrrtOjzzyiNq2bavMzExt2bJFu3bt0ltvvXXe4/bq1Uvjxo3TM888o5iYGC1cuFBLliwptF+LFi20dOlSLV68WLVr11aDBg0UGBhYaL9JkyZp4MCBuuOOO3THHXdo+/bteuKJJ3TzzTc7NW5fyG+//aYHH3xQgwcPVqNGjZSWlqapU6cqKirqnCtSAQBKjzMWAFDFubi4aObMmdq7d6969uypzp07O/5yP2/ePMXFxen1119Xnz59dNddd+m7775T9+7dL3jcsWPH6sEHH9Srr76qm266SQcOHCh0DQxJmjJliurVq6ebb75ZnTt31pdfflnk8QYMGKC5c+dq8+bNuuGGG/Tcc89pzJgx+uCDD0r0ekNDQxUaGqqpU6eqT58+Gjt2rCIiIvTdd9/J1dW1RMcCABSfzRhjrC4CAAAAQOXGGQsAAAAApUawAAAAAFBqBAsAAAAApUawAAAAAFBqBAsAAAAApUawAAAAAFBqBAsAAAAApUawAAAAAFBqBAsAAAAApUawAAAAAFBqBAsAAAAApfb/7E3RDCXj/Z0AAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 800x550 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from envs.custom_mazes.env_utils import policy_image_fourrooms, value_image_fourrooms\n",
    "from functools import partial\n",
    "from utils.evaluation import supply_rng, add_to, flatten\n",
    "from IPython.display import clear_output\n",
    "from envs.env_utils import EpisodeMonitor\n",
    "from collections import defaultdict\n",
    "import numpy as np\n",
    "\n",
    "def evaluate_fourrooms(\n",
    "    agent,\n",
    "    env,\n",
    "    task_id=None,\n",
    "    config=None,\n",
    "    num_eval_episodes=10,\n",
    "    num_video_episodes=0,\n",
    "    video_frame_skip=3,\n",
    "    eval_temperature=0.0,\n",
    "    eval_gaussian=None,\n",
    "):\n",
    "\n",
    "    actor_fn = supply_rng(agent.sample_actions, rng=jax.random.PRNGKey(np.random.randint(0, 2**32)))\n",
    "    trajs = []\n",
    "    stats = defaultdict(list)\n",
    "    pbar = range(num_eval_episodes + num_video_episodes)\n",
    "    renders = []\n",
    "    for i in pbar:\n",
    "        traj = defaultdict(list)\n",
    "        should_render = i >= num_eval_episodes\n",
    "        observation, info = env.unwrapped.setup_goals(seed=None, task_num=task_id, start_pos=START_POS)\n",
    "        goal = info.get(\"goal_pos\", None)\n",
    "        done = False\n",
    "        step = 0\n",
    "        render = []\n",
    "        latent_z = agent.infer_z(goal)\n",
    "\n",
    "        while not done:\n",
    "            action = actor_fn(observations=observation, latent_z=latent_z, temperature=eval_temperature)\n",
    "            next_observation, reward, terminated, truncated, info = env.step(jax.device_get(action.squeeze()))\n",
    "            done = terminated or truncated\n",
    "            step += 1\n",
    "\n",
    "            if should_render and (step % video_frame_skip == 0 or done):\n",
    "                # frame = env.unwrapped.render(return_img=True).copy()\n",
    "                render.append(env.unwrapped.render(return_img=True).copy())\n",
    "\n",
    "            transition = dict(\n",
    "                observation=observation,\n",
    "                next_observation=next_observation,\n",
    "                action=action,\n",
    "                reward=reward,\n",
    "                done=done,\n",
    "                info=info,\n",
    "            )\n",
    "            add_to(traj, transition)\n",
    "            observation = next_observation\n",
    "        if i < num_eval_episodes:\n",
    "            add_to(stats, flatten(info))\n",
    "            trajs.append(traj)\n",
    "        else:\n",
    "            renders.append(np.array(render))\n",
    "\n",
    "    for k, v in stats.items():\n",
    "        stats[k] = np.mean(v)\n",
    "\n",
    "    return stats, trajs, renders\n",
    "\n",
    "def visualize_value_image(task_num):\n",
    "    env = FourRoomsMazeEnv(Maze(seed=0))\n",
    "    env.reset()\n",
    "    observation, info = env.setup_goals(seed=None, task_num=task_num, start_pos=START_POS)\n",
    "    goal = info.get(\"goal_pos\", None)\n",
    "    latent_z = jax.device_get(fb_agent.infer_z(goal)[None])\n",
    "    N, M = env.maze.size\n",
    "    pred_value_img = value_image_fourrooms(env, example_batch, N=N, M=M,\n",
    "                                value_fn=partial(fb_agent.predict_q, z=latent_z), goal=goal)\n",
    "    return pred_value_img\n",
    "\n",
    "def visualize_policy(task_num):\n",
    "    env = FourRoomsMazeEnv(Maze(seed=0))\n",
    "    env.reset()\n",
    "    observation, info = env.setup_goals(seed=None, task_num=task_num, start_pos=START_POS)\n",
    "    goal = info.get(\"goal_pos\", None)\n",
    "    latent_z = fb_agent.infer_z(goal)\n",
    "    start = info.get(\"start_pos\", None)\n",
    "    example_batch = whole_dataset.sample(1)\n",
    "    pred_policy_img = policy_image_fourrooms(env, example_batch, N=env.maze.size[0], M=env.maze.size[1],\n",
    "                                                    action_fn=partial(supply_rng(fb_agent.sample_actions, rng=jax.random.PRNGKey(np.random.randint(0, 2**32))), latent_z=latent_z, temperature=0.0),\n",
    "                                                    goal=goal, start=start)\n",
    "    return pred_policy_img\n",
    "\n",
    "training_steps = 100_001\n",
    "pbar = tqdm(range(training_steps))\n",
    "env = FourRoomsMazeEnv(Maze(seed=0), max_steps=NUM_TRAIN_STEPS)\n",
    "# q = EpisodeMonitor(env, filter_regexes=['.*privileged.*', '.*proprio.*'])\n",
    "eval_history_train = []\n",
    "eval_history_test = []\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 % 10_000 == 0:\n",
    "        clear_output()\n",
    "        fig, ax = plt.subplots(nrows=1, ncols=2, figsize=(20, 10))\n",
    "        pred_policy_img = visualize_policy(task_num=2)\n",
    "        ax[0].imshow(pred_policy_img)\n",
    "        pred_value_img = visualize_value_image(task_num=2)\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)\n",
    "    \n",
    "    if update_step % 25_000 == 0:\n",
    "        clear_output()\n",
    "        fig, ax = plt.subplots()\n",
    "        # ax.plot(eval_history_test, label='Test')\n",
    "        ax.plot(eval_history_train, label='Train')\n",
    "        ax.set_xlabel(\"Iterations\")\n",
    "        ax.set_ylabel(\"Accuracy\")\n",
    "        ax.set_xticks(np.arange(len(np.arange(training_steps, step=25_000))), labels=np.arange(training_steps, step=25_000))\n",
    "        plt.legend()\n",
    "        plt.title(\"Performance of FB on Train layouts across tasks\")\n",
    "        plt.tight_layout()\n",
    "        display(fig)\n",
    "        plt.close(fig)\n",
    "        \n",
    "        eval_metrics = {}\n",
    "        overall_metrics = defaultdict(list)\n",
    "        \n",
    "        for task_id in range(3):\n",
    "            for env_id in range(NUM_TRAIN_LAYOUTS):\n",
    "                env = FourRoomsMazeEnv(Maze(seed=env_id), NUM_TRAIN_STEPS)\n",
    "                env = EpisodeMonitor(env, filter_regexes=['.*privileged.*', '.*proprio.*'])\n",
    "                # env.reset(options={\"start\": (1, 1)})\n",
    "                eval_info, _, _ = evaluate_fourrooms(\n",
    "                        agent=fb_agent,\n",
    "                        env=env,\n",
    "                        task_id=task_id,\n",
    "                        config=None,\n",
    "                        num_eval_episodes=10,\n",
    "                        num_video_episodes=0,\n",
    "                        video_frame_skip=1,\n",
    "                        eval_temperature=0.0,\n",
    "                        eval_gaussian=None\n",
    "                    )\n",
    "                eval_metrics.update(\n",
    "                    {f'evaluation/task_{task_id}_{k}': v for k, v in eval_info.items() if k != 'total.timesteps'}\n",
    "                )\n",
    "                for k, v in eval_info.items():\n",
    "                    overall_metrics[k].append(v)\n",
    "                    \n",
    "        for k, v in overall_metrics.items():\n",
    "            eval_metrics[f'evaluation/overall_{k}_train'] = np.mean(v)\n",
    "            \n",
    "        eval_history_train.append(eval_metrics['evaluation/overall_episode.final_reward_train'])\n",
    "\n",
    "        # eval_metrics = {}\n",
    "        # overall_metrics = defaultdict(list)\n",
    "\n",
    "        # for task_id in range(3):\n",
    "        #     for env_id in range(NUM_TRAIN_LAYOUTS+50, NUM_TRAIN_LAYOUTS + 60):\n",
    "        #         env = FourRoomsMazeEnv(Maze(seed=env_id), max_steps=NUM_TRAIN_STEPS)\n",
    "        #         env = EpisodeMonitor(env, filter_regexes=['.*privileged.*', '.*proprio.*'])\n",
    "        #         # env.reset(options={\"start\": (1, 1)})\n",
    "        #         eval_info, _, _ = evaluate_fourrooms(\n",
    "        #                 agent=fb_agent,\n",
    "        #                 env=env,\n",
    "        #                 task_id=task_id,\n",
    "        #                 config=None,\n",
    "        #                 num_eval_episodes=10,\n",
    "        #                 num_video_episodes=0,\n",
    "        #                 video_frame_skip=1,\n",
    "        #                 eval_temperature=0.0,\n",
    "        #                 eval_gaussian=None\n",
    "        #             )\n",
    "        #         eval_metrics.update(\n",
    "        #             {f'evaluation/task_{task_id}_{k}': v for k, v in eval_info.items() if k != 'total.timesteps'}\n",
    "        #             )\n",
    "        #         for k, v in eval_info.items():\n",
    "        #             overall_metrics[k].append(v)\n",
    "                    \n",
    "        # for k, v in overall_metrics.items():\n",
    "        #     eval_metrics[f'evaluation/overall_{k}_ood'] = np.mean(v)\n",
    "        # eval_history_test.append(eval_metrics['evaluation/overall_episode.final_reward_ood'])\n",
    "        \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "ename": "TypeError",
     "evalue": "'<' not supported between instances of 'XTick' and 'XTick'",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mTypeError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[7], line 6\u001b[0m\n\u001b[1;32m      4\u001b[0m ax\u001b[38;5;241m.\u001b[39mset_xlabel(\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mIterations\u001b[39m\u001b[38;5;124m\"\u001b[39m)\n\u001b[1;32m      5\u001b[0m ax\u001b[38;5;241m.\u001b[39mset_ylabel(\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mAccuracy\u001b[39m\u001b[38;5;124m\"\u001b[39m)\n\u001b[0;32m----> 6\u001b[0m \u001b[43max\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mset_xticks\u001b[49m\u001b[43m(\u001b[49m\u001b[43max\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mset_xticks\u001b[49m\u001b[43m(\u001b[49m\u001b[43mnp\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43marange\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mlen\u001b[39;49m\u001b[43m(\u001b[49m\u001b[43mnp\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43marange\u001b[49m\u001b[43m(\u001b[49m\u001b[43mtraining_steps\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mstep\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m25_000\u001b[39;49m\u001b[43m)\u001b[49m\u001b[43m)\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m      7\u001b[0m \u001b[43m                            \u001b[49m\u001b[43mlabels\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mnp\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43marange\u001b[49m\u001b[43m(\u001b[49m\u001b[43mtraining_steps\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mstep\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m25_000\u001b[39;49m\u001b[43m)\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mlabels\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mnp\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43marange\u001b[49m\u001b[43m(\u001b[49m\u001b[43mtraining_steps\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mstep\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m25_000\u001b[39;49m\u001b[43m)\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m      8\u001b[0m plt\u001b[38;5;241m.\u001b[39mlegend()\n\u001b[1;32m      9\u001b[0m plt\u001b[38;5;241m.\u001b[39mtitle(\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mPerformance of FB on Train layouts across tasks\u001b[39m\u001b[38;5;124m\"\u001b[39m)\n",
      "File \u001b[0;32m~/anaconda3/envs/jax2/lib/python3.9/site-packages/matplotlib/axes/_base.py:74\u001b[0m, in \u001b[0;36m_axis_method_wrapper.__set_name__.<locals>.wrapper\u001b[0;34m(self, *args, **kwargs)\u001b[0m\n\u001b[1;32m     73\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m\u001b[38;5;250m \u001b[39m\u001b[38;5;21mwrapper\u001b[39m(\u001b[38;5;28mself\u001b[39m, \u001b[38;5;241m*\u001b[39margs, \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mkwargs):\n\u001b[0;32m---> 74\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mget_method\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[43m)\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43margs\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",
      "File \u001b[0;32m~/anaconda3/envs/jax2/lib/python3.9/site-packages/matplotlib/axis.py:2183\u001b[0m, in \u001b[0;36mAxis.set_ticks\u001b[0;34m(self, ticks, labels, minor, **kwargs)\u001b[0m\n\u001b[1;32m   2178\u001b[0m     first_key \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mnext\u001b[39m(\u001b[38;5;28miter\u001b[39m(kwargs))\n\u001b[1;32m   2179\u001b[0m     \u001b[38;5;28;01mraise\u001b[39;00m \u001b[38;5;167;01mValueError\u001b[39;00m(\n\u001b[1;32m   2180\u001b[0m         \u001b[38;5;124mf\u001b[39m\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mIncorrect use of keyword argument \u001b[39m\u001b[38;5;132;01m{\u001b[39;00mfirst_key\u001b[38;5;132;01m!r}\u001b[39;00m\u001b[38;5;124m. Keyword arguments \u001b[39m\u001b[38;5;124m\"\u001b[39m\n\u001b[1;32m   2181\u001b[0m         \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mother than \u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mminor\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124m modify the text labels and can only be used if \u001b[39m\u001b[38;5;124m\"\u001b[39m\n\u001b[1;32m   2182\u001b[0m         \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mlabels\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124m are passed as well.\u001b[39m\u001b[38;5;124m\"\u001b[39m)\n\u001b[0;32m-> 2183\u001b[0m result \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_set_tick_locations\u001b[49m\u001b[43m(\u001b[49m\u001b[43mticks\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mminor\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mminor\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   2184\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m labels \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m:\n\u001b[1;32m   2185\u001b[0m     \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mset_ticklabels(labels, minor\u001b[38;5;241m=\u001b[39mminor, \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mkwargs)\n",
      "File \u001b[0;32m~/anaconda3/envs/jax2/lib/python3.9/site-packages/matplotlib/axis.py:2128\u001b[0m, in \u001b[0;36mAxis._set_tick_locations\u001b[0;34m(self, ticks, minor)\u001b[0m\n\u001b[1;32m   2125\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mlen\u001b[39m(ticks):\n\u001b[1;32m   2126\u001b[0m     \u001b[38;5;28;01mfor\u001b[39;00m axis \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_get_shared_axis():\n\u001b[1;32m   2127\u001b[0m         \u001b[38;5;66;03m# set_view_interval maintains any preexisting inversion.\u001b[39;00m\n\u001b[0;32m-> 2128\u001b[0m         axis\u001b[38;5;241m.\u001b[39mset_view_interval(\u001b[38;5;28;43mmin\u001b[39;49m\u001b[43m(\u001b[49m\u001b[43mticks\u001b[49m\u001b[43m)\u001b[49m, \u001b[38;5;28mmax\u001b[39m(ticks))\n\u001b[1;32m   2129\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39maxes\u001b[38;5;241m.\u001b[39mstale \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mTrue\u001b[39;00m\n\u001b[1;32m   2130\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m minor:\n",
      "\u001b[0;31mTypeError\u001b[0m: '<' not supported between instances of 'XTick' and 'XTick'"
     ]
    },
    {
     "data": {
      "image/png": "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zWZr8jFr1qwm5xw+fFjXXXedIiIiFBcXp3vvvVf19fVNztmyZYtGjhyp0NBQ9e3bV08//fRZebjmPN/8+fPPuh4GDhzoevzUqVOaPXu2YmJi1LlzZ914440qLi5u8hpcL/4lJSXlrGvGZrNp9uzZkvgZA+ntt9/W+PHjlZiYKJvNphdeeKHJ48YYzZs3T3a7XeHh4crKytJnn33W5Jzjx49r6tSpioyMVHR0tG6//XZVVlY2OWfXrl264oorFBYWpuTkZD3yyCNnZVm7dq0GDhyosLAwDR06VOvXr3c7i1sMOtyqVatMSEiIeeqpp8zHH39s7rzzThMdHW2Ki4utjoY2NGbMGPP3v//d7Nmzx+Tl5Zlx48aZnj17msrKStc5o0aNMnfeeacpLCx0fZSXl7ser6+vN0OGDDFZWVlmx44dZv369SY2NtbMnTvXdc4XX3xhIiIizJw5c8zevXvNY489ZgIDA82GDRtc53DNeYecnBxz8cUXN7keSkpKXI/PmjXLJCcnm02bNplt27aZ733ve+b73/++63GuF/9z9OjRJtfLxo0bjSTz5ptvGmP4GQNj1q9fb371q1+Z559/3kgy69ata/L4okWLTFRUlHnhhRfMzp07zYQJE0zv3r1NTU2N65xrr73WDB8+3Lz//vvmnXfeMX379jWTJ092PV5eXm7i4+PN1KlTzZ49e8yzzz5rwsPDzdKlS13n/Pvf/zaBgYHmkUceMXv37jUPPfSQCQ4ONrt373YrizsouRa49NJLzezZs12fNzQ0mMTERPPwww9bmArt7ejRo0aSeeutt1zHRo0aZe6+++7vfM769etNQECAKSoqch3761//aiIjI83p06eNMcbcd9995uKLL27yvEmTJpkxY8a4Puea8w45OTlm+PDhzT5WVlZmgoODzdq1a13HPvnkEyPJbN261RjD9QJj7r77btOnTx/jcDiMMfyMQVPfLrkOh8MkJCSY3/3ud65jZWVlJjQ01Dz77LPGGGP27t1rJJmPPvrIdc6rr75qbDabyc/PN8YY85e//MV07drVdc0YY8z9999vBgwY4Pr85ptvNtddd12TPBkZGWbmzJktzuIuhit0sNraWuXm5iorK8t1LCAgQFlZWdq6dauFydDeysvLJUndunVrcvyZZ55RbGyshgwZorlz56q6utr12NatWzV06FDFx8e7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",
      "text/plain": [
       "<Figure size 800x550 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Final\n",
    "fig, ax = plt.subplots()\n",
    "ax.plot(eval_history_train, label='Train')\n",
    "ax.set_xlabel(\"Iterations\")\n",
    "ax.set_ylabel(\"Accuracy\")\n",
    "ax.set_xticks(ax.set_xticks(np.arange(len(np.arange(training_steps, step=25_000))),\n",
    "                            labels=np.arange(training_steps, step=25_000)), labels=np.arange(training_steps, step=25_000))\n",
    "plt.legend()\n",
    "plt.title(\"Performance of FB on Train layouts across tasks\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n",
      "\u001b[A"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A"
     ]
    }
   ],
   "source": [
    "# # RANDOM\n",
    "# from tqdm import trange\n",
    "\n",
    "# def evaluate_fourrooms(\n",
    "#     env,\n",
    "#     task_id=None,\n",
    "#     config=None,\n",
    "#     num_eval_episodes=10,\n",
    "#     num_video_episodes=0,\n",
    "#     video_frame_skip=3,\n",
    "#     eval_temperature=0.0,\n",
    "#     eval_gaussian=None,\n",
    "# ):\n",
    "#     trajs = []\n",
    "#     stats = defaultdict(list)\n",
    "#     pbar = trange(num_eval_episodes + num_video_episodes, leave=False, colour='red', position=2)\n",
    "#     renders = []\n",
    "#     for i in pbar:\n",
    "#         traj = defaultdict(list)\n",
    "#         should_render = i >= num_eval_episodes\n",
    "#         observation, info = env.unwrapped.setup_goals(seed=None, task_num=task_id, start_pos=START_POS)\n",
    "#         goal = info.get(\"goal_pos\", None)\n",
    "#         done = False\n",
    "#         step = 0\n",
    "#         render = []\n",
    "#         while not done:\n",
    "#             action = env.action_space.sample()\n",
    "#             next_observation, reward, terminated, truncated, info = env.step(jax.device_get(action.squeeze()))\n",
    "#             done = terminated or truncated\n",
    "#             step += 1\n",
    "\n",
    "#             if should_render and (step % video_frame_skip == 0 or done):\n",
    "#                 frame = env.unwrapped.render(return_img=True).copy()\n",
    "#                 render.append(frame)\n",
    "\n",
    "#             transition = dict(\n",
    "#                 observation=observation,\n",
    "#                 next_observation=next_observation,\n",
    "#                 action=action,\n",
    "#                 reward=reward,\n",
    "#                 done=done,\n",
    "#                 info=info,\n",
    "#             )\n",
    "#             add_to(traj, transition)\n",
    "#             observation = next_observation\n",
    "#         if i < num_eval_episodes:\n",
    "#             add_to(stats, flatten(info))\n",
    "#             trajs.append(traj)\n",
    "#         else:\n",
    "#             renders.append(np.array(render))\n",
    "\n",
    "#     for k, v in stats.items():\n",
    "#         stats[k] = np.mean(v)\n",
    "\n",
    "#     return stats, trajs, renders\n",
    "\n",
    "# eval_metrics = {}\n",
    "# overall_metrics = defaultdict(list)\n",
    "# eval_history_train = []\n",
    "# eval_history_test = []\n",
    "# for task_id in range(2, 5): # static for 4 rooms\n",
    "#     for env_id in range(NUM_TRAIN_LAYOUTS):\n",
    "#         env = FourRoomsMazeEnv(Maze(seed=env_id, maze_type='fourrooms_random_layouts'), NUM_TRAIN_STEPS)\n",
    "#         env = EpisodeMonitor(env, filter_regexes=['.*privileged.*', '.*proprio.*'])\n",
    "#         # env.reset(options={\"start\": (1, 1)})\n",
    "#         eval_info, _, _ = evaluate_fourrooms(\n",
    "#                 env=env,\n",
    "#                 task_id=task_id,\n",
    "#                 config=None,\n",
    "#                 num_eval_episodes=10,\n",
    "#                 num_video_episodes=0,\n",
    "#                 video_frame_skip=1,\n",
    "#                 eval_temperature=0.0,\n",
    "#                 eval_gaussian=None\n",
    "#             )\n",
    "#         eval_metrics.update(\n",
    "#             {f'evaluation/task_{task_id}_{k}': v for k, v in eval_info.items() if k != 'total.timesteps'}\n",
    "#         )\n",
    "#         for k, v in eval_info.items():\n",
    "#             overall_metrics[k].append(v)\n",
    "            \n",
    "# for k, v in overall_metrics.items():\n",
    "#     eval_metrics[f'evaluation/overall_{k}_train'] = np.mean(v)\n",
    "    \n",
    "# eval_history_train.append(eval_metrics['evaluation/overall_episode.final_reward_train'])\n",
    "\n",
    "# eval_metrics = {}\n",
    "# overall_metrics = defaultdict(list)\n",
    "            \n",
    "# for task_id in range(2, 5):\n",
    "#     for env_id in range(100, 110):\n",
    "#         env = FourRoomsMazeEnv(Maze(seed=env_id, maze_type='fourrooms_random_layouts'), max_steps=NUM_TRAIN_STEPS)\n",
    "#         env = EpisodeMonitor(env, filter_regexes=['.*privileged.*', '.*proprio.*'])\n",
    "#         # env.reset(options={\"start\": (1, 1)})\n",
    "#         eval_info, _, _ = evaluate_fourrooms(\n",
    "#                 env=env,\n",
    "#                 task_id=task_id,\n",
    "#                 config=None,\n",
    "#                 num_eval_episodes=10, ##\n",
    "#                 num_video_episodes=0,\n",
    "#                 video_frame_skip=1,\n",
    "#                 eval_temperature=0.0,\n",
    "#                 eval_gaussian=None\n",
    "#             )\n",
    "#         eval_metrics.update(\n",
    "#             {f'evaluation/task_{task_id}_{k}': v for k, v in eval_info.items() if k != 'total.timesteps'}\n",
    "#             )\n",
    "#         for k, v in eval_info.items():\n",
    "#             overall_metrics[k].append(v)\n",
    "            \n",
    "# for k, v in overall_metrics.items():\n",
    "#     eval_metrics[f'evaluation/overall_{k}_ood'] = np.mean(v)\n",
    "# eval_history_test.append(eval_metrics['evaluation/overall_episode.final_reward_ood'])"
   ]
  },
  {
   "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
}
