{ "cells": [ { "cell_type": "markdown", "id": "33ee711d", "metadata": { "papermill": { "duration": 0.009299, "end_time": "2021-07-26T21:27:21.119445", "exception": false, "start_time": "2021-07-26T21:27:21.110146", "status": "completed" }, "tags": [] }, "source": [ "\n", "# How to train a Deep Q Network\n", "\n", "* **Author:** PL team\n", "* **License:** CC BY-SA\n", "* **Generated:** 2021-07-26T23:14:45.695289\n", "\n", "Main takeaways:\n", "\n", "1. RL has the same flow as previous models we have seen, with a few additions\n", "2. Handle unsupervised learning by using an IterableDataset where the dataset itself is constantly updated during training\n", "3. Each training step carries has the agent taking an action in the environment and storing the experience in the IterableDataset\n", "\n", "\n", "---\n", "Open in [![Open In Colab](){height=\"20px\" width=\"117px\"}](https://colab.research.google.com/github/PytorchLightning/lightning-tutorials/blob/publication/.notebooks/lightning_examples/reinforce-learning-DQN.ipynb)\n", "\n", "Give us a ⭐ [on Github](https://www.github.com/PytorchLightning/pytorch-lightning/)\n", "| Check out [the documentation](https://pytorch-lightning.readthedocs.io/en/latest/)\n", "| Join us [on Slack](https://join.slack.com/t/pytorch-lightning/shared_invite/zt-pw5v393p-qRaDgEk24~EjiZNBpSQFgQ)" ] }, { "cell_type": "markdown", "id": "f9c378db", "metadata": { "papermill": { "duration": 0.007663, "end_time": "2021-07-26T21:27:21.135216", "exception": false, "start_time": "2021-07-26T21:27:21.127553", "status": "completed" }, "tags": [] }, "source": [ "### Setup\n", "This notebook requires some packages besides pytorch-lightning." ] }, { "cell_type": "code", "execution_count": 1, "id": "621d9c7d", "metadata": { "colab": {}, "colab_type": "code", "execution": { "iopub.execute_input": "2021-07-26T21:27:21.157525Z", "iopub.status.busy": "2021-07-26T21:27:21.157055Z", "iopub.status.idle": "2021-07-26T21:27:23.530688Z", "shell.execute_reply": "2021-07-26T21:27:23.530184Z" }, "id": "LfrJLKPFyhsK", "lines_to_next_cell": 0, "papermill": { "duration": 2.387983, "end_time": "2021-07-26T21:27:23.530821", "exception": false, "start_time": "2021-07-26T21:27:21.142838", "status": "completed" }, "tags": [] }, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "\u001b[33mWARNING: Value for scheme.platlib does not match. Please report this to \r\n", "distutils: /usr/local/lib/python3.9/dist-packages\r\n", "sysconfig: /usr/lib/python3.9/site-packages\u001b[0m\r\n", "\u001b[33mWARNING: Value for scheme.purelib does not match. Please report this to \r\n", "distutils: /usr/local/lib/python3.9/dist-packages\r\n", "sysconfig: /usr/lib/python3.9/site-packages\u001b[0m\r\n", "\u001b[33mWARNING: Value for scheme.headers does not match. Please report this to \r\n", "distutils: /usr/local/include/python3.9/UNKNOWN\r\n", "sysconfig: /usr/include/python3.9/UNKNOWN\u001b[0m\r\n", "\u001b[33mWARNING: Value for scheme.scripts does not match. Please report this to \r\n", "distutils: /usr/local/bin\r\n", "sysconfig: /usr/bin\u001b[0m\r\n", "\u001b[33mWARNING: Value for scheme.data does not match. Please report this to \r\n", "distutils: /usr/local\r\n", "sysconfig: /usr\u001b[0m\r\n", "\u001b[33mWARNING: Additional context:\r\n", "user = False\r\n", "home = None\r\n", "root = None\r\n", "prefix = None\u001b[0m\r\n" ] } ], "source": [ "! pip install --quiet \"gym\" \"pytorch-lightning>=1.3\" \"torch>=1.6, <1.9\" \"torchmetrics>=0.3\"" ] }, { "cell_type": "code", "execution_count": 2, "id": "eac80594", "metadata": { "execution": { "iopub.execute_input": "2021-07-26T21:27:23.552854Z", "iopub.status.busy": "2021-07-26T21:27:23.552379Z", "iopub.status.idle": "2021-07-26T21:27:24.701512Z", "shell.execute_reply": "2021-07-26T21:27:24.701070Z" }, "papermill": { "duration": 1.161732, "end_time": "2021-07-26T21:27:24.701630", "exception": false, "start_time": "2021-07-26T21:27:23.539898", "status": "completed" }, "tags": [] }, "outputs": [], "source": [ "import os\n", "from collections import deque, namedtuple, OrderedDict\n", "from typing import List, Tuple\n", "\n", "import gym\n", "import numpy as np\n", "import torch\n", "from pytorch_lightning import LightningModule, Trainer\n", "from torch import nn, Tensor\n", "from torch.optim import Adam, Optimizer\n", "from torch.utils.data import DataLoader\n", "from torch.utils.data.dataset import IterableDataset\n", "\n", "PATH_DATASETS = os.environ.get('PATH_DATASETS', '.')\n", "AVAIL_GPUS = min(1, torch.cuda.device_count())" ] }, { "cell_type": "code", "execution_count": 3, "id": "0bfffc03", "metadata": { "execution": { "iopub.execute_input": "2021-07-26T21:27:24.724549Z", "iopub.status.busy": "2021-07-26T21:27:24.724078Z", "iopub.status.idle": "2021-07-26T21:27:24.726193Z", "shell.execute_reply": "2021-07-26T21:27:24.725793Z" }, "papermill": { "duration": 0.01604, "end_time": "2021-07-26T21:27:24.726293", "exception": false, "start_time": "2021-07-26T21:27:24.710253", "status": "completed" }, "tags": [] }, "outputs": [], "source": [ "class DQN(nn.Module):\n", " \"\"\"\n", " Simple MLP network\n", " \"\"\"\n", "\n", " def __init__(self, obs_size: int, n_actions: int, hidden_size: int = 128):\n", " \"\"\"\n", " Args:\n", " obs_size: observation/state size of the environment\n", " n_actions: number of discrete actions available in the environment\n", " hidden_size: size of hidden layers\n", " \"\"\"\n", " super().__init__()\n", " self.net = nn.Sequential(\n", " nn.Linear(obs_size, hidden_size),\n", " nn.ReLU(),\n", " nn.Linear(hidden_size, n_actions),\n", " )\n", "\n", " def forward(self, x):\n", " return self.net(x.float())" ] }, { "cell_type": "markdown", "id": "0f0e84d9", "metadata": { "papermill": { "duration": 0.008295, "end_time": "2021-07-26T21:27:24.742720", "exception": false, "start_time": "2021-07-26T21:27:24.734425", "status": "completed" }, "tags": [] }, "source": [ "### Memory" ] }, { "cell_type": "code", "execution_count": 4, "id": "aeab7bc1", "metadata": { "execution": { "iopub.execute_input": "2021-07-26T21:27:24.762210Z", "iopub.status.busy": "2021-07-26T21:27:24.761745Z", "iopub.status.idle": "2021-07-26T21:27:24.763836Z", "shell.execute_reply": "2021-07-26T21:27:24.763445Z" }, "papermill": { "duration": 0.013053, "end_time": "2021-07-26T21:27:24.763933", "exception": false, "start_time": "2021-07-26T21:27:24.750880", "status": "completed" }, "tags": [] }, "outputs": [], "source": [ "\n", "# Named tuple for storing experience steps gathered in training\n", "Experience = namedtuple(\n", " 'Experience',\n", " field_names=['state', 'action', 'reward', 'done', 'new_state'],\n", ")" ] }, { "cell_type": "code", "execution_count": 5, "id": "c8a2fcb5", "metadata": { "execution": { "iopub.execute_input": "2021-07-26T21:27:24.786200Z", "iopub.status.busy": "2021-07-26T21:27:24.785725Z", "iopub.status.idle": "2021-07-26T21:27:24.787827Z", "shell.execute_reply": "2021-07-26T21:27:24.787369Z" }, "papermill": { "duration": 0.01577, "end_time": "2021-07-26T21:27:24.787925", "exception": false, "start_time": "2021-07-26T21:27:24.772155", "status": "completed" }, "tags": [] }, "outputs": [], "source": [ "class ReplayBuffer:\n", " \"\"\"\n", " Replay Buffer for storing past experiences allowing the agent to learn from them\n", "\n", " Args:\n", " capacity: size of the buffer\n", " \"\"\"\n", "\n", " def __init__(self, capacity: int) -> None:\n", " self.buffer = deque(maxlen=capacity)\n", "\n", " def __len__(self) -> None:\n", " return len(self.buffer)\n", "\n", " def append(self, experience: Experience) -> None:\n", " \"\"\"\n", " Add experience to the buffer\n", "\n", " Args:\n", " experience: tuple (state, action, reward, done, new_state)\n", " \"\"\"\n", " self.buffer.append(experience)\n", "\n", " def sample(self, batch_size: int) -> Tuple:\n", " indices = np.random.choice(len(self.buffer), batch_size, replace=False)\n", " states, actions, rewards, dones, next_states = zip(*(self.buffer[idx] for idx in indices))\n", "\n", " return (\n", " np.array(states),\n", " np.array(actions),\n", " np.array(rewards, dtype=np.float32),\n", " np.array(dones, dtype=np.bool),\n", " np.array(next_states),\n", " )" ] }, { "cell_type": "code", "execution_count": 6, "id": "ce6fab64", "metadata": { "execution": { "iopub.execute_input": "2021-07-26T21:27:24.809140Z", "iopub.status.busy": "2021-07-26T21:27:24.808674Z", "iopub.status.idle": "2021-07-26T21:27:24.810365Z", "shell.execute_reply": "2021-07-26T21:27:24.810736Z" }, "lines_to_next_cell": 2, "papermill": { "duration": 0.014597, "end_time": "2021-07-26T21:27:24.810849", "exception": false, "start_time": "2021-07-26T21:27:24.796252", "status": "completed" }, "tags": [] }, "outputs": [], "source": [ "class RLDataset(IterableDataset):\n", " \"\"\"\n", " Iterable Dataset containing the ExperienceBuffer\n", " which will be updated with new experiences during training\n", "\n", " Args:\n", " buffer: replay buffer\n", " sample_size: number of experiences to sample at a time\n", " \"\"\"\n", "\n", " def __init__(self, buffer: ReplayBuffer, sample_size: int = 200) -> None:\n", " self.buffer = buffer\n", " self.sample_size = sample_size\n", "\n", " def __iter__(self) -> Tuple:\n", " states, actions, rewards, dones, new_states = self.buffer.sample(self.sample_size)\n", " for i in range(len(dones)):\n", " yield states[i], actions[i], rewards[i], dones[i], new_states[i]" ] }, { "cell_type": "markdown", "id": "5bf5f70e", "metadata": { "lines_to_next_cell": 2, "papermill": { "duration": 0.008338, "end_time": "2021-07-26T21:27:24.827809", "exception": false, "start_time": "2021-07-26T21:27:24.819471", "status": "completed" }, "tags": [] }, "source": [ "### Agent" ] }, { "cell_type": "code", "execution_count": 7, "id": "497f6af6", "metadata": { "execution": { "iopub.execute_input": "2021-07-26T21:27:24.853026Z", "iopub.status.busy": "2021-07-26T21:27:24.852318Z", "iopub.status.idle": "2021-07-26T21:27:24.854718Z", "shell.execute_reply": "2021-07-26T21:27:24.855093Z" }, "lines_to_next_cell": 2, "papermill": { "duration": 0.019026, "end_time": "2021-07-26T21:27:24.855209", "exception": false, "start_time": "2021-07-26T21:27:24.836183", "status": "completed" }, "tags": [] }, "outputs": [], "source": [ "class Agent:\n", " \"\"\"\n", " Base Agent class handeling the interaction with the environment\n", " \"\"\"\n", "\n", " def __init__(self, env: gym.Env, replay_buffer: ReplayBuffer) -> None:\n", " \"\"\"\n", " Args:\n", " env: training environment\n", " replay_buffer: replay buffer storing experiences\n", " \"\"\"\n", " self.env = env\n", " self.replay_buffer = replay_buffer\n", " self.reset()\n", " self.state = self.env.reset()\n", "\n", " def reset(self) -> None:\n", " \"\"\" Resents the environment and updates the state\"\"\"\n", " self.state = self.env.reset()\n", "\n", " def get_action(self, net: nn.Module, epsilon: float, device: str) -> int:\n", " \"\"\"Using the given network, decide what action to carry out\n", " using an epsilon-greedy policy\n", "\n", " Args:\n", " net: DQN network\n", " epsilon: value to determine likelihood of taking a random action\n", " device: current device\n", "\n", " Returns:\n", " action\n", " \"\"\"\n", " if np.random.random() < epsilon:\n", " action = self.env.action_space.sample()\n", " else:\n", " state = torch.tensor([self.state])\n", "\n", " if device not in ['cpu']:\n", " state = state.cuda(device)\n", "\n", " q_values = net(state)\n", " _, action = torch.max(q_values, dim=1)\n", " action = int(action.item())\n", "\n", " return action\n", "\n", " @torch.no_grad()\n", " def play_step(\n", " self,\n", " net: nn.Module,\n", " epsilon: float = 0.0,\n", " device: str = 'cpu',\n", " ) -> Tuple[float, bool]:\n", " \"\"\"Carries out a single interaction step between the agent and the environment\n", "\n", " Args:\n", " net: DQN network\n", " epsilon: value to determine likelihood of taking a random action\n", " device: current device\n", "\n", " Returns:\n", " reward, done\n", " \"\"\"\n", "\n", " action = self.get_action(net, epsilon, device)\n", "\n", " # do step in the environment\n", " new_state, reward, done, _ = self.env.step(action)\n", "\n", " exp = Experience(self.state, action, reward, done, new_state)\n", "\n", " self.replay_buffer.append(exp)\n", "\n", " self.state = new_state\n", " if done:\n", " self.reset()\n", " return reward, done" ] }, { "cell_type": "markdown", "id": "a0c997f1", "metadata": { "lines_to_next_cell": 2, "papermill": { "duration": 0.008391, "end_time": "2021-07-26T21:27:24.872244", "exception": false, "start_time": "2021-07-26T21:27:24.863853", "status": "completed" }, "tags": [] }, "source": [ "### DQN Lightning Module" ] }, { "cell_type": "code", "execution_count": 8, "id": "6a4d50ff", "metadata": { "execution": { "iopub.execute_input": "2021-07-26T21:27:24.905411Z", "iopub.status.busy": "2021-07-26T21:27:24.904908Z", "iopub.status.idle": "2021-07-26T21:27:24.906984Z", "shell.execute_reply": "2021-07-26T21:27:24.906522Z" }, "papermill": { "duration": 0.02641, "end_time": "2021-07-26T21:27:24.907082", "exception": false, "start_time": "2021-07-26T21:27:24.880672", "status": "completed" }, "tags": [] }, "outputs": [], "source": [ "class DQNLightning(LightningModule):\n", " \"\"\" Basic DQN Model \"\"\"\n", "\n", " def __init__(\n", " self,\n", " batch_size: int = 16,\n", " lr: float = 1e-2,\n", " env: str = \"CartPole-v0\",\n", " gamma: float = 0.99,\n", " sync_rate: int = 10,\n", " replay_size: int = 1000,\n", " warm_start_size: int = 1000,\n", " eps_last_frame: int = 1000,\n", " eps_start: float = 1.0,\n", " eps_end: float = 0.01,\n", " episode_length: int = 200,\n", " warm_start_steps: int = 1000,\n", " ) -> None:\n", " \"\"\"\n", " Args:\n", " batch_size: size of the batches\")\n", " lr: learning rate\n", " env: gym environment tag\n", " gamma: discount factor\n", " sync_rate: how many frames do we update the target network\n", " replay_size: capacity of the replay buffer\n", " warm_start_size: how many samples do we use to fill our buffer at the start of training\n", " eps_last_frame: what frame should epsilon stop decaying\n", " eps_start: starting value of epsilon\n", " eps_end: final value of epsilon\n", " episode_length: max length of an episode\n", " warm_start_steps: max episode reward in the environment\n", " \"\"\"\n", " super().__init__()\n", " self.save_hyperparameters()\n", "\n", " self.env = gym.make(self.hparams.env)\n", " obs_size = self.env.observation_space.shape[0]\n", " n_actions = self.env.action_space.n\n", "\n", " self.net = DQN(obs_size, n_actions)\n", " self.target_net = DQN(obs_size, n_actions)\n", "\n", " self.buffer = ReplayBuffer(self.hparams.replay_size)\n", " self.agent = Agent(self.env, self.buffer)\n", " self.total_reward = 0\n", " self.episode_reward = 0\n", " self.populate(self.hparams.warm_start_steps)\n", "\n", " def populate(self, steps: int = 1000) -> None:\n", " \"\"\"\n", " Carries out several random steps through the environment to initially fill\n", " up the replay buffer with experiences\n", "\n", " Args:\n", " steps: number of random steps to populate the buffer with\n", " \"\"\"\n", " for i in range(steps):\n", " self.agent.play_step(self.net, epsilon=1.0)\n", "\n", " def forward(self, x: Tensor) -> Tensor:\n", " \"\"\"\n", " Passes in a state x through the network and gets the q_values of each action as an output\n", "\n", " Args:\n", " x: environment state\n", "\n", " Returns:\n", " q values\n", " \"\"\"\n", " output = self.net(x)\n", " return output\n", "\n", " def dqn_mse_loss(self, batch: Tuple[Tensor, Tensor]) -> Tensor:\n", " \"\"\"\n", " Calculates the mse loss using a mini batch from the replay buffer\n", "\n", " Args:\n", " batch: current mini batch of replay data\n", "\n", " Returns:\n", " loss\n", " \"\"\"\n", " states, actions, rewards, dones, next_states = batch\n", "\n", " state_action_values = self.net(states).gather(1, actions.unsqueeze(-1)).squeeze(-1)\n", "\n", " with torch.no_grad():\n", " next_state_values = self.target_net(next_states).max(1)[0]\n", " next_state_values[dones] = 0.0\n", " next_state_values = next_state_values.detach()\n", "\n", " expected_state_action_values = next_state_values * self.hparams.gamma + rewards\n", "\n", " return nn.MSELoss()(state_action_values, expected_state_action_values)\n", "\n", " def training_step(self, batch: Tuple[Tensor, Tensor], nb_batch) -> OrderedDict:\n", " \"\"\"\n", " Carries out a single step through the environment to update the replay buffer.\n", " Then calculates loss based on the minibatch recieved\n", "\n", " Args:\n", " batch: current mini batch of replay data\n", " nb_batch: batch number\n", "\n", " Returns:\n", " Training loss and log metrics\n", " \"\"\"\n", " device = self.get_device(batch)\n", " epsilon = max(\n", " self.hparams.eps_end,\n", " self.hparams.eps_start - self.global_step + 1 / self.hparams.eps_last_frame,\n", " )\n", "\n", " # step through environment with agent\n", " reward, done = self.agent.play_step(self.net, epsilon, device)\n", " self.episode_reward += reward\n", "\n", " # calculates training loss\n", " loss = self.dqn_mse_loss(batch)\n", "\n", " if self.trainer.use_dp or self.trainer.use_ddp2:\n", " loss = loss.unsqueeze(0)\n", "\n", " if done:\n", " self.total_reward = self.episode_reward\n", " self.episode_reward = 0\n", "\n", " # Soft update of target network\n", " if self.global_step % self.hparams.sync_rate == 0:\n", " self.target_net.load_state_dict(self.net.state_dict())\n", "\n", " log = {\n", " 'total_reward': torch.tensor(self.total_reward).to(device),\n", " 'reward': torch.tensor(reward).to(device),\n", " 'train_loss': loss\n", " }\n", " status = {\n", " 'steps': torch.tensor(self.global_step).to(device),\n", " 'total_reward': torch.tensor(self.total_reward).to(device)\n", " }\n", "\n", " return OrderedDict({'loss': loss, 'log': log, 'progress_bar': status})\n", "\n", " def configure_optimizers(self) -> List[Optimizer]:\n", " \"\"\" Initialize Adam optimizer\"\"\"\n", " optimizer = Adam(self.net.parameters(), lr=self.hparams.lr)\n", " return [optimizer]\n", "\n", " def __dataloader(self) -> DataLoader:\n", " \"\"\"Initialize the Replay Buffer dataset used for retrieving experiences\"\"\"\n", " dataset = RLDataset(self.buffer, self.hparams.episode_length)\n", " dataloader = DataLoader(\n", " dataset=dataset,\n", " batch_size=self.hparams.batch_size,\n", " )\n", " return dataloader\n", "\n", " def train_dataloader(self) -> DataLoader:\n", " \"\"\"Get train loader\"\"\"\n", " return self.__dataloader()\n", "\n", " def get_device(self, batch) -> str:\n", " \"\"\"Retrieve device currently being used by minibatch\"\"\"\n", " return batch[0].device.index if self.on_gpu else 'cpu'" ] }, { "cell_type": "markdown", "id": "5d09b815", "metadata": { "papermill": { "duration": 0.008511, "end_time": "2021-07-26T21:27:24.924163", "exception": false, "start_time": "2021-07-26T21:27:24.915652", "status": "completed" }, "tags": [] }, "source": [ "### Trainer" ] }, { "cell_type": "code", "execution_count": 9, "id": "77233e6e", "metadata": { "execution": { "iopub.execute_input": "2021-07-26T21:27:24.944963Z", "iopub.status.busy": "2021-07-26T21:27:24.944501Z", "iopub.status.idle": "2021-07-26T21:27:40.167092Z", "shell.execute_reply": "2021-07-26T21:27:40.166617Z" }, "papermill": { "duration": 15.234387, "end_time": "2021-07-26T21:27:40.167206", "exception": false, "start_time": "2021-07-26T21:27:24.932819", "status": "completed" }, "tags": [] }, "outputs": [ { "name": "stderr", "output_type": "stream", "text": [ "GPU available: True, used: True\n" ] }, { "name": "stderr", "output_type": "stream", "text": [ "TPU available: False, using: 0 TPU cores\n" ] }, { "name": "stderr", "output_type": "stream", "text": [ "LOCAL_RANK: 0 - CUDA_VISIBLE_DEVICES: [0,1]\n" ] }, { "name": "stderr", "output_type": "stream", "text": [ "\n", " | Name | Type | Params\n", "------------------------------------\n", "0 | net | DQN | 898 \n", "1 | target_net | DQN | 898 \n", "------------------------------------\n", "1.8 K Trainable params\n", "0 Non-trainable params\n", "1.8 K Total params\n", "0.007 Total estimated model params size (MB)\n" ] }, { "name": "stderr", "output_type": "stream", "text": [ "/home/AzDevOps_azpcontainer/.local/lib/python3.9/site-packages/pytorch_lightning/trainer/data_loading.py:102: UserWarning: The dataloader, train dataloader, does not have many workers which may be a bottleneck. Consider increasing the value of the `num_workers` argument` (try 12 which is the number of cpus on this machine) in the `DataLoader` init to improve performance.\n", " rank_zero_warn(\n" ] }, { "data": { "application/vnd.jupyter.widget-view+json": { "model_id": "292eb170164e439ab3c69c33cde369f8", "version_major": 2, "version_minor": 0 }, "text/plain": [ "Training: 0it [00:00, ?it/s]" ] }, "metadata": {}, "output_type": "display_data" }, { "name": "stderr", "output_type": "stream", "text": [ "/tmp/ipykernel_728/3502201211.py:32: DeprecationWarning: `np.bool` is a deprecated alias for the builtin `bool`. To silence this warning, use `bool` by itself. Doing this will not modify any behavior and is safe. If you specifically wanted the numpy scalar type, use `np.bool_` here.\n", "Deprecated in NumPy 1.20; for more details and guidance: https://numpy.org/devdocs/release/1.20.0-notes.html#deprecations\n", " np.array(dones, dtype=np.bool),\n", "/home/AzDevOps_azpcontainer/.local/lib/python3.9/site-packages/pytorch_lightning/trainer/deprecated_api.py:70: LightningDeprecationWarning: Internal: `use_dp` is deprecated in v1.2 and will be removed in v1.4.\n", " rank_zero_deprecation(\"Internal: `use_dp` is deprecated in v1.2 and will be removed in v1.4.\")\n", "/home/AzDevOps_azpcontainer/.local/lib/python3.9/site-packages/pytorch_lightning/trainer/deprecated_api.py:92: LightningDeprecationWarning: Internal: `use_ddp2` is deprecated in v1.2 and will be removed in v1.4.\n", " rank_zero_deprecation(\"Internal: `use_ddp2` is deprecated in v1.2 and will be removed in v1.4.\")\n" ] } ], "source": [ "\n", "model = DQNLightning()\n", "\n", "trainer = Trainer(\n", " gpus=AVAIL_GPUS,\n", " max_epochs=200,\n", " val_check_interval=100,\n", ")\n", "\n", "trainer.fit(model)" ] }, { "cell_type": "code", "execution_count": 10, "id": "2d1234a3", "metadata": { "execution": { "iopub.execute_input": "2021-07-26T21:27:40.192086Z", "iopub.status.busy": "2021-07-26T21:27:40.191606Z", "iopub.status.idle": "2021-07-26T21:27:41.734141Z", "shell.execute_reply": "2021-07-26T21:27:41.733727Z" }, "papermill": { "duration": 1.556234, "end_time": "2021-07-26T21:27:41.734258", "exception": false, "start_time": "2021-07-26T21:27:40.178024", "status": "completed" }, "tags": [] }, "outputs": [ { "data": { "text/html": [ "\n", " \n", " \n", " " ], "text/plain": [ "" ] }, "metadata": {}, "output_type": "display_data" } ], "source": [ "# Start tensorboard.\n", "%load_ext tensorboard\n", "%tensorboard --logdir lightning_logs/" ] }, { "cell_type": "markdown", "id": "de74debc", "metadata": { "papermill": { "duration": 0.010636, "end_time": "2021-07-26T21:27:41.756088", "exception": false, "start_time": "2021-07-26T21:27:41.745452", "status": "completed" }, "tags": [] }, "source": [ "## Congratulations - Time to Join the Community!\n", "\n", "Congratulations on completing this notebook tutorial! If you enjoyed this and would like to join the Lightning\n", "movement, you can do so in the following ways!\n", "\n", "### Star [Lightning](https://github.com/PyTorchLightning/pytorch-lightning) on GitHub\n", "The easiest way to help our community is just by starring the GitHub repos! This helps raise awareness of the cool\n", "tools we're building.\n", "\n", "### Join our [Slack](https://join.slack.com/t/pytorch-lightning/shared_invite/zt-pw5v393p-qRaDgEk24~EjiZNBpSQFgQ)!\n", "The best way to keep up to date on the latest advancements is to join our community! Make sure to introduce yourself\n", "and share your interests in `#general` channel\n", "\n", "\n", "### Contributions !\n", "The best way to contribute to our community is to become a code contributor! At any time you can go to\n", "[Lightning](https://github.com/PyTorchLightning/pytorch-lightning) or [Bolt](https://github.com/PyTorchLightning/lightning-bolts)\n", "GitHub Issues page and filter for \"good first issue\".\n", "\n", "* [Lightning good first issue](https://github.com/PyTorchLightning/pytorch-lightning/issues?q=is%3Aopen+is%3Aissue+label%3A%22good+first+issue%22)\n", "* [Bolt good first issue](https://github.com/PyTorchLightning/lightning-bolts/issues?q=is%3Aopen+is%3Aissue+label%3A%22good+first+issue%22)\n", "* You can also contribute your own notebooks with useful examples !\n", "\n", "### Great thanks from the entire Pytorch Lightning Team for your interest !\n", "\n", "![Pytorch Lightning](){height=\"60px\" width=\"240px\"}" ] } ], "metadata": { "jupytext": { "cell_metadata_filter": "id,colab,colab_type,-all", "formats": "ipynb,py:percent", "main_language": "python" }, "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.9.6" }, "papermill": { "default_parameters": {}, "duration": 22.150892, "end_time": "2021-07-26T21:27:42.273682", "environment_variables": {}, "exception": null, "input_path": "lightning_examples/reinforce-learning-DQN/dqn.ipynb", "output_path": ".notebooks/lightning_examples/reinforce-learning-DQN.ipynb", "parameters": {}, "start_time": "2021-07-26T21:27:20.122790", "version": "2.3.3" }, "widgets": { "application/vnd.jupyter.widget-state+json": { "state": { "1c6b56b096d041c0919c5b92c5d97c34": { "model_module": "@jupyter-widgets/controls", "model_module_version": "1.5.0", "model_name": "HTMLModel", "state": { "_dom_classes": [], "_model_module": "@jupyter-widgets/controls", "_model_module_version": "1.5.0", "_model_name": "HTMLModel", "_view_count": null, "_view_module": "@jupyter-widgets/controls", "_view_module_version": "1.5.0", "_view_name": "HTMLView", "description": "", "description_tooltip": null, "layout": "IPY_MODEL_7a65c0674f01458586dd48ac6ab27869", "placeholder": "​", "style": "IPY_MODEL_3c7b01fc55e3423b895c4d2536db3ed1", "value": " 13/? [00:00<00:00, 196.93it/s, loss=95.8, v_num=2]" } }, "292eb170164e439ab3c69c33cde369f8": { "model_module": "@jupyter-widgets/controls", "model_module_version": "1.5.0", "model_name": "HBoxModel", "state": { "_dom_classes": [], "_model_module": "@jupyter-widgets/controls", "_model_module_version": "1.5.0", "_model_name": "HBoxModel", "_view_count": null, "_view_module": "@jupyter-widgets/controls", "_view_module_version": "1.5.0", "_view_name": "HBoxView", "box_style": "", "children": [ "IPY_MODEL_ad6c1a4b38634d258a51585b70198b79", "IPY_MODEL_a7aaf59959e642c0a4ae1f7135a7b0fd", "IPY_MODEL_1c6b56b096d041c0919c5b92c5d97c34" ], "layout": "IPY_MODEL_ef6e47263c85484b8b0c4ec4a8bb578a" } }, "2f7877ab98ce43e4b597d36b0a04a87e": { "model_module": "@jupyter-widgets/base", "model_module_version": "1.2.0", "model_name": "LayoutModel", "state": { "_model_module": "@jupyter-widgets/base", "_model_module_version": "1.2.0", "_model_name": "LayoutModel", "_view_count": null, "_view_module": "@jupyter-widgets/base", "_view_module_version": "1.2.0", "_view_name": "LayoutView", "align_content": null, "align_items": null, "align_self": null, "border": null, "bottom": null, "display": null, "flex": "2", "flex_flow": null, "grid_area": null, "grid_auto_columns": null, "grid_auto_flow": null, "grid_auto_rows": null, "grid_column": null, "grid_gap": null, "grid_row": null, "grid_template_areas": null, "grid_template_columns": null, "grid_template_rows": null, "height": null, "justify_content": null, "justify_items": null, "left": null, "margin": null, "max_height": null, "max_width": null, "min_height": null, "min_width": null, "object_fit": null, "object_position": null, "order": null, "overflow": null, "overflow_x": null, "overflow_y": null, "padding": null, "right": null, "top": null, "visibility": null, "width": null } }, "3c7b01fc55e3423b895c4d2536db3ed1": { "model_module": "@jupyter-widgets/controls", "model_module_version": "1.5.0", "model_name": "DescriptionStyleModel", "state": { "_model_module": "@jupyter-widgets/controls", "_model_module_version": "1.5.0", "_model_name": "DescriptionStyleModel", "_view_count": null, "_view_module": "@jupyter-widgets/base", "_view_module_version": "1.2.0", "_view_name": "StyleView", "description_width": "" } }, "7a65c0674f01458586dd48ac6ab27869": { "model_module": "@jupyter-widgets/base", "model_module_version": "1.2.0", "model_name": "LayoutModel", "state": { "_model_module": "@jupyter-widgets/base", "_model_module_version": "1.2.0", "_model_name": "LayoutModel", "_view_count": null, "_view_module": "@jupyter-widgets/base", "_view_module_version": "1.2.0", "_view_name": "LayoutView", "align_content": null, "align_items": null, "align_self": null, "border": null, "bottom": null, "display": null, "flex": null, "flex_flow": null, "grid_area": null, "grid_auto_columns": null, "grid_auto_flow": null, "grid_auto_rows": null, "grid_column": null, "grid_gap": null, "grid_row": null, "grid_template_areas": null, "grid_template_columns": null, "grid_template_rows": null, "height": null, "justify_content": null, "justify_items": null, "left": null, "margin": null, "max_height": null, "max_width": null, "min_height": null, "min_width": null, "object_fit": null, "object_position": null, "order": null, "overflow": null, "overflow_x": null, "overflow_y": null, "padding": null, "right": null, "top": null, "visibility": null, "width": null } }, "a7aaf59959e642c0a4ae1f7135a7b0fd": { "model_module": "@jupyter-widgets/controls", "model_module_version": "1.5.0", "model_name": "FloatProgressModel", "state": { "_dom_classes": [], "_model_module": "@jupyter-widgets/controls", "_model_module_version": "1.5.0", "_model_name": "FloatProgressModel", "_view_count": null, "_view_module": "@jupyter-widgets/controls", "_view_module_version": "1.5.0", "_view_name": "ProgressView", "bar_style": "success", "description": "", "description_tooltip": null, "layout": "IPY_MODEL_2f7877ab98ce43e4b597d36b0a04a87e", "max": 1.0, "min": 0.0, "orientation": "horizontal", "style": "IPY_MODEL_ba7905dc03934fb9ba1e0f987957ffff", "value": 1.0 } }, "ad6c1a4b38634d258a51585b70198b79": { "model_module": "@jupyter-widgets/controls", "model_module_version": "1.5.0", "model_name": "HTMLModel", "state": { "_dom_classes": [], "_model_module": "@jupyter-widgets/controls", "_model_module_version": "1.5.0", "_model_name": "HTMLModel", "_view_count": null, "_view_module": "@jupyter-widgets/controls", "_view_module_version": "1.5.0", "_view_name": "HTMLView", "description": "", "description_tooltip": null, "layout": "IPY_MODEL_df04e4b2b00644d0a5f086876becc8ae", "placeholder": "​", "style": "IPY_MODEL_ada489fa15ca472a9286a5baa6783400", "value": "Epoch 199: " } }, "ada489fa15ca472a9286a5baa6783400": { "model_module": "@jupyter-widgets/controls", "model_module_version": "1.5.0", "model_name": "DescriptionStyleModel", "state": { "_model_module": "@jupyter-widgets/controls", "_model_module_version": "1.5.0", "_model_name": "DescriptionStyleModel", "_view_count": null, "_view_module": "@jupyter-widgets/base", "_view_module_version": "1.2.0", "_view_name": "StyleView", "description_width": "" } }, "ba7905dc03934fb9ba1e0f987957ffff": { "model_module": "@jupyter-widgets/controls", "model_module_version": "1.5.0", "model_name": "ProgressStyleModel", "state": { "_model_module": "@jupyter-widgets/controls", "_model_module_version": "1.5.0", "_model_name": "ProgressStyleModel", "_view_count": null, "_view_module": "@jupyter-widgets/base", "_view_module_version": "1.2.0", "_view_name": "StyleView", "bar_color": null, "description_width": "" } }, "df04e4b2b00644d0a5f086876becc8ae": { "model_module": "@jupyter-widgets/base", "model_module_version": "1.2.0", "model_name": "LayoutModel", "state": { "_model_module": "@jupyter-widgets/base", "_model_module_version": "1.2.0", "_model_name": "LayoutModel", "_view_count": null, "_view_module": "@jupyter-widgets/base", "_view_module_version": "1.2.0", "_view_name": "LayoutView", "align_content": null, "align_items": null, "align_self": null, "border": null, "bottom": null, "display": null, "flex": null, "flex_flow": null, "grid_area": null, "grid_auto_columns": null, "grid_auto_flow": null, "grid_auto_rows": null, "grid_column": null, "grid_gap": null, "grid_row": null, "grid_template_areas": null, "grid_template_columns": null, "grid_template_rows": null, "height": null, "justify_content": null, "justify_items": null, "left": null, "margin": null, "max_height": null, "max_width": null, "min_height": null, "min_width": null, "object_fit": null, "object_position": null, "order": null, "overflow": null, "overflow_x": null, "overflow_y": null, "padding": null, "right": null, "top": null, "visibility": null, "width": null } }, "ef6e47263c85484b8b0c4ec4a8bb578a": { "model_module": "@jupyter-widgets/base", "model_module_version": "1.2.0", "model_name": "LayoutModel", "state": { "_model_module": "@jupyter-widgets/base", "_model_module_version": "1.2.0", "_model_name": "LayoutModel", "_view_count": null, "_view_module": "@jupyter-widgets/base", "_view_module_version": "1.2.0", "_view_name": "LayoutView", "align_content": null, "align_items": null, "align_self": null, "border": null, "bottom": null, "display": "inline-flex", "flex": null, "flex_flow": "row wrap", "grid_area": null, "grid_auto_columns": null, "grid_auto_flow": null, "grid_auto_rows": null, "grid_column": null, "grid_gap": null, "grid_row": null, "grid_template_areas": null, "grid_template_columns": null, "grid_template_rows": null, "height": null, "justify_content": null, "justify_items": null, "left": null, "margin": null, "max_height": null, "max_width": null, "min_height": null, "min_width": null, "object_fit": null, "object_position": null, "order": null, "overflow": null, "overflow_x": null, "overflow_y": null, "padding": null, "right": null, "top": null, "visibility": null, "width": "100%" } } }, "version_major": 2, "version_minor": 0 } } }, "nbformat": 4, "nbformat_minor": 5 }