{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Getting started with Regym"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "This notebook will illustrate some basic workflow that can be achieved with the `regym` framework. This is not an exhaustive list.\n",
    "\n",
    "The workflow of a (single agent) Reinforcement Learning experiment normally goes as follows:\n",
    "1. An environment is selected\n",
    "2. An agent fit for observing and acting in the environment is generated.\n",
    "3. For a given number of episodes, the agent collects information from its interaction with the environment. These interactions are used to change the agent's policy by using a Reinforcement Learning algorithm. \n",
    "4. A trail of episode trajectories are generated by the agent, which can be used to meassure its performance."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Step 1: generating an environment"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "In order to select an environment, we only need to import the `generate_task` function inside the `regym.environments` submodule. This function takes a single argument which can be:\n",
    "\n",
    "1. The name of an OpenAI Gym environment locally installed in your system (i.e, it can be created via `gym.make(environment_name)`\n",
    "2. Relative path to a Unity executable which was compiled using Unity ML-Agents\n",
    "\n",
    "For this tutorial, let's use the iconic Cart Pole environment."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "metadata": {},
   "outputs": [],
   "source": [
    "from regym.environments import generate_task\n",
    "\n",
    "task = generate_task('CartPole-v0')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 49,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Task(name='CartPole-v0', env=<TimeLimit<CartPoleEnv<CartPole-v0>>>, state_space_size=None, action_space_size=None, observation_dim=4, observation_type='Continuous', action_dim=2, action_type='Discrete', hash_function=None)\n"
     ]
    }
   ],
   "source": [
    "print(task)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "A task is a wrapper around an OpenAI Gym compliant environment. Under the hood, `regym` parses information from the underlying environment. This information will be used to initialize the agents that will be used to act in the task's environment. Having this layer of abstraction allows us to treat OpenAI Gym and Unity environments equally. The most important elements of a task are:\n",
    "\n",
    "1. `task.env`: The environment where the agents will act, it MUST comply with the OpenAI Gym interface.\n",
    "2. `task.observation_type` and `task.action_type`: Both of which can be either `Discrete` or `Continuous`. Certain algorithms cannot be used in continuous scenarios, or they require certain changes to be made. By parsing the environment we can find out what kind of observation and action spaces we will have to work with. But all of this is done for you!"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Step 2: building an agent"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Now that we have specified what task we want to solve, we need to create an agent that is able to solve it. \n",
    "\n",
    "There are many possible agents that we can create to solve the task above. the submodule `regym.rl_algorithms` imports a set of functions, implemented by all agents in our framework, with the signature: `build_X_Agent(task, config, agent_name)`. This function takes three arguments:\n",
    "\n",
    "1. The task that needs solving. From this task, the agent will figure out the input and output size of the    corresponding layers of the neural net representing the agent's policy or value networks. Different tasks (e.g tasks with discrete / continuous action spaces) require different neural network topologies.\n",
    "2.    Config is a dictionary containing keyword - value pairs of the algorithm's hyperparameters. **Refer to source code for a list of relevant hyper parameters for each algorithm / agent**.\n",
    "3.    The agent name gives a string ID, nothing fancy.\n",
    "\n",
    "For this example we will use `build_PPO_Agent` to create an agent that gathers experience from the environment and uses Proximal Policy Optimization to change the agent's policy. Note that we could have choosen from any other of the (non-tabular) implemented algorithm, such as `build_DQN_Agent` or `build_A2C_Agent` functions. "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 50,
   "metadata": {},
   "outputs": [],
   "source": [
    "from regym.rl_algorithms import build_PPO_Agent"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 51,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Hyper parameters for our PPO agent\n",
    "def ppo_config_dict():\n",
    "     config = dict()\n",
    "     config['discount'] = 0.99\n",
    "     config['use_gae'] = True\n",
    "     config['use_cuda'] = False\n",
    "     config['gae_tau'] = 0.95\n",
    "     config['entropy_weight'] = 0.01\n",
    "     config['gradient_clip'] = 5\n",
    "     config['optimization_epochs'] = 10\n",
    "     config['mini_batch_size'] = 256\n",
    "     config['ppo_ratio_clip'] = 0.2\n",
    "     config['learning_rate'] = 3.0e-4\n",
    "     config['adam_eps'] = 1.0e-5\n",
    "     config['horizon'] = 128\n",
    "     config['phi_arch'] = 'MLP'\n",
    "     config['actor_arch'] = 'None'\n",
    "     config['critic_arch'] = 'None'\n",
    "     return config"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 52,
   "metadata": {},
   "outputs": [],
   "source": [
    "config = ppo_config_dict()\n",
    "agent = build_PPO_Agent(task, config, \"Cool-ProximalPolicyOptimization\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Step 3: training the agent"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "![rl-loop](https://proxy.duckduckgo.com/iu/?u=http%3A%2F%2Fwww.kdnuggets.com%2Fwp-content%2Fuploads%2Fiot-agent-feedback-loop.jpg&f=1)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Now we have both a `task` and an `agent` to solve it. The only step left is to let the agent train in the environment held by the task. Luckily, the submodule `regym.rl_loops` features both single agent and multiagent loops. The most basic unit of training is an episode. An `agent` can be trained for single episode in a given `task` by invoking the `run_episode(environment, agent, training=True)` function. This function will return a list containing the trajectory experienced by the agent, that is, a list of tuples`[(observation, action, reward, successive_observation, done_flag)...]`"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 53,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 3000/3000 [03:46<00:00,  7.31it/s]\n"
     ]
    }
   ],
   "source": [
    "from regym.rl_loops.singleagent_loops.rl_loop import run_episode\n",
    "from tqdm import tqdm\n",
    "\n",
    "training_episodes = 3000\n",
    "# This will take a while\n",
    "training_trajectories = [run_episode(task.env, agent, training=True) for _ in tqdm(range(training_episodes))]\n",
    "task.env.close()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Step 4: Plot learning"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "In the `CartPole-v0` environment, each step gives a reward of `1.0`, which means that the length of an episode is positively correlated to how well an agent performed in it. We can plot the length of each trajectory (y axis) against the iteration number of such trajectory (x axis)."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 54,
   "metadata": {},
   "outputs": [],
   "source": [
    "%matplotlib inline\n",
    "\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 55,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "def plot_episode_length_over_trajectories(trajectories):\n",
    "    xs = [i + 1 for i in range(len(trajectories))]\n",
    "    ys = [len(trajectory) for trajectory in trajectories]\n",
    "    xs_subsampled, ys_subsampled = xs[::50], ys[::50]\n",
    "    plt.xlabel(\"Training episode\")\n",
    "    plt.ylabel(\"Episode length\")\n",
    "    plt.suptitle(\"Evolution of episode length during training (higher is better)\")\n",
    "    plt.plot(xs_subsampled, ys_subsampled, 'r-')\n",
    "    \n",
    "plot_episode_length_over_trajectories(training_trajectories)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We can see that this (very noisy) graph shows an upward curve, meaning that the agent keeps improving it's policy, which at the beginning had only survived for ~50 steps, but as training progreses, the average episode length also increases. \n",
    "\n",
    "And we are done!\n",
    "\n",
    "In this notebook we've shown how to:\n",
    "1. Create a task fit to be used by the agents that can be created in this framework\n",
    "2. Generated an agent given a task and a dictionary of hyperparameters\n",
    "3. Trained the agent in the given task and how to process the trajectories generated by the agent in the task's environment\n",
    "\n",
    "**All of this was done with less than 10 lines of code!**"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.7.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
