{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "from env.decom_lunar_lander import LunarLander as LunarLander_decom_reward\n",
    "import torch\n",
    "from torch import nn\n",
    "from torch.autograd import variable\n",
    "import time\n",
    "import os\n",
    "import shutil\n",
    "import numpy as np\n",
    "\n",
    "from tqdm import tqdm\n",
    "from random import uniform, randint, sample, random\n",
    "from collections import deque\n",
    "\n",
    "import io\n",
    "import base64\n",
    "from IPython.display import HTML\n",
    "\n",
    "from models.feature_q_model import feature_q_model\n",
    "from memory.memory import ReplayBuffer, ReplayBuffer_decom\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.image as mpimg\n",
    "import matplotlib as mpl\n",
    "from tensorboardX import SummaryWriter\n",
    "%matplotlib inline\n",
    "\n",
    "FloatTensor = torch.cuda.FloatTensor\n",
    "LongTensor = torch.cuda.LongTensor\n",
    "Writer = SummaryWriter(log_dir=\"CartPole_summary\")\n",
    "mpl.style.use('bmh')\n",
    "plt.rcParams[\"font.family\"] = \"Arial\"#\"Helvetica\"\n",
    "%config InlineBackend.figure_format = 'retina'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "ENV_NAME = 'LunarLander_ESP'\n",
    "env = LunarLander_decom_reward()\n",
    "\n",
    "ACTION_DICT = {\n",
    "    \"NOOP\": 0,\n",
    "    \"LEFT\":1,\n",
    "    \"MAIN\":2,\n",
    "    \"RIGHT\":3\n",
    "}\n",
    "action_name = {\n",
    "    0: 'Nop',\n",
    "    1: 'right engine',\n",
    "    2: 'main engine',\n",
    "    3: 'left engine'\n",
    "}\n",
    "FEATRUESNAME = [\"F1 Distance\", \"F2 Velocity\", \"F3 Tilt-angle\", \"F4 Right landing leg\", \n",
    "                \"F5 Left Landing Leg\", \"F6 Main Engine Uses\", \"F7 Side Engine Uses\", \"F8 Landing\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "result_floder = ENV_NAME\n",
    "result_floder_exp = \"{}/{}\".format(result_floder, ENV_NAME + \"_exp\")\n",
    "result_file = ENV_NAME + \"/results.txt\"\n",
    "result_file_GVFs_loss = ENV_NAME + \"/results_GVFs_loss.txt\"\n",
    "if not os.path.isdir(result_floder):\n",
    "    os.mkdir(result_floder)\n",
    "if not os.path.isdir(result_floder_exp):\n",
    "    os.mkdir(result_floder_exp)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "def MSX(vector):\n",
    "    vector = np.array(vector)\n",
    "    indeces = np.argsort(vector)[::-1]\n",
    "    negative_sum = sum(vector[vector < 0])\n",
    "    pos_sum = 0\n",
    "    MSX_idx = []\n",
    "    for idx in indeces:\n",
    "        pos_sum += vector[idx]\n",
    "        MSX_idx.append(idx)\n",
    "        if pos_sum > abs(negative_sum):\n",
    "            break\n",
    "    return MSX_idx, vector[MSX_idx]\n",
    "\n",
    "def print_Cartpole(state):\n",
    "    print(\"Cart Position: \", state[0], end = \"  \")\n",
    "    print(\"Cart Velocity: \", state[1])\n",
    "    print(\"Pole Angle: \", state[2], end = \"  \")\n",
    "    print(\"Pole Velocity At Tip: \", state[3])\n",
    "def plot_action_group(values, group, elements = [], title = 'decomposition values', y_label = \"\", q_values = None, IGX_action = None, exp_count = -1):\n",
    "    plt.clf()\n",
    "    x = np.arange(len(group))  # the label locations\n",
    "    fig, ax = plt.subplots()\n",
    "\n",
    "    # Add some text for labels, title and custom x-axis tick labels, etc.\n",
    "    plt.ylabel(y_label)\n",
    "    plt.title(title)\n",
    "\n",
    "    # set width of bar\n",
    "    length = len(values[0])\n",
    "    barWidth = 1 / (len(values) + 1)\n",
    "    x_labels_feature = [\"F{}\".format(x + 1) for x in range(len(values))]\n",
    "    for i in range(len(values)):\n",
    "        r = [j + barWidth * i for j in range(length)]\n",
    "        if len(elements) > 0:\n",
    "            plt.bar(r, values[i] , width=barWidth,  label=elements[i])\n",
    "#             for j, rr in enumerate(r):\n",
    "#                 plt.text(rr - barWidth / 4, values[i][j] if values[i][j] > 0 else 0, x_labels_feature[i], fontsize=5)\n",
    "        else:\n",
    "            plt.bar(r, values[i], width=barWidth)\n",
    "    \n",
    "    print(barWidth)\n",
    "    y_lim = plt.gca().get_ylim()\n",
    "    gap = (y_lim[1] - 0) / 30\n",
    "    for i in range(len(values)):\n",
    "        r = [j + barWidth * i for j in range(length)]\n",
    "        for j, rr in enumerate(r):\n",
    "            if values[i][j] != 0:\n",
    "                plt.text(rr - barWidth / 7, -gap * 1.5 if values[i][j] > 0 else gap, x_labels_feature[i], fontsize=8)\n",
    "    \n",
    "    r = [j + barWidth * (i + 1) for j in range(length)]\n",
    "    for rr in r[:-1]:\n",
    "        plt.axvline(x = rr, alpha = 0.5, linestyle='--')\n",
    "\n",
    "    # Add xticks on the middle of the group bars\n",
    "    plt.xlabel('action', fontweight='bold')\n",
    "    center_pos = (1 - barWidth * 2) / 2\n",
    "    \n",
    "    if IGX_action is not None:\n",
    "        group = [\"\\\"{}\\\"\\n greater than\".format(IGX_action)] + group\n",
    "        x_lim_left = plt.gca().get_xlim()[0]\n",
    "        pos = [x_lim_left] + [r + center_pos for r in range(length)]\n",
    "        plt.xticks(pos, group, ha='center')\n",
    "    else:\n",
    "        plt.xticks([r + center_pos for r in range(length)], group, ha='center')\n",
    "    \n",
    "    plt.gca().xaxis.grid(False)\n",
    "    plt.gca().yaxis.grid(True)\n",
    "    plt.gca().set_axisbelow(True)\n",
    "    if np.min(values) >= -gap * 3:\n",
    "        plt.gca().set_ylim(bottom = -gap * 3)\n",
    "    plt.ticklabel_format(style='sci', axis='y', scilimits=(0,0))\n",
    "    \n",
    "    y_lim = plt.gca().get_ylim()\n",
    "    np_values = np.array(values).T\n",
    "    txt_pos = y_lim[1] - (y_lim[1] - y_lim[0]) / 25\n",
    "    max_values = np_values.max(axis = 1)\n",
    "    sum_values = np_values.sum(axis = 1)\n",
    "    \n",
    "    for i, v in enumerate(max_values):\n",
    "        if q_values is None:\n",
    "            plt.gca().text(i + center_pos, txt_pos, \"sum ≈ {}\".format(np.round(sum_values[i], 4)), color='black', fontweight='ultralight', ha='center')\n",
    "        else:\n",
    "            plt.gca().text(i + center_pos, txt_pos, \"Q_v ≈ {}\".format(np.round(q_values[i], 4)), color='black', fontweight='ultralight', ha='center')\n",
    "\n",
    "    plt.savefig(\"{}/{}-{}.png\".format(result_floder_exp, title, exp_count))\n",
    "    \n",
    "def display_frames_as_gif(frames, video_name):\n",
    "    \"\"\"\n",
    "    Displays a list of frames as a gif, with controls\n",
    "    \"\"\"\n",
    "    Writer = animation.writers['ffmpeg']\n",
    "    writer = Writer(fps=15, metadata=dict(artist='Me'), bitrate=1800)\n",
    "    #plt.figure(figsize=(frames[0].shape[1] / 72.0, frames[0].shape[0] / 72.0), dpi = 72)\n",
    "    patch = plt.imshow(frames[0])\n",
    "    plt.axis('off')\n",
    "\n",
    "    def animate(i):\n",
    "        patch.set_data(frames[i])\n",
    "\n",
    "    anim = animation.FuncAnimation(plt.gcf(), animate, frames = len(frames), interval=50)\n",
    "#     display(display_animation(anim, default_mode='loop'))\n",
    "    anim.save(result_floder + '/' + video_name, writer=writer)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [],
   "source": [
    "def plot_IG_MSX(values, values_msx, group, elements = [], title = 'decomposition values', y_label = \"\", q_values = None, IGX_action = None, exp_count = -1):\n",
    "    plt.clf()\n",
    "    x = np.arange(len(group))  # the label locations\n",
    "    fig, ax = plt.subplots()\n",
    "\n",
    "    # Add some text for labels, title and custom x-axis tick labels, etc.\n",
    "    plt.ylabel(y_label)\n",
    "    plt.title(title)\n",
    "\n",
    "    # set width of bar\n",
    "    length = len(values[0])\n",
    "    barWidth = 1 / (len(values) + 1)\n",
    "    x_labels_feature = [\"F{}\".format(x + 1) for x in range(len(values))]\n",
    "    for i in range(len(values)):\n",
    "        r = [j + barWidth * i for j in range(length)]\n",
    "        \n",
    "        for j in range(len(values[i])):\n",
    "            if values_msx[i][j] != 0:\n",
    "                plt.bar(r[j], values[i][j], width=barWidth, hatch=\"////\",\n",
    "                        color = plt.rcParams['axes.prop_cycle'].by_key()['color'][i])\n",
    "            else:\n",
    "                plt.bar(r[j], values[i][j], width=barWidth,\n",
    "                        color = plt.rcParams['axes.prop_cycle'].by_key()['color'][i])\n",
    "    \n",
    "    y_lim = plt.gca().get_ylim()\n",
    "    gap = (y_lim[1] - 0) / 30\n",
    "    for i in range(len(values)):\n",
    "        r = [j + barWidth * i for j in range(length)]\n",
    "        for j, rr in enumerate(r):\n",
    "            if values[i][j] != 0:\n",
    "                plt.text(rr - barWidth / 4, -gap * 1.5 if values[i][j] > 0 else gap, x_labels_feature[i], fontsize=5)\n",
    "    \n",
    "    r = [j + barWidth * (i + 1) for j in range(length)]\n",
    "    for rr in r[:-1]:\n",
    "        plt.axvline(x = rr, alpha = 0.5, linestyle='--')\n",
    "\n",
    "    # Add xticks on the middle of the group bars\n",
    "    plt.xlabel('action', fontweight='bold')\n",
    "    center_pos = (1 - barWidth * 2) / 2\n",
    "    \n",
    "    plt.xticks([r + center_pos for r in range(length)], group, ha='center')\n",
    "    \n",
    "    plt.gca().xaxis.grid(False)\n",
    "    plt.gca().yaxis.grid(True)\n",
    "    plt.gca().set_axisbelow(True)\n",
    "    if np.min(values) >= -gap * 3:\n",
    "        plt.gca().set_ylim(bottom = -gap * 3)\n",
    "    plt.ticklabel_format(style='sci', axis='y', scilimits=(0,0))\n",
    "    \n",
    "    y_lim = plt.gca().get_ylim()\n",
    "    np_values = np.array(values).T\n",
    "    txt_pos = y_lim[1] - (y_lim[1] - y_lim[0]) / 25\n",
    "    max_values = np_values.max(axis = 1)\n",
    "    sum_values = np_values.sum(axis = 1)\n",
    "    \n",
    "    for i, v in enumerate(max_values):\n",
    "        if q_values is None:\n",
    "            plt.gca().text(i + center_pos, txt_pos, \"{} > {}\".format(IGX_action, group[i]), color='black', fontweight='ultralight', ha='center')\n",
    "        else:\n",
    "            plt.gca().text(i + center_pos, txt_pos, \"Q_v ≈ {}\".format(np.round(q_values[i], 4)), color='black', fontweight='ultralight', ha='center')\n",
    "\n",
    "    plt.savefig(\"{}/{}-{}.png\".format(result_floder_exp, title, exp_count))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "hyperparams_Lunarlander = {\n",
    "    'epsilon_decay_steps' : 200000, \n",
    "    'final_epsilon' : 0.01,\n",
    "    'batch_size' : 128, \n",
    "    'update_steps' : 3, \n",
    "    'memory_size' : 100000, \n",
    "    'beta' : 0.99, \n",
    "    'model_replace_freq' : 1,\n",
    "    'learning_rate' : 0.0001,\n",
    "    'privous_state' : 1\n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "class ESP_agent(object):\n",
    "    def __init__(self, env, hyper_params, action_space = len(ACTION_DICT)):\n",
    "        \n",
    "        self.env = env\n",
    "        self.max_episode_steps = env._max_episode_steps\n",
    "        \n",
    "        \"\"\"\n",
    "            beta: The discounted factor of Q-value function\n",
    "            (epsilon): The explore or exploit policy epsilon. \n",
    "            initial_epsilon: When the 'steps' is 0, the epsilon is initial_epsilon, 1\n",
    "            final_epsilon: After the number of 'steps' reach 'epsilon_decay_steps', \n",
    "                The epsilon set to the 'final_epsilon' determinately.\n",
    "            epsilon_decay_steps: The epsilon will decrease linearly along with the steps from 0 to 'epsilon_decay_steps'.\n",
    "        \"\"\"\n",
    "        self.beta = hyper_params['beta']\n",
    "        self.initial_epsilon = 1\n",
    "        self.final_epsilon = hyper_params['final_epsilon']\n",
    "        self.epsilon_decay_steps = hyper_params['epsilon_decay_steps']\n",
    "\n",
    "        \"\"\"\n",
    "            episode: Record training episode\n",
    "            steps: Add 1 when predicting an action\n",
    "            learning: The trigger of agent learning. It is on while training agent. It is off while testing agent.\n",
    "            action_space: The action space of the current environment, e.g 2.\n",
    "        \"\"\"\n",
    "        self.episode = 0\n",
    "        self.steps = 0\n",
    "        self.best_reward = -1000\n",
    "        self.learning = True\n",
    "        self.action_space = action_space\n",
    "        self.privous_state = hyper_params['privous_state']\n",
    "\n",
    "        \"\"\"\n",
    "            input_len The input length of the neural network. It equals to the length of the state vector.\n",
    "            output_len: The output length of the neural network. It is equal to the action space.\n",
    "            eval_model: The model for predicting action for the agent.\n",
    "            target_model: The model for calculating Q-value of next_state to update 'eval_model'.\n",
    "        \"\"\"\n",
    "        state = env.reset()\n",
    "        self.reward_true_weight = env.reward_true_weight\n",
    "        input_len = len(state) * self.privous_state + action_space\n",
    "        output_len = 1\n",
    "        self.state_len = len(state)\n",
    "        self.decom_reward_len = len(env.reward_true_weight)\n",
    "        self.action_vector = self.get_action_vector()\n",
    "        self.eval_model = feature_q_model(input_len, self.decom_reward_len, output_len, learning_rate = hyper_params['learning_rate'])\n",
    "        self.target_model = feature_q_model(input_len, self.decom_reward_len, output_len)\n",
    "#         memory: Store and sample experience replay.\n",
    "        self.memory = ReplayBuffer_decom(hyper_params['memory_size'])\n",
    "        \n",
    "        \"\"\"\n",
    "            batch_size: Mini batch size for training model.\n",
    "            update_steps: The frequence of traning model\n",
    "            model_replace_freq: The frequence of replacing 'target_model' by 'eval_model'\n",
    "        \"\"\"\n",
    "        \n",
    "        self.batch_size = hyper_params['batch_size']\n",
    "        self.update_steps = hyper_params['update_steps']\n",
    "        self.model_replace_freq = hyper_params['model_replace_freq']\n",
    "        \n",
    "        self.q_value_gd = []\n",
    "        self.v_feature_input= []\n",
    "        self.loss_accumulate = deque(maxlen=1000)\n",
    "        self.optimizer_com = self.eval_model.optimizer_com\n",
    "        self.loss_fn = self.eval_model.loss_fn\n",
    "        \n",
    "        if os.path.isdir(\"LunarLander_summary/Lunarlander/GQF/\"):\n",
    "            shutil.rmtree(\"LunarLander_summary/Lunarlander/GQF/\")\n",
    "        if os.path.isdir(\"LunarLander_summary/Lunarlander/GQF Q_f LOSS\"):\n",
    "            shutil.rmtree(\"LunarLander_summary/Lunarlander/GQF Q_f LOSS\")\n",
    "        if os.path.isdir(\"LunarLander_summary/Lunarlander/GQF Q_r LOSS\"):\n",
    "            shutil.rmtree(\"LunarLander_summary/Lunarlander/GQF Q_r LOSS\")\n",
    "        self.exp_count = 0\n",
    "        \n",
    "    # Linear decrease function for epsilon\n",
    "    def linear_decrease(self, initial_value, final_value, curr_steps, final_decay_steps):\n",
    "        decay_rate = curr_steps / final_decay_steps\n",
    "        if decay_rate > 1:\n",
    "            decay_rate = 1\n",
    "        return initial_value - (initial_value - final_value) * decay_rate\n",
    "    \n",
    "    def get_action_vector(self):\n",
    "        action_vector = np.zeros((self.action_space, self.action_space))\n",
    "        for i in range(len(action_vector)):\n",
    "            action_vector[i, i] = 1\n",
    "        \n",
    "        return FloatTensor(action_vector)\n",
    "    \n",
    "    def concat_state_action(self, states, actions = None, is_full_action = False):\n",
    "        if is_full_action:\n",
    "            com_state = FloatTensor(states).repeat((1, self.action_space)).view((-1, self.state_len))\n",
    "            actions = self.action_vector.repeat((len(states), 1))\n",
    "        else:\n",
    "            com_state = states.clone()\n",
    "            actions = actions.clone()\n",
    "        state_action = torch.cat((com_state, actions), 1)\n",
    "        return state_action\n",
    "        \n",
    "    def explore_or_exploit_policy(self, state):\n",
    "        p = uniform(0, 1)\n",
    "        # Get decreased epsilon\n",
    "        epsilon = self.linear_decrease(self.initial_epsilon, \n",
    "                               self.final_epsilon,\n",
    "                               self.steps,\n",
    "                               self.epsilon_decay_steps)\n",
    "        \n",
    "        if p < epsilon:\n",
    "            #return action, None\n",
    "            return randint(0, self.action_space - 1)\n",
    "        else:\n",
    "            #return action, Q-value\n",
    "            return self.greedy_policy(state)[0]\n",
    "        \n",
    "    def greedy_policy(self, state):\n",
    "        state_ft = FloatTensor(state).view(-1, self.state_len)\n",
    "        state_action = self.concat_state_action(state_ft, is_full_action = True)\n",
    "        feature_vectors, q_values = self.eval_model.predict_batch(state_action)\n",
    "        best_action = q_values.max(0)[1]\n",
    "        return best_action.item(), feature_vectors[best_action.item()]\n",
    "    \n",
    "    def update_batch(self):\n",
    "#         print(self.update_steps)\n",
    "        if len(self.memory) < self.batch_size or self.steps % self.update_steps != 0:\n",
    "            return\n",
    "\n",
    "        batch = self.memory.sample(self.batch_size)\n",
    "\n",
    "        (states_actions, _, reward, next_states,\n",
    "         is_terminal, rewards_decom) = batch\n",
    "        \n",
    "#         states_actions = states_actions\n",
    "        next_states = FloatTensor(next_states)\n",
    "        terminal = FloatTensor([1 if t else 0 for t in is_terminal])\n",
    "        reward = FloatTensor(reward)\n",
    "        rewards_decom = FloatTensor(rewards_decom)\n",
    "        batch_index = torch.arange(self.batch_size,\n",
    "                                   dtype=torch.long)\n",
    "        \n",
    "        # Current Q Values\n",
    "        feature_vector, q_values = self.eval_model.predict_batch(states_actions)\n",
    "        \n",
    "        next_state_actions = self.concat_state_action(next_states, is_full_action = True)\n",
    "        feature_vector_next, q_next = self.target_model.predict_batch(next_state_actions)\n",
    "        \n",
    "        q_next = q_next.view((-1, self.action_space))\n",
    "        feature_vector_next = feature_vector_next.view((-1, self.action_space, self.decom_reward_len))\n",
    "        \n",
    "        q_max, idx = q_next.max(1)\n",
    "        feature_vector_max = feature_vector_next[batch_index, idx]\n",
    "        q_max = (1 - terminal) * q_max\n",
    "        q_target = reward + self.beta * q_max\n",
    "        q_target = q_target.unsqueeze(1)\n",
    "        feature_vector_max = (1 - terminal.view(-1, 1)) * feature_vector_max\n",
    "        feature_vector_target = rewards_decom + self.beta * feature_vector_max\n",
    "        \n",
    "        self.eval_model.fit(q_values, q_target, feature_vector, feature_vector_target)\n",
    "        \n",
    "    def learn_and_evaluate(self, training_episodes, test_interval):\n",
    "        test_number = training_episodes // test_interval\n",
    "        all_results = []\n",
    "        \n",
    "        for i in range(test_number):\n",
    "            # learn\n",
    "            self.learn(test_interval)\n",
    "            f = open(result_file, \"a+\")\n",
    "            f.write(str(\"\\n***{}\".format((i + 1) * test_interval) + \"\\n\"))\n",
    "            f.close()\n",
    "            f = open(result_file_GVFs_loss, \"a+\")\n",
    "            f.write(str(\"\\n***{}\".format((i + 1) * test_interval) + \"\\n\"))\n",
    "            f.close()\n",
    "            # evaluate\n",
    "            avg_reward = self.evaluate((i + 1) * test_interval)\n",
    "            all_results.append(avg_reward)\n",
    "            \n",
    "        return all_results\n",
    "    \n",
    "    def learn(self, test_interval):\n",
    "        for episode in tqdm(range(test_interval), desc=\"Training\"):\n",
    "            state = self.env.reset()\n",
    "            done = False\n",
    "            steps = 0\n",
    "            \n",
    "            while steps < self.max_episode_steps and not done:\n",
    "                steps += 1\n",
    "                self.steps += 1\n",
    "                \n",
    "                action = self.explore_or_exploit_policy(state)\n",
    "                next_state, reward, done, _, rewards_decom = self.env.step(action)\n",
    "                action_vector = np.zeros(self.action_space)\n",
    "                action_vector[action] = 1\n",
    "                self.memory.add(np.concatenate((state.copy(), action_vector.copy()), axis=0), -1, reward, next_state, done, rewards_decom)\n",
    "                self.update_batch()\n",
    "                \n",
    "                if self.steps % self.model_replace_freq == 0:\n",
    "                    self.target_model.replace_soft(self.eval_model)\n",
    "                state = next_state\n",
    "                \n",
    "    def evaluate(self, episode_num, trials = 100):\n",
    "        total_reward = 0\n",
    "        total_steps = 0\n",
    "        all_GVFs_loss = 0\n",
    "        for _ in tqdm(range(trials), desc=\"Evaluating\"):\n",
    "            state = self.env.reset()\n",
    "            done = False\n",
    "            steps = 0\n",
    "            all_features_gt = FloatTensor(np.zeros((self.max_episode_steps, self.decom_reward_len)))\n",
    "            all_features_predict = FloatTensor(np.zeros((self.max_episode_steps, self.decom_reward_len)))\n",
    "            discounted_para = FloatTensor(np.zeros((self.max_episode_steps, 1)))\n",
    "            while steps < self.max_episode_steps and not done:\n",
    "                steps += 1\n",
    "                action, fv = self.greedy_policy(state)\n",
    "                state, reward, done, _, rewards_decom = self.env.step(action)\n",
    "                total_reward += reward\n",
    "                \n",
    "                all_features_gt[:steps] += (FloatTensor(rewards_decom) * (self.beta ** discounted_para))[:steps]\n",
    "                all_features_predict[steps - 1] = fv\n",
    "                discounted_para[:steps] += 1\n",
    "                \n",
    "            total_steps += steps\n",
    "            with torch.no_grad():\n",
    "                all_GVFs_loss += self.eval_model.loss_fn(all_features_gt[:steps], all_features_predict[:steps]).item()\n",
    "                \n",
    "        avg_reward = total_reward / trials\n",
    "        avg_GVFs_loss = all_GVFs_loss / trials\n",
    "        print(\"avg score: {}\".format(avg_reward))\n",
    "        print(\"avg GVF loss: {}\".format(avg_GVFs_loss))\n",
    "        f = open(result_file, \"a+\")\n",
    "        f.write(str(avg_reward) + \"\\n\")\n",
    "        f.close()\n",
    "        f = open(result_file_GVFs_loss, \"a+\")\n",
    "        f.write(str(avg_GVFs_loss) + \"\\n\")\n",
    "        f.close()\n",
    "        if avg_reward >= self.best_reward:\n",
    "            self.best_reward = avg_reward\n",
    "            self.save_model()\n",
    "            print(\"save\")\n",
    "        self.best_reward *= 0.99\n",
    "        Writer.add_scalars(main_tag='Lunarlander/DQN',\n",
    "                                tag_scalar_dict = {'GQF':avg_reward}, \n",
    "#                                 scalar_value=,\n",
    "                                global_step=episode_num)\n",
    "        return avg_reward\n",
    "\n",
    "#################################################################################################\n",
    "#################################################################################################  \n",
    "############################################### MSX #############################################  \n",
    "#################################################################################################  \n",
    "#################################################################################################  \n",
    "    def explore_or_exploit_policy_eval(self, state, epsilon = 0):\n",
    "        p = uniform(0, 1)\n",
    "        if p < epsilon:\n",
    "            #return action, None\n",
    "            return randint(0, self.action_space - 1)\n",
    "        else:\n",
    "            #return action, Q-value\n",
    "            return self.greedy_policy(state)[0]\n",
    "\n",
    "    def evaluate_combination_action_large_diff(self, example_num = 10, epsilon = 0, states = []):\n",
    "        self.eval_model.eval_mode()\n",
    "        loss_fn = nn.MSELoss()\n",
    "        \n",
    "        all_states = []\n",
    "        all_frames = []\n",
    "        state = self.env.reset()\n",
    "        done = False\n",
    "        steps = 0\n",
    "        \n",
    "        state_len = len(state)\n",
    "        \n",
    "#         q_vs = []\n",
    "        while steps < self.max_episode_steps and not done:\n",
    "            steps += 1\n",
    "            self.steps += 1\n",
    "\n",
    "            action = self.explore_or_exploit_policy_eval(state, epsilon = epsilon)\n",
    "#             q_vs.append(q_v.item())\n",
    "            state, reward, done, _, _ = self.env.step(action)\n",
    "            \n",
    "            all_frames.append(env.render(mode='rgb_array'))\n",
    "            \n",
    "            all_states.append(state.copy())\n",
    "        for s in states:\n",
    "            all_states.append(s)\n",
    "            all_frames.append(np.zeros_like(all_frames[0]))\n",
    "        examples = FloatTensor(all_states)\n",
    "        eval_frames = np.array(all_frames)\n",
    "\n",
    "        com_examples = self.concat_state_action(examples, is_full_action = True)\n",
    "        v_features, q_value = self.eval_model.predict_batch(com_examples)\n",
    "        q_value = q_value.view((-1, self.action_space))\n",
    "        v_features = v_features.view((-1, self.action_space, self.decom_reward_len))\n",
    "        \n",
    "        q_best_values, predict_best_action = q_value.max(1)\n",
    "        q_worst_values, predict_worst_action = q_value.min(1)\n",
    "        \n",
    "        q_diff = q_best_values - q_worst_values\n",
    "        \n",
    "        q_diff_idx = q_diff.argsort(descending = True)[:example_num]\n",
    "        if len(states) > 0:\n",
    "            q_diff_idx = torch.cat((q_diff_idx, LongTensor(list(range(len(all_frames) - len(states), len(all_frames))))))\n",
    "        q_diff_idx_np = np.array(q_diff_idx.tolist())\n",
    "        q_diff = q_diff[q_diff_idx]\n",
    "        print(\"diff example length: {}\".format(len(q_diff)))\n",
    "        q_value = q_value[q_diff_idx]\n",
    "        v_features = v_features[q_diff_idx]\n",
    "        q_best_values, predict_best_action = q_value.max(1)\n",
    "        q_worst_values, predict_worst_action = q_value.min(1)\n",
    "        \n",
    "        eval_frames = eval_frames[q_diff_idx_np]\n",
    "        examples = examples[q_diff_idx_np]\n",
    "        \n",
    "        total_acc_1 = 0\n",
    "        total_acc_2 = 0\n",
    "    \n",
    "        print(\"diff action value: {}\".format(q_diff))\n",
    "        count = 0\n",
    "        for i, vf in enumerate(v_features):\n",
    "            p_a = predict_best_action[i].item()\n",
    "            sub_actions = q_value[i].argsort()[:-1]\n",
    "    \n",
    "            print(\"=========================================================================\")\n",
    "            \n",
    "#             print(\"true target action: {}\\n true baseline action: {}\".format(action_name[t_a], action_name[true_sub_action]))\n",
    "            show_image = True\n",
    "            print_Cartpole(examples[i].tolist())\n",
    "        \n",
    "            IGs = []\n",
    "            ans = []\n",
    "            MSX_values = []\n",
    "            baseline_values = []\n",
    "            GVFs = []\n",
    "            q_print_values = []\n",
    "            GVFs.append(v_features[i][p_a].tolist())\n",
    "            ans.append(action_name[p_a])\n",
    "            q_print_values.append(q_value[i][p_a].item())\n",
    "            print(\"target action {}, value: {}\".format(action_name[p_a], q_best_values[i]))\n",
    "            for sb in sub_actions.tolist()[::-1]:\n",
    "#                 print(\"=========================================================================\")\n",
    "                sub_action = sb\n",
    "#                 print(\"target action: {}\\n baseline action: {}\".format(action_name[p_a], action_name[sub_action]))\n",
    "                ans.append(action_name[sub_action])\n",
    "                GVFs.append(v_features[i][sub_action].tolist())\n",
    "                msx_idx, msx_value, intergated_grad = self.differenc_vector_2(self.eval_model.q_model, \n",
    "                                      (v_features[i][p_a], eval_frames[i], q_value[i][p_a])\n",
    "                                      ,(v_features[i][sub_action], eval_frames[i], q_value[i][sub_action]), False, iteration = 30, show_image = show_image)\n",
    "                ig = np.array(intergated_grad[0].tolist())\n",
    "                IGs.append(ig)\n",
    "                baseline_values.append(q_value[i][sub_action].item())\n",
    "                q_print_values.append(q_value[i][sub_action].item())\n",
    "                orginal_pos_MSX = np.zeros(len(ig))\n",
    "                orginal_pos_MSX[msx_idx] = ig[msx_idx]\n",
    "                MSX_values.append(orginal_pos_MSX)\n",
    "                if show_image:\n",
    "                    show_image = False\n",
    "                    \n",
    "            for v, k in zip(baseline_values, ans):\n",
    "                print(\"baseline action {}, value: {}\".format(k, v))\n",
    "            \n",
    "            plot_action_group(np.array(GVFs).T, ans, FEATRUESNAME, title = \"GVFs\", y_label = \"GVFs Values\", q_values = q_print_values, exp_count = self.exp_count)\n",
    "            plot_IG_MSX(np.array(IGs).T, np.array(MSX_values).T, ans[1:], FEATRUESNAME, title = \"IG & MSX\", y_label = \"IGX\", IGX_action = ans[0], exp_count = self.exp_count)\n",
    "        \n",
    "    def differenc_vector_2(self, model, target, baseline, verbose = True, iteration = 100, show_image = True):\n",
    "        t_feature, t_frame, t_value = target\n",
    "        b_feature, b_frame, b_value = baseline\n",
    "        \n",
    "        if verbose or show_image:\n",
    "            plt.clf()\n",
    "            print(\"target value: {}\".format(t_value.item()))\n",
    "            print(\"baseline value: {}\".format(b_value.item()))\n",
    "            self.exp_count += 1\n",
    "            plt.imshow(t_frame)\n",
    "            plt.axis('off')\n",
    "            plt.savefig(\"{}/game_state-{}.png\".format(result_floder_exp, self.exp_count))\n",
    "        (msx_idx, msx_value), intergated_grad = self.intergated_gradients(model, t_feature, baseline = b_feature, verbose = verbose, iteration = iteration)\n",
    "        return msx_idx, msx_value, intergated_grad\n",
    "    def intergated_gradients(self, model, x, iteration = 100, baseline = None, verbose = True):\n",
    "        y_baseline = model(baseline).item()\n",
    "        x = x.view(1, -1)\n",
    "        x.size()[1]\n",
    "        if baseline is None:\n",
    "            baseline = torch.zeros_like(x)\n",
    "        elif verbose:\n",
    "            print(\"baseline: {}\".format(baseline))\n",
    "        \n",
    "        intergated_grad = torch.zeros_like(x)\n",
    "    \n",
    "        for i in range(iteration):\n",
    "            new_input = baseline + ((i + 1) / iteration * (x - baseline))\n",
    "            new_input = new_input.clone().detach().requires_grad_(True)\n",
    "    #         print(new_input)\n",
    "\n",
    "            y = model(new_input)\n",
    "            loss = abs(y_baseline - y)\n",
    "            self.optimizer_com.zero_grad()\n",
    "            loss.backward()\n",
    "            intergated_grad += (new_input.grad) / iteration\n",
    "            \n",
    "        intergated_grad *= x - baseline\n",
    "        \n",
    "        print(\"target GVFs: {}\".format(x.tolist()))\n",
    "        print(\"baseline GVFs: {}\".format(baseline.tolist()))\n",
    "        MSX_idx, MSX_values = MSX(intergated_grad.tolist()[0])\n",
    "        if verbose:\n",
    "            msx_vector = np.zeros(self.decom_reward_len)\n",
    "            msx_vector[MSX_idx] = MSX_values\n",
    "            plot_msx_plus(msx_vector, tittle = 'MSX+')\n",
    "    \n",
    "        return (MSX_idx, MSX_values), intergated_grad\n",
    "    \n",
    "    def generate_video(self, video_name):\n",
    "        self.eval_model.eval_mode()\n",
    "        loss_fn = nn.MSELoss()\n",
    "        total_reward = 0\n",
    "        all_frames = []\n",
    "        state = self.env.reset()\n",
    "        done = False\n",
    "        steps = 0\n",
    "        \n",
    "        while steps < self.max_episode_steps and not done:\n",
    "            steps += 1\n",
    "\n",
    "            action, q_v, feature_vector = self.greedy_policy(state)\n",
    "            state, reward, done, _ = self.env.step(action)\n",
    "            total_reward += reward\n",
    "            all_frames.append(env.render(mode='rgb_array'))\n",
    "        print(total_reward)\n",
    "        display_frames_as_gif(all_frames, video_name)\n",
    "        print(\"video generted\")\n",
    "\n",
    "    def save_model(self, path = None, name = \"best_model.pt\"):\n",
    "        if path is None:\n",
    "            path = result_floder\n",
    "        self.eval_model.save(\"{}/{}\".format(path, name))\n",
    "        \n",
    "    # load model\n",
    "    def load_model(self, path = \"\", name = \"best_model.pt\"):\n",
    "        if path is None:\n",
    "            path = result_floder\n",
    "        self.eval_model.load(\"{}/{}\".format(path, name))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Train Agent"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Training: 100%|██████████| 1/1 [00:00<00:00,  9.56it/s]\n",
      "Evaluating:   0%|          | 0/100 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "*************0*************\n",
      "Using GPU\n",
      "Using GPU\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Evaluating: 100%|██████████| 100/100 [00:05<00:00, 17.29it/s]\n",
      "Training: 100%|██████████| 1/1 [00:00<00:00,  6.75it/s]\n",
      "Evaluating:   0%|          | 0/100 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "avg score: -467.76467760885504\n",
      "avg GVF loss: 0.8125906953215599\n",
      "save\n",
      "*************1*************\n",
      "Using GPU\n",
      "Using GPU\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Evaluating: 100%|██████████| 100/100 [00:05<00:00, 18.26it/s]\n",
      "Training: 100%|██████████| 1/1 [00:00<00:00, 12.49it/s]\n",
      "Evaluating:   0%|          | 0/100 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "avg score: -520.776397956336\n",
      "avg GVF loss: 0.7738027387857437\n",
      "save\n",
      "*************2*************\n",
      "Using GPU\n",
      "Using GPU\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Evaluating: 100%|██████████| 100/100 [00:11<00:00,  8.79it/s]\n",
      "Training: 100%|██████████| 1/1 [00:00<00:00,  8.07it/s]\n",
      "Evaluating:   0%|          | 0/100 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "avg score: -232.1571990997025\n",
      "avg GVF loss: 0.20456642724573612\n",
      "save\n",
      "*************3*************\n",
      "Using GPU\n",
      "Using GPU\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Evaluating: 100%|██████████| 100/100 [00:07<00:00, 12.69it/s]\n",
      "Training: 100%|██████████| 1/1 [00:00<00:00, 14.52it/s]\n",
      "Evaluating:   2%|▏         | 2/100 [00:00<00:05, 17.02it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "avg score: -351.7423305052292\n",
      "avg GVF loss: 0.304332318007946\n",
      "save\n",
      "*************4*************\n",
      "Using GPU\n",
      "Using GPU\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Evaluating: 100%|██████████| 100/100 [00:05<00:00, 18.44it/s]\n",
      "Training: 100%|██████████| 1/1 [00:00<00:00,  9.77it/s]\n",
      "Evaluating:   0%|          | 0/100 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "avg score: -131.80959255570903\n",
      "avg GVF loss: 0.24804360240697862\n",
      "save\n",
      "*************5*************\n",
      "Using GPU\n",
      "Using GPU\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Evaluating: 100%|██████████| 100/100 [00:05<00:00, 17.42it/s]\n",
      "Training: 100%|██████████| 1/1 [00:00<00:00, 16.92it/s]\n",
      "Evaluating:   0%|          | 0/100 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "avg score: -119.63983266611609\n",
      "avg GVF loss: 0.2531967306882143\n",
      "save\n",
      "*************6*************\n",
      "Using GPU\n",
      "Using GPU\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Evaluating: 100%|██████████| 100/100 [00:09<00:00, 10.90it/s]\n",
      "Training: 100%|██████████| 1/1 [00:00<00:00,  9.48it/s]\n",
      "Evaluating:   0%|          | 0/100 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "avg score: -867.0144351084491\n",
      "avg GVF loss: 1.902255216985941\n",
      "save\n",
      "*************7*************\n",
      "Using GPU\n",
      "Using GPU\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Evaluating: 100%|██████████| 100/100 [00:05<00:00, 18.19it/s]\n",
      "Training: 100%|██████████| 1/1 [00:00<00:00,  9.37it/s]\n",
      "Evaluating:   0%|          | 0/100 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "avg score: -538.7255830605329\n",
      "avg GVF loss: 0.8320143651962281\n",
      "save\n",
      "*************8*************\n",
      "Using GPU\n",
      "Using GPU\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Evaluating: 100%|██████████| 100/100 [00:08<00:00, 11.70it/s]\n",
      "Training: 100%|██████████| 1/1 [00:00<00:00, 11.17it/s]\n",
      "Evaluating:   0%|          | 0/100 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "avg score: -306.30542055085397\n",
      "avg GVF loss: 0.3470861122012138\n",
      "save\n",
      "*************9*************\n",
      "Using GPU\n",
      "Using GPU\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Evaluating: 100%|██████████| 100/100 [00:11<00:00,  8.86it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "avg score: -792.1922465553308\n",
      "avg GVF loss: 0.8166571001708508\n",
      "save\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "training_episodes, test_interval = 10000, 200\n",
    "for i in range(10):\n",
    "    print(\"*************{}*************\".format(i))\n",
    "    agent = ESP_agent(env, hyperparams_Lunarlander)\n",
    "    result = agent.learn_and_evaluate(training_episodes, test_interval)\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Using GPU\n",
      "Using GPU\n",
      "diff example length: 5\n",
      "diff action value: tensor([75.5785, 35.7887, 33.3691, 32.1456, 31.3678], device='cuda:0',\n",
      "       grad_fn=<IndexBackward>)\n",
      "=========================================================================\n",
      "Cart Position:  -0.08127546310424805  Cart Velocity:  0.0614144429564476\n",
      "Pole Angle:  -0.0720595270395279  Pole Velocity At Tip:  -0.5613115429878235\n",
      "target action main engine, value: 120.38631439208984\n",
      "target value: 120.38631439208984\n",
      "baseline value: 104.25971984863281\n",
      "target GVFs: [[0.006120526231825352, -0.4026716947555542, -0.5928359627723694, 0.5927577018737793, -0.3891296982765198, 0.233500674366951, 0.18705502152442932, 0.27623414993286133]]\n",
      "baseline GVFs: [0.01819036155939102, -0.3553779721260071, -0.5804427266120911, 0.6032930612564087, -0.35235029458999634, 0.22107890248298645, 0.2150578498840332, 0.15096738934516907]\n",
      "target GVFs: [[0.006120526231825352, -0.4026716947555542, -0.5928359627723694, 0.5927577018737793, -0.3891296982765198, 0.233500674366951, 0.18705502152442932, 0.27623414993286133]]\n",
      "baseline GVFs: [0.022786982357501984, -0.3416402339935303, -0.5562114119529724, 0.6014419794082642, -0.26871997117996216, 0.22201456129550934, 0.19472350180149078, 0.11214524507522583]\n",
      "target GVFs: [[0.006120526231825352, -0.4026716947555542, -0.5928359627723694, 0.5927577018737793, -0.3891296982765198, 0.233500674366951, 0.18705502152442932, 0.27623414993286133]]\n",
      "baseline GVFs: [0.049440063536167145, -0.2698545455932617, -0.4356043040752411, 0.5715351104736328, -0.2636503279209137, 0.17991353571414948, 0.17292048037052155, -0.09635888040065765]\n",
      "baseline action main engine, value: 104.25971984863281\n",
      "baseline action left engine, value: 97.23545837402344\n",
      "baseline action Nop, value: 44.807830810546875\n",
      "0.1111111111111111\n",
      "=========================================================================\n",
      "Cart Position:  -0.23836001753807068  Cart Velocity:  -0.012313475832343102\n",
      "Pole Angle:  -0.2482163906097412  Pole Velocity At Tip:  -0.16947539150714874\n",
      "target action main engine, value: 43.7557258605957\n",
      "target value: 43.7557258605957\n",
      "baseline value: 35.79540252685547\n",
      "target GVFs: [[0.03001299872994423, -0.22508423030376434, -0.0033869370818138123, -0.1136551946401596, -0.22608759999275208, 0.10405217111110687, 0.2021036446094513, 0.3249448239803314]]\n",
      "baseline GVFs: [0.04222574830055237, -0.200609490275383, 0.014310844242572784, -0.07523785531520844, -0.21468812227249146, 0.07502073049545288, 0.16636529564857483, 0.2993684411048889]\n",
      "target GVFs: [[0.03001299872994423, -0.22508423030376434, -0.0033869370818138123, -0.1136551946401596, -0.22608759999275208, 0.10405217111110687, 0.2021036446094513, 0.3249448239803314]]\n",
      "baseline GVFs: [0.026056379079818726, -0.17926423251628876, 0.011690519750118256, -0.10768027603626251, -0.2023150622844696, 0.08799313008785248, 0.17728757858276367, 0.21630549430847168]\n",
      "target GVFs: [[0.03001299872994423, -0.22508423030376434, -0.0033869370818138123, -0.1136551946401596, -0.22608759999275208, 0.10405217111110687, 0.2021036446094513, 0.3249448239803314]]\n",
      "baseline GVFs: [0.039827510714530945, -0.17859292030334473, 0.016211621463298798, -0.11477531492710114, -0.21361637115478516, 0.10414275527000427, 0.1882340908050537, 0.14424729347229004]\n",
      "baseline action main engine, value: 35.79540252685547\n",
      "baseline action Nop, value: 21.912046432495117\n",
      "baseline action left engine, value: 7.967008590698242\n",
      "0.1111111111111111\n",
      "=========================================================================\n",
      "Cart Position:  -0.08062563091516495  Cart Velocity:  0.07394793629646301\n",
      "Pole Angle:  -0.029979348182678223  Pole Velocity At Tip:  -0.5549432039260864\n",
      "target action main engine, value: 118.52242279052734\n",
      "target value: 118.52242279052734\n",
      "baseline value: 118.21977233886719\n",
      "target GVFs: [[-0.04625726491212845, -0.34785065054893494, -0.5248734354972839, 0.5316280722618103, 0.6065350770950317, 0.25612571835517883, 0.2030850201845169, 0.23808026313781738]]\n",
      "baseline GVFs: [-0.031119562685489655, -0.33795690536499023, -0.5428862571716309, 0.5708529949188232, 0.6166709065437317, 0.2421310395002365, 0.18065723776817322, 0.21851935982704163]\n",
      "target GVFs: [[-0.04625726491212845, -0.34785065054893494, -0.5248734354972839, 0.5316280722618103, 0.6065350770950317, 0.25612571835517883, 0.2030850201845169, 0.23808026313781738]]\n",
      "baseline GVFs: [-0.03152026981115341, -0.30161160230636597, -0.48614802956581116, 0.5445380210876465, 0.6348210573196411, 0.22748731076717377, 0.18023206293582916, 0.16063374280929565]\n",
      "target GVFs: [[-0.04625726491212845, -0.34785065054893494, -0.5248734354972839, 0.5316280722618103, 0.6065350770950317, 0.25612571835517883, 0.2030850201845169, 0.23808026313781738]]\n",
      "baseline GVFs: [-0.0294480100274086, -0.28302207589149475, -0.4208415448665619, 0.5475295782089233, 0.6039595603942871, 0.19868279993534088, 0.17240551114082336, 0.06723286956548691]\n",
      "baseline action main engine, value: 118.21977233886719\n",
      "baseline action left engine, value: 103.93346405029297\n",
      "baseline action Nop, value: 85.15330505371094\n",
      "0.1111111111111111\n",
      "=========================================================================\n",
      "Cart Position:  -0.1251538246870041  Cart Velocity:  0.006278667598962784\n",
      "Pole Angle:  -0.3692109286785126  Pole Velocity At Tip:  -0.06501534581184387\n",
      "target action main engine, value: 77.49540710449219\n",
      "target value: 77.49540710449219\n",
      "baseline value: 77.3550796508789\n",
      "target GVFs: [[0.038595087826251984, -0.32144272327423096, -0.026825636625289917, -0.2144901007413864, 0.6863986849784851, 0.12570884823799133, 0.23249362409114838, 0.4857533574104309]]\n",
      "baseline GVFs: [0.026356719434261322, -0.33293479681015015, -0.031917594373226166, -0.22553037106990814, 0.700491189956665, 0.12258705496788025, 0.23712654411792755, 0.4501154124736786]\n",
      "target GVFs: [[0.038595087826251984, -0.32144272327423096, -0.026825636625289917, -0.2144901007413864, 0.6863986849784851, 0.12570884823799133, 0.23249362409114838, 0.4857533574104309]]\n",
      "baseline GVFs: [0.0305742546916008, -0.3121856451034546, -0.02591213583946228, -0.23395784199237823, 0.6660795211791992, 0.13467590510845184, 0.240104541182518, 0.3676728308200836]\n",
      "target GVFs: [[0.038595087826251984, -0.32144272327423096, -0.026825636625289917, -0.2144901007413864, 0.6863986849784851, 0.12570884823799133, 0.23249362409114838, 0.4857533574104309]]\n",
      "baseline GVFs: [0.05190097540616989, -0.2885543406009674, -0.0401221439242363, -0.24418579041957855, 0.5373845100402832, 0.1403590440750122, 0.1822822093963623, 0.23662078380584717]\n",
      "baseline action main engine, value: 77.3550796508789\n",
      "baseline action Nop, value: 65.6366958618164\n",
      "baseline action right engine, value: 45.34980392456055\n",
      "0.1111111111111111\n",
      "=========================================================================\n",
      "Cart Position:  -0.08040132373571396  Cart Velocity:  0.08633387833833694\n",
      "Pole Angle:  0.025311851873993874  Pole Velocity At Tip:  -0.5710498094558716\n",
      "target action main engine, value: 125.9283218383789\n",
      "target value: 125.9283218383789\n",
      "baseline value: 122.22525787353516\n",
      "target GVFs: [[-0.06020388752222061, -0.3727439045906067, -0.5396175980567932, 0.5565600991249084, 0.6217281222343445, 0.25732243061065674, 0.198570117354393, 0.2688293755054474]]\n",
      "baseline GVFs: [-0.042772166430950165, -0.35153788328170776, -0.547016441822052, 0.5832228064537048, 0.6292094588279724, 0.2387813776731491, 0.17525117099285126, 0.23033177852630615]\n",
      "target GVFs: [[-0.06020388752222061, -0.3727439045906067, -0.5396175980567932, 0.5565600991249084, 0.6217281222343445, 0.25732243061065674, 0.198570117354393, 0.2688293755054474]]\n",
      "baseline GVFs: [-0.04130113869905472, -0.3226379156112671, -0.5062272548675537, 0.5654840469360352, 0.6415852308273315, 0.22821569442749023, 0.17743270099163055, 0.18621447682380676]\n",
      "target GVFs: [[-0.06020388752222061, -0.3727439045906067, -0.5396175980567932, 0.5565600991249084, 0.6217281222343445, 0.25732243061065674, 0.198570117354393, 0.2688293755054474]]\n",
      "baseline GVFs: [-0.04440069943666458, -0.3026583194732666, -0.4418947994709015, 0.5719107389450073, 0.6183648109436035, 0.19978594779968262, 0.1682203710079193, 0.09349025785923004]\n",
      "baseline action main engine, value: 122.22525787353516\n",
      "baseline action left engine, value: 111.818603515625\n",
      "baseline action Nop, value: 94.56049346923828\n",
      "0.1111111111111111\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "\u001b[0;32m<ipython-input-22-da43260becdc>\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[1;32m      2\u001b[0m \u001b[0magent\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mESP_agent\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0menv\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mhyperparams_Lunarlander\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      3\u001b[0m \u001b[0magent\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mload_model\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mpath\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mbest_ckpt_path\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m----> 4\u001b[0;31m \u001b[0magent\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mevaluate_combination_action_large_diff\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mexample_num\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;36m5\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mepsilon\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;36m0.1\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m",
      "\u001b[0;32m<ipython-input-16-116debd83402>\u001b[0m in \u001b[0;36mevaluate_combination_action_large_diff\u001b[0;34m(self, example_num, epsilon, states)\u001b[0m\n\u001b[1;32m    357\u001b[0m                 \u001b[0mprint\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m\"baseline action {}, value: {}\"\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mformat\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mk\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mv\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    358\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 359\u001b[0;31m             \u001b[0mplot_action_group\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mnp\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0marray\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mGVFs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mT\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mans\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mFEATRUESNAME\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mtitle\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;34m\"GVFs\"\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0my_label\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;34m\"GVFs Values\"\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mq_values\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mq_print_values\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mexp_count\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mexp_count\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    360\u001b[0m             \u001b[0mplot_IG_MSX\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mnp\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0marray\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mIGs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mT\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mnp\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0marray\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mMSX_values\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mT\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mans\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;36m1\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mFEATRUESNAME\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mtitle\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;34m\"IG & MSX\"\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0my_label\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;34m\"IGX\"\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mIGX_action\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mans\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;36m0\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mexp_count\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mexp_count\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    361\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m<ipython-input-8-71ced3f6fac0>\u001b[0m in \u001b[0;36mplot_action_group\u001b[0;34m(values, group, elements, title, y_label, q_values, IGX_action, exp_count)\u001b[0m\n\u001b[1;32m     83\u001b[0m             \u001b[0mplt\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mgca\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtext\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mi\u001b[0m \u001b[0;34m+\u001b[0m \u001b[0mcenter_pos\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mtxt_pos\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m\"Q_v ≈ {}\"\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mformat\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mnp\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mround\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mq_values\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0mi\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;36m4\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcolor\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;34m'black'\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mfontweight\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;34m'ultralight'\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mha\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;34m'center'\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     84\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 85\u001b[0;31m     \u001b[0mplt\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0msavefig\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m\"{}/{}-{}.png\"\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mformat\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mresult_floder_exp\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mtitle\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mexp_count\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     86\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     87\u001b[0m \u001b[0;32mdef\u001b[0m \u001b[0mdisplay_frames_as_gif\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mframes\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mvideo_name\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.local/lib/python3.6/site-packages/matplotlib/pyplot.py\u001b[0m in \u001b[0;36msavefig\u001b[0;34m(*args, **kwargs)\u001b[0m\n\u001b[1;32m    722\u001b[0m     \u001b[0mfig\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mgcf\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    723\u001b[0m     \u001b[0mres\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mfig\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0msavefig\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0margs\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 724\u001b[0;31m     \u001b[0mfig\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mcanvas\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdraw_idle\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m   \u001b[0;31m# need this if 'transparent=True' to reset colors\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    725\u001b[0m     \u001b[0;32mreturn\u001b[0m \u001b[0mres\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    726\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.local/lib/python3.6/site-packages/matplotlib/backend_bases.py\u001b[0m in \u001b[0;36mdraw_idle\u001b[0;34m(self, *args, **kwargs)\u001b[0m\n\u001b[1;32m   1929\u001b[0m         \u001b[0;32mif\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_is_idle_drawing\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1930\u001b[0m             \u001b[0;32mwith\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_idle_draw_cntx\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 1931\u001b[0;31m                 \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0margs\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   1932\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1933\u001b[0m     \u001b[0;34m@\u001b[0m\u001b[0mcbook\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdeprecated\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m\"3.2\"\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.local/lib/python3.6/site-packages/matplotlib/backends/backend_agg.py\u001b[0m in \u001b[0;36mdraw\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m    391\u001b[0m              (self.toolbar._wait_cursor_for_draw_cm() if self.toolbar\n\u001b[1;32m    392\u001b[0m               else nullcontext()):\n\u001b[0;32m--> 393\u001b[0;31m             \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mfigure\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mrenderer\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    394\u001b[0m             \u001b[0;31m# A GUI class may be need to update a window using this draw, so\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    395\u001b[0m             \u001b[0;31m# don't forget to call the superclass.\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.local/lib/python3.6/site-packages/matplotlib/artist.py\u001b[0m in \u001b[0;36mdraw_wrapper\u001b[0;34m(artist, renderer, *args, **kwargs)\u001b[0m\n\u001b[1;32m     36\u001b[0m                 \u001b[0mrenderer\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mstart_filter\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     37\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 38\u001b[0;31m             \u001b[0;32mreturn\u001b[0m \u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0martist\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mrenderer\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m*\u001b[0m\u001b[0margs\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     39\u001b[0m         \u001b[0;32mfinally\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     40\u001b[0m             \u001b[0;32mif\u001b[0m \u001b[0martist\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mget_agg_filter\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mis\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0;32mNone\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.local/lib/python3.6/site-packages/matplotlib/figure.py\u001b[0m in \u001b[0;36mdraw\u001b[0;34m(self, renderer)\u001b[0m\n\u001b[1;32m   1734\u001b[0m             \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mpatch\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mrenderer\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1735\u001b[0m             mimage._draw_list_compositing_images(\n\u001b[0;32m-> 1736\u001b[0;31m                 renderer, self, artists, self.suppressComposite)\n\u001b[0m\u001b[1;32m   1737\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1738\u001b[0m             \u001b[0mrenderer\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mclose_group\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m'figure'\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.local/lib/python3.6/site-packages/matplotlib/image.py\u001b[0m in \u001b[0;36m_draw_list_compositing_images\u001b[0;34m(renderer, parent, artists, suppress_composite)\u001b[0m\n\u001b[1;32m    135\u001b[0m     \u001b[0;32mif\u001b[0m \u001b[0mnot_composite\u001b[0m \u001b[0;32mor\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0mhas_images\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    136\u001b[0m         \u001b[0;32mfor\u001b[0m \u001b[0ma\u001b[0m \u001b[0;32min\u001b[0m \u001b[0martists\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 137\u001b[0;31m             \u001b[0ma\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mrenderer\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    138\u001b[0m     \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    139\u001b[0m         \u001b[0;31m# Composite any adjacent images together\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.local/lib/python3.6/site-packages/matplotlib/artist.py\u001b[0m in \u001b[0;36mdraw_wrapper\u001b[0;34m(artist, renderer, *args, **kwargs)\u001b[0m\n\u001b[1;32m     36\u001b[0m                 \u001b[0mrenderer\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mstart_filter\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     37\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 38\u001b[0;31m             \u001b[0;32mreturn\u001b[0m \u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0martist\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mrenderer\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m*\u001b[0m\u001b[0margs\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     39\u001b[0m         \u001b[0;32mfinally\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     40\u001b[0m             \u001b[0;32mif\u001b[0m \u001b[0martist\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mget_agg_filter\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mis\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0;32mNone\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.local/lib/python3.6/site-packages/matplotlib/axes/_base.py\u001b[0m in \u001b[0;36mdraw\u001b[0;34m(self, renderer, inframe)\u001b[0m\n\u001b[1;32m   2628\u001b[0m             \u001b[0mrenderer\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mstop_rasterizing\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2629\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 2630\u001b[0;31m         \u001b[0mmimage\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_draw_list_compositing_images\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mrenderer\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0martists\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   2631\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2632\u001b[0m         \u001b[0mrenderer\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mclose_group\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m'axes'\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.local/lib/python3.6/site-packages/matplotlib/image.py\u001b[0m in \u001b[0;36m_draw_list_compositing_images\u001b[0;34m(renderer, parent, artists, suppress_composite)\u001b[0m\n\u001b[1;32m    135\u001b[0m     \u001b[0;32mif\u001b[0m \u001b[0mnot_composite\u001b[0m \u001b[0;32mor\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0mhas_images\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    136\u001b[0m         \u001b[0;32mfor\u001b[0m \u001b[0ma\u001b[0m \u001b[0;32min\u001b[0m \u001b[0martists\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 137\u001b[0;31m             \u001b[0ma\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mrenderer\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    138\u001b[0m     \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    139\u001b[0m         \u001b[0;31m# Composite any adjacent images together\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.local/lib/python3.6/site-packages/matplotlib/artist.py\u001b[0m in \u001b[0;36mdraw_wrapper\u001b[0;34m(artist, renderer, *args, **kwargs)\u001b[0m\n\u001b[1;32m     36\u001b[0m                 \u001b[0mrenderer\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mstart_filter\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     37\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 38\u001b[0;31m             \u001b[0;32mreturn\u001b[0m \u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0martist\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mrenderer\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m*\u001b[0m\u001b[0margs\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     39\u001b[0m         \u001b[0;32mfinally\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     40\u001b[0m             \u001b[0;32mif\u001b[0m \u001b[0martist\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mget_agg_filter\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mis\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0;32mNone\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.local/lib/python3.6/site-packages/matplotlib/axis.py\u001b[0m in \u001b[0;36mdraw\u001b[0;34m(self, renderer, *args, **kwargs)\u001b[0m\n\u001b[1;32m   1230\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1231\u001b[0m         \u001b[0;32mfor\u001b[0m \u001b[0mtick\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mticks_to_draw\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 1232\u001b[0;31m             \u001b[0mtick\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mrenderer\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   1233\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1234\u001b[0m         \u001b[0;31m# scale up the axis label box to also find the neighbors, not\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.local/lib/python3.6/site-packages/matplotlib/artist.py\u001b[0m in \u001b[0;36mdraw_wrapper\u001b[0;34m(artist, renderer, *args, **kwargs)\u001b[0m\n\u001b[1;32m     36\u001b[0m                 \u001b[0mrenderer\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mstart_filter\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     37\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 38\u001b[0;31m             \u001b[0;32mreturn\u001b[0m \u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0martist\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mrenderer\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m*\u001b[0m\u001b[0margs\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     39\u001b[0m         \u001b[0;32mfinally\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     40\u001b[0m             \u001b[0;32mif\u001b[0m \u001b[0martist\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mget_agg_filter\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mis\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0;32mNone\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.local/lib/python3.6/site-packages/matplotlib/axis.py\u001b[0m in \u001b[0;36mdraw\u001b[0;34m(self, renderer)\u001b[0m\n\u001b[1;32m    295\u001b[0m         for artist in [self.gridline, self.tick1line, self.tick2line,\n\u001b[1;32m    296\u001b[0m                        self.label1, self.label2]:\n\u001b[0;32m--> 297\u001b[0;31m             \u001b[0martist\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mrenderer\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    298\u001b[0m         \u001b[0mrenderer\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mclose_group\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m__name__\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    299\u001b[0m         \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mstale\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;32mFalse\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.local/lib/python3.6/site-packages/matplotlib/artist.py\u001b[0m in \u001b[0;36mdraw_wrapper\u001b[0;34m(artist, renderer, *args, **kwargs)\u001b[0m\n\u001b[1;32m     36\u001b[0m                 \u001b[0mrenderer\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mstart_filter\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     37\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 38\u001b[0;31m             \u001b[0;32mreturn\u001b[0m \u001b[0mdraw\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0martist\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mrenderer\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m*\u001b[0m\u001b[0margs\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     39\u001b[0m         \u001b[0;32mfinally\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     40\u001b[0m             \u001b[0;32mif\u001b[0m \u001b[0martist\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mget_agg_filter\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mis\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0;32mNone\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.local/lib/python3.6/site-packages/matplotlib/text.py\u001b[0m in \u001b[0;36mdraw\u001b[0;34m(self, renderer)\u001b[0m\n\u001b[1;32m    731\u001b[0m                     textrenderer.draw_text(gc, x, y, clean_line,\n\u001b[1;32m    732\u001b[0m                                            \u001b[0mtextobj\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_fontproperties\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mangle\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 733\u001b[0;31m                                            ismath=ismath, mtext=mtext)\n\u001b[0m\u001b[1;32m    734\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    735\u001b[0m         \u001b[0mgc\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mrestore\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.local/lib/python3.6/site-packages/matplotlib/backends/backend_agg.py\u001b[0m in \u001b[0;36mdraw_text\u001b[0;34m(self, gc, x, y, s, prop, angle, ismath, mtext)\u001b[0m\n\u001b[1;32m    190\u001b[0m         \u001b[0mxd\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0md\u001b[0m \u001b[0;34m*\u001b[0m \u001b[0msin\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mradians\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mangle\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    191\u001b[0m         \u001b[0myd\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0md\u001b[0m \u001b[0;34m*\u001b[0m \u001b[0mcos\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mradians\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mangle\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 192\u001b[0;31m         \u001b[0mx\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mround\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mx\u001b[0m \u001b[0;34m+\u001b[0m \u001b[0mxo\u001b[0m \u001b[0;34m+\u001b[0m \u001b[0mxd\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    193\u001b[0m         \u001b[0my\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mround\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0my\u001b[0m \u001b[0;34m+\u001b[0m \u001b[0myo\u001b[0m \u001b[0;34m+\u001b[0m \u001b[0myd\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    194\u001b[0m         \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_renderer\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdraw_text_image\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mfont\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mx\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0my\u001b[0m \u001b[0;34m+\u001b[0m \u001b[0;36m1\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mangle\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mgc\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 432x288 with 0 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 432x288 with 0 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 432x288 with 0 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 432x288 with 0 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 432x288 with 0 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 432x288 with 0 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 432x288 with 0 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 432x288 with 0 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 432x288 with 0 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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q1enevXq3HLLLSxbtizbscXFxfHEE0/QsGFDKlSoQOPGjXn55ZfTfE/j4uJ49tlnadq0KZUqVeKaa67h7bff5vz585l6Dd5SzqXkzft64MABhgwZQqNGjahQoQL16tWjf//+7Nq1K9PxbNiwgbvuuovLL7+cChUq0LRpU4YPH87JkyezFZs3vw8nT55kzJgxXH311VSqVImmTZsycuTINGPwFeVd2u655x6uv/76DPfZt28f1apVY+LEiV7FEB0dTe/evalRowYVK1akdevWTJ069aLPu1jse/bsoV+/ftSpU4cKFSrQvHlzxo0bl+Y5KzExkQ8//JDWrVtTqVIlrrjiCgYOHMi+ffvSaDnnBEP+efN5dqF77rmHsLAw/vjjj0wfr1GjRumec5YsWZJi39OnTzN27FiaN29OxYoVadq0Kc8//zzHjh1L1e6pU6cYPXo0zZo183wmTp48Oc0hLb6oA/IDDcvJJ5KSkhgzZgxvvfUWRYoUoW3btnTs2JEjR46wYsUKhgwZwrvvvssnn3xCnTp1/B2uT/z+++8MHDgw3ce3bNlChw4diI+PJzIykho1arBx40b++9//8t1337FkyRIuvfRSz/7R0dEULlyYwYMHp2orM8OtVq5cSa9evTh9+jSRkZFUqVKF7777jltvvZXXXnuNBx54IEuxufeJjY3luuuuo0ePHqxevZp33nmH1atXM3/+fAoWdH5VT5w44dm3Q4cOdO7cmZ9++olhw4bx448/MnPmTIwxmf4/zohyLjVv3tcDBw5w4403snfvXv7973/TvXt3tm3bxhdffMGSJUv49ttvufzyyzOMZ8WKFfTs2ROAm2++mfDwcH788UcmTJjAihUrmD9/PkWKFPE6Nsj878P58+e5/fbbWblyJddddx0dOnRg06ZNjBs3jmXLlrFgwQJPDL6gvEvfO++8w7x582jUqFG6+8THx9OnTx9OnDjhVQybN28mMjKS06dP07VrV8qVK8fChQsZMmQIu3btYtiwYVmK/a+//qJdu3YcPHiQyMhIatasyXfffcfIkSPZtGkTH330UYr9BwwYwOeff07jxo154IEHiI2NZebMmaxcuZLvv/+e0NBQr16Xt4Il/7z5PLvQ3LlzmTdvnlfHO3LkCH/99RfNmjWjbdu2qR6vWbOm5/a5c+fo1asXUVFRtGrVio4dO7Ju3Tref/991qxZw9dff01ISAjg/DHYt29fFi9eTLt27ejSpQtLlizh6aefZvfu3YwcOTLFcbJbB+QXKu7ziTFjxvDmm2/SokULpkyZQuXKlT2PuXs6XnzxRTp27MiqVasoX768H6PNvhUrVnD//fdz+PDhdPd54YUXOH78OB9//DE333yzZ/sbb7zBK6+8wuuvv86rr77q2R4dHU3dunV59tlnvY4nMTGRRx99lISEBP773/9yyy23AJCQkEDPnj0ZOnQobdq08RRr3sQ2depUYmNj6d+/P2PGjAHAWstDDz3E559/zueff84dd9wBwPjx44mNjWXMmDH079/f0+6DDz7IrFmzWLx4MTfddJPXry8tyrnUvHlfX3vtNfbu3cvIkSN55JFHPPt+9tlnPPTQQ7z44ovMnDkzw5iefPJJkpKSWLhwIc2aNQOc3Hj88ceZNm0aU6ZM8bSdU78PM2bMYOXKlTz88MOMHj3as33EiBGMHz+eGTNmZFgIeEt5l1piYiIjRozgnXfeybCtPXv20KdPHzZu3Oh1HKNHj+bkyZNMnz6dTp06AfD8889zww038J///Ie+fftSrVo1r2OfMGECBw8eZPTo0Tz88MMAvPTSS3Tr1o2vvvqK+++/n1atWgEwZ84cPv/8c7p3786HH37IJZc4gwveeecdXn75ZT744AOefvppr1+bN4Ih/7z9PEvuyJEjWXoPoqOjAbj11ltTfHal5f333ycqKopHH32U4cOHe7Y//fTTTJ48mdmzZ3P77bcD8OWXX7J48WIGDhzIiBEjACdve/bsyXvvvccdd9xBgwYNUsSR1TogP9GwnHwgOjqa8ePHU716dT7//PMUJxuAAgUK8PDDD/Pcc88RFxfHM88846dIsy8hIYFBgwbRrVs3kpKSaNy4cZr7nThxguXLlxMREZGikAF47LHHKFKkSIqv+Y4fP86ePXto2LBhluJat24du3bt4oYbbvCcCAGKFi3Kiy++yLlz55g0aVKWYlu/fj0Ad911l2ebMYa7774bgF9++cWz/Y8//qBy5cqpvj7s3r07AGvWrMnS67uQci41b9/Xr7/+mrJly3oKGrdevXpRo0YNvvvuO5KSktKNa8uWLcTGxhIZGekp7MHJjaeeegrAc7yc/H3YsWMHl156KY899liK7T169AB8l3OgvEvLxo0badOmDe+88w5t2rRJt72JEyfSqlUrNm/efNFhO2lZv349pUuX9hT2ACVKlKB79+4kJSWxdu1ar2MH59wJ0Lt3b8+2QoUKceeddwIpz2+TJk2iRIkSvP76657CHuC+++7jtttuo1y5cl6/Lm8ES/5583l2oaFDh3L27Fmuuuoqr47pLu4zc86ZPHkyVatW5YUXXkixfeDAgdxxxx0pvimcPHkyBQsWZMiQIZ5thQoV4vnnn8day/Tp0z3bs1sH5Ccq7vOBjz/+mMTERIYMGULJkiXT3e/RRx+lXLlyzJ8/n4MHD3p9nNatW1OxYkWOHz+e6rHx48cTFhbGl19+mWEbv//+O8uXL2fHjh1pPn706NEMn//3338zY8YMbrrpJn744YcUf3EnZ61l2LBhDBgwINVjBQoUoGDBginGA//2228A6bZ3Me5xhWmd0NwnitWrV2cptrAwZ3XVPXv2pNjXPcY0+VCKSZMmsWnTJs8wHbfY2FgAn334KedS8+Z9TUxM5PHHH+eZZ55JUaS4hYSEcPbsWc6dO5fqMbeSJUsybNiwFH/0uRUuXBjAc7yc/H0YMWIEsbGxqXJr69atgO9yDpR3aVm4cCE7d+5k2LBhfPbZZ+m29/7771OlShXmz59Pr169Mjx2WsqUKcOJEydSxe0+D5UtW9br2OGf89uff/6ZZrvJhyf+9NNPXHfddZQpUybFvsWLF2fixIn07dvX69fljWDJP28+z5JbsmQJ//vf/xg1apTX31hktrjfsmULe/bsoUOHDhQqlHIp06pVq/Luu+/StWtXAM6cOcO6deu44oorKF26dIp9mzVrRrFixVi1apVnW3brgPxExX0+sGDBAgDat2+f4X4hISG0a9eOxMREFi1a5PVxevXqxZkzZ1i4cGGqx2bPnk3JkiWJjIxM87nHjh2ja9eutGzZkm7dunHVVVfRvHlz3n77bfbu3Ut8fDxTp05NMTwhLaVLl2bBggXMnDmTSpUqpbtfqVKleOSRR7j11ltTPbZs2TLi4+OpV6+eZ5v7xHL48GG6detGjRo1qFGjBvfcc4+nMM6Ie3zf2bNnUz3mPkG7i3NvY+vduzchISEMHTqUn376iVOnThEVFcWwYcMoVapUmsUdOAXd33//zZQpU3j11Ve57LLLsvShnhblXGrevK8FChTgoYceSnOC1tatW4mNjaVGjRqeIj0tlStXZtCgQbRr1y7VY/PnzwfwHC83fx+OHDnCF198wVNPPUVoaKhPJ6Ep71Lr0KEDa9euZdCgQRQoUCDd/d566y1++OEHrr766gyPm56+ffuSmJjIgw8+yI4dOzhx4gQzZszg008/pXHjxp6hM97EDnDvvfdijOHRRx9l8+bNnDx5kq+//pq3336bypUr06VLF8ApVpOSkqhXrx5r1qyhW7duVK1alVq1avHoo4+mOSnc14Il/7z5PHM7ceIEjz/+OP/6178837p447fffiMsLIzp06fTsmVLKlWqRJMmTRgzZgxnzpzx7BcTEwM457bFixfToUMHKleuTIMGDXjhhRdSdFLs2bOH8+fPU7169VTHK1CgAJUrV2b79u2ebdmtA/ITFfd53NmzZ9m3bx+hoaGZ6iGrXbs2ALt37/b6WD169OCSSy5J1WOwdetWNm/eTOfOndOdOLdp0yaioqLo3LkzDz74IE2bNmX79u0MGzaMK664gqpVqzJkyJCLTvYsVaoU11xzjdexu506dcrzVV6fPn08292/1P/5z38oWbIkffr0oVmzZsybN4927dpddIWKiIgIAL755ptUKzx88803AGn2wmQmtoiICGbPns3p06fp2LEjl112GV26dKFAgQIsXLiQqlWrptneK6+8Qt26dXnqqacoVaoUs2bNStV7kRXKOe+k976mJSkpiWeeeYakpCTuueeeLB3v4MGDnrHzF2vD178P06dP5/LLL6dfv36cOXOGTz/9lBo1vL54YpqUd2mLiIjIVC9p27ZtMyz+L6Zfv3688cYb/PDDD1x11VVUq1aNQYMG0bp1a2bPnp2ibW9+ZyIjI/n444/Ztm0b119/PVWqVOHuu++mRo0aLFy4kFKlSgGwf/9+wBmG1KlTJ86ePcvdd99N3bp1+eSTT+jYseNFz7HZEUz5l5XPs2HDhnHkyBHGjRvn9etNSkpiy5YtxMXFMXHiRFq3bs1dd91FwYIFef3117n99ts9cbjzYNGiRdx2222EhoZy7733Ur58ed577z169uzp+cYzLi4OIN1J1iVLluTUqVOetrNbB+QnmlCbx8XFxWGtpXjx4pna313cZTTBKT2VK1emVatWLFu2jKNHj3racp+A3Kt2pCU0NJR58+bRsmVLz7bo6GimTZvG2rVrSUxMpHXr1jz55JNex5VZZ8+epW/fvmzZsoWOHTvSrVs3z2MFChSgSpUqvPvuu7Ru3dqz/fPPP6d///48+uijfP/99+m2XaVKFbp06cLcuXO57777eOmllyhfvjzffvstI0aMoFixYil6H7yJ7e+//2bkyJHs37+fDh06cPnll7Nx40aioqIYMmQI//vf/9I8eVWvXp3Bgwezfft2FixYQKdOnTyrTGSHci7zMnpfL+SeCLt8+XKaNGnCQw895PXxjh8/zu23387Bgwfp169firH43sSW1d+HsmXLMnjwYPbu3cu8efPo2bMnU6dOTXP1C28p7/xrzZo1jBs3jpCQEHr06EGpUqX4/vvvWb58OWPGjOG1117L0kpc27dvZ/To0Zw+fZpu3boRHh7OTz/9xPr163n++eeZNGkShQsX9iwv+d133/HEE08wdOhQTxsvvPAC7733Hq+++iqvvPKKz15zcsGUf95+nq1cuZKpU6cyfPjwNHvJL+bQoUPUrFmT0qVLM336dM/n2enTp+nbty+LFi1iypQp9O/f35MHixYtYty4cZ4OjMTERB544AHmzJnDlClTeOihhzxFu/ubiAu5vxk9ffo0JUqUyHYdkJ+o5z6PCwsLwxhDfHx8pvZ375fReMGMuP8qdn/tD84Jp2LFihlO0mrYsCE7d+6kZ8+eXHvttXTp0oUvv/ySXr16sXjxYpYtW8Zdd93FlClTshTXxZw8eZI777yTxYsX07Rp01TrO7/++uts3LgxxS804In3119/vejXchMmTKB169bMnz+fFi1aUL16dR555BGef/55qlWrRtGiRbMUW79+/Vi9ejWTJ09m5syZjBw5krlz5zJq1ChWr16daiKjW+/evXn55ZeZNm0an3zyCYcPH+bhhx9O93LVfZpVTPEvPcq5zLnY+5rc+fPnGThwINOnT6d69erMmDEj3Q+k9Bw6dIhbbrmFDRs20L59e0aNGpXl2LL6+xAZGcnLL7/Mhx9+6On1e/jhh9Nd7z6zOQfKO39y/9EYHx/PsmXLPIX0Dz/8wM0338zkyZNTLVmZGYmJidxxxx3s2LHDU5SNHj2aJUuW0L9/f+bPn+8p1t1zU8qVK5dqNZbnnnuO4sWLM2fOnAyP502+XSjY8i+zn2cJCQkMHjyYiIiIVIsDZFb58uX54YcfmDt3boqOqiJFini+hZw1axbwTx5ceeWVKb6ZLFCggGc1nK+++srzfCDduUtnzpzBGEOxYsUA39QB+YWK+zwuJCSE8PBwjh8/zoEDBy66v3uSW1a/Ku/SpQtFihTx9CBs3ryZrVu30q1btzQnBrr9+OOPDBw4kD///JPw8HBiYmJ46623aNeuHZdffjlXXnklLVu2JCoqKktxZcRd9Hz33XdcddVVzJo1y6sT7pVXXglc/OvV0NBQ5syZw6xZs3j55Zd54403WLt2Lf369ePgwYNpfnV+sdj27t3L8uXLufbaa1P1+g4YMIC6desyb968i65Z3b59e66//nq2bNnCzp07M/vS06Scuzhvcu7UqVP07t2bTz/9lMsvv5y5c+dSsaJ3hcfOnTtp3749GzduJDIykqlTp6aaVJ2V2NKS2d+Hxo0b06tXLw4dOuSTFXOUd/7zzTffcOTIEfr160etWrU820NCQhg7diwAn376qdftrlmzhm3bttGjR48Uw3iMMQwfPpzQ0FBPu+7hOQ0aNEg1kbJ48eLUrFmTffv2cfr0aa/jyIxgy7/Mfp698sor/PHHH7z99tvZGvaVnmrVqlG6dGnPJF93HrjPQ8lVqVKF0NBQz0UA3d94pDdc68SJExQvXjzD/0+3zJ738gsV9/mAe2mytCbfJD/RnT9/nqVLlwLw73//O0vHKlWqFO3bt2fFihXExcVl6mtCcJaemjBhAj/++COzZ8/mt99+Y9q0adx4441Yazly5AiRkZG8+eabWYorPXv27CEyMpJ169bRpk0bvvzyy1RDWM6fP8+6detSLLmWnPv/MDMX4jHG0KZNGwYPHsx9993HZZddxp49ezh8+DB169b1Ora9e/cCpHsxlLp165KUlMS+ffs4f/4833//fbpXEKxSpQqQta+JL6ScS19m3le3o0ePcsstt7B48WKuvPJKFixYwGWXXebV8TZt2kSHDh3YuXMnd9xxBx9//HG6E3Fz4vdh1apVnomGF/JlzoHyzl8yOg+VL1+eSy+9NNVqN9ltNyQkhJo1a3Lo0CFOnz7tuYhRWpM8wemdLViwoNffeHkj2PIvM59nc+fO5fz581x33XUprubqPidERER4VkRKz99//82PP/6YZg5Zazlz5oznnObOg/R64xMTEz3fKlStWpWQkJA0C/LExET27t3rmRvhyzogP9CY+3ygT58+TJ06lXHjxtG9e3fPX7bnz58nIiKCa6+9luHDhzNv3jz++usv2rdvn61J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1/PrX/+aa4vE21e/+lVOPvlkFixYwAknnNDl+3z33XdzxebBaaedxm233cby5cvZfvvtueCCC9h888158cUXuzyezTffnIMOOohf/OIXRdfvsssunH/++fz3f/83Dz/8cJe/kxdeeIEPfvCDzJs3jw033JBddtmF4447jhEjRjB06FCgiWXLlrJixUoAZrc80Wm/d+yM54ruqyfWl3gDuOy0m4v25TvvvDMHHXQQDz5YfFRqa6xMnDiRYcOGce+99/Lwww9z1llnceWVV3LUUUcxd+5cVq5c2W3c3n///XzlK18pGnOtfdzZZ59ddBpHyGLlmmuuaevjZs2axcsvv8zWW2/Nsccey2677cbIkSP50pe+VHYft+eee3Lbbbcxd+5cTj/99C5j/6GHHuLZZ5/lgx/8IA8//DARwdSpU7nkkkt48skneeqZBMCgQYNoamrq8jO3nvu4YnnEl7/85U5jbtSoUVx77bVA1sdNnDiRhx56iMMPP7ytjxszZgyf+9znyu7jLr/8co455hjmzJnDGWec0eX73NzczEsvvcSAc05v6+N23HFHzj33XAYOHAh03WePHj2afffdlxtuuKHo+tY+bvr06TzwwANd/k5a+7jm5mYGDBjA29/+do4//nhGjBjRZ/le6+dTS0sLAwaUN8Cmksn9wPwBQEQMA96Zv+z4V70Z0PtstWda8udBnazfKH9e2VkFO+64Y6dJZ0dTpkxZZ9mECRPWuePcRhttxNSpU9cpu86ye+aWtN/+Yuedd273XGjQoEHstddenHvuuUXXtSbNADvttBMnnngiKSUmTpwIwNixYxk7dizDhw/vsh6A5cuXs88++zB9+vR263fYYQeGDx/ebR277bYbZ5xxBjNmzGD69OmMGzeOa6+9lvPPP59Zs2axww47cOGFF3ZZx6abbsqSJUvaLb/rrrs499xzOeWUU9qOqbu2vOUtb+HAAw+kpaWFb33rW6SUmD59OkcffTS33HIL2y4f1m6bUaNGdR57j9VXvBU7zsLYK+V9btXS0sJRRx1FSolddtmlXd3d1fGTn/yk03iDrI/ori1NTU1FY+6jH/0os2bNYurUqd3Wsemmm3LCCSe0W94ac6eeeipTp05l00037bKOt73tbW3xdtlll7XF2+WXX84tt9zCwIEDefj+59hss1FsttkoAKZO2hLorN/rfXK/PmlqyuYdLxZ7Q4cOLTneJk2axOmnn05Kia233hqAPfbYA+g+bocNG8aqVau67OOGDRvW6z4Ouo79rvq4T3ziE23H1N3xbLTRRhx//PGd9nGPDfxnu226/MxtgD6ucFlPYq6lpYUzzjijT/q4KVOmlPQ+X3TRRZ32ca2x213MfepTn2q3vGMfN3jw4JL7uEsuuaRoH9d6TB31Kt8j+3xqbm6mpaVlnXWlqGRyPx8YHxHDU0ovA+8lO1v+j5RS2+mfiJhEdlFt8X+X+s7LwJtkw26KGV5QTjU2fPhwJk+e3G25888/n1dffbXseubNm8e0adO6LNNVHU888QTXXnst11xzDYceeiiQ/aHut99+/PGPf2zr0Lprx3bbbdf28/Llyzn++OM56qijOOKII0puy3333cejjz7KDTfcwP777w/AJptswuGHH85DDz0EeJOTrlQi5ioRb93VU82Y6028tSagKs4+zj6u2uzjGqePq+QFtTcAg4EbI+I04DKyITk/BYiIIRFxJHBjvvzXFdx3t1JKq4FFZP9YFLMd8EJKaVkn67Weue+++7juuus477zzytp+zZo1zJ8/v+i3B6X67W9/y8iRIznkkEPalg0aNIi//OUvnHjiiWXVedlll/HKK69wzjnn9Gi71q8FN9lkk7Zlm22WXWpSOPRJ5etNzFUi3mD9iTnjre/Zx7VnzPU9+7i1+nO8VTK5vwh4kGy6yUuBLcmGb12Qr58M/Ixsvvt78jLVNhtoiogdChfm03juAPy5Bm1SESkl1qxZs86j1erVqznttNP47Gc/u85XX6XWM3/+fFavXs3MmTPZddddGT16NO9+97u57777Sq7j0UcfZcKECfzmN79hr732YosttmD//fdfZ1xid8fTavHixXzve9/j9NNPZ4st1p24qat6pkyZwq677sp5553HP/7xD/7xj39wzjnnsNVWW7H33nuX+JtvXJWIuUrEW3f1VDPmjLe+Yx9nH1dt9nGN08dVbFhOSmlFfgfY48kuln0C+EE+RAfgMbKhOz8BLszPpFfb9WQ3q7ogIo5MKb2Z36H2a/n6q2rQJhXxhz/8gdGjR6+zfMGCBYwaNYqLL76YgQMHctppp/HYY4+VVc+jjz4KZH/4l19+Oa+++iqXX345hx12GLfffnvbV39d1bFkyRL+/ve/c+aZZ3L22WezxRZbcMUVV3DkkUdyzz33sO2225Z0PK2uvfZaNthgA44//viyfi+XXnopRx55JLvuuiuQXZxz0003MWTIEBxx1rVKxFwl4q27eqoZc72LN3XFPs4+rtrs4xqnj6voHWpTSq8C3+lk3WJgp0rur6dSSrdFxM+BjwD3RMQdZPPw7wf8P+DmWrZPa+29996cf/756ywfPnw48+bN44orruCmm25iww03LLJ1afXst99+/PSnP+Wggw5qq2fq1Km84x3v4Nvf/jbf/va3u61jzZo1vPDCC8yYMYN99923rfykSZP41re+xcUXX9xtHa1SSvzoRz/iqKOOare81ON5+OGHOeSQQ9h11105/fTTiQiuuOIKDj/8cGbMmEE2ak6dqUTMVSLeuqunmjHXm3h761vf2unvSfZx9nHVZx/XOH1cWcl9ROwKrEgp/b3C7amGY4B5wHHAZ4CngC8D30gppRq2SwU23XTTohervPHGG5x22mlt07CtWbOGN954o23dG2+80XYFe1f1QDal4Xve8552yzbZZBOmTJnCvHnzSqpj6NChDBkyhH322afdssmTJ7edxeiujlYPPPAAzz33HB/+8Ic7LdNVPd///vcZMmQIP//5zxk2LJsZZ7/99mPvvffmoosuYp8TKzPFX72qRMxVIt66q6eaMdebePvud7/b5b4bnX1ccfZxfcc+bl312seVO+b+QeDaSjakWlJKr6eUzkspbZ9S2jiltEP+urP577UeeeaZZ5g7dy5XXXUVo0ePZvTo0Rx00EEAfOhDH+Kwww4rua67776bX/7yl+ssb2lpabtopjvbbbddW+dX6PXXXycb8VW6mTNnMnr0aPbaa68ebdfqmWeeaZsbu9XGG2/Mbrvtxt/+9rey6lTlYq4S8QbrT8wZb33DPq5zxlzfsI8rrj/HW28uqO30NxwRyyJiRi/qlopqampi5syZ7R5XXZVdKnHJJZdwySWXlFzXnXfeySmnnNLuRhOLFy9mzpw5vPOd7+xiy7UOPPBAXnvtNW65pe0+bbz88svMmTOn6Ny3XXnggQfYc889e9x5tRo/fjzz5s1rd6O11atX8/DDD7eNUVTPVSrmKhFvsP7EnPHWN+zjOmfM9Q37uOL6c7xVdMx9gRHApn1UtxrYoEGD1vkKbYMNsjAeP358l7NKdHTcccdx9dVXc9RRR/GFL3yB1157jW984xuM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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "image/png": {
       "height": 277,
       "width": 379
      },
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "best_ckpt_path = \"{}/best_check_point\".format(result_floder)\n",
    "agent = ESP_agent(env, hyperparams_Lunarlander)\n",
    "agent.load_model(path = best_ckpt_path)\n",
    "agent.evaluate_combination_action_large_diff(example_num = 5, epsilon = 0.1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Evaluating:   0%|          | 0/100 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Using GPU\n",
      "Using GPU\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Evaluating: 100%|██████████| 100/100 [00:25<00:00,  4.00it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "avg score: 240.0433012723144\n",
      "avg GVF loss: 0.027799279659520833\n",
      "save\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "240.0433012723144"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "best_ckpt_path = \"{}/best_check_point\".format(result_floder)\n",
    "agent = ESP_agent(env, hyperparams_Lunarlander)\n",
    "agent.load_model(path = best_ckpt_path)\n",
    "agent.evaluate(0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[NbConvertApp] Converting notebook LL_ESP.ipynb to script\n",
      "[NbConvertApp] Writing 29299 bytes to LL_ESP.py\n"
     ]
    }
   ],
   "source": [
    "!jupyter nbconvert --to script LL_ESP.ipynb"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.6.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
