{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import wandb\n",
    "import os\n",
    "import pickle\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "import pandas as pd\n",
    "\n",
    "\n",
    "def get_history(user=\"anon\", project=\"anon\", query={},\n",
    "                **kwargs):\n",
    "    api = wandb.Api()\n",
    "    runs = api.runs(path=f\"{user}/{project}\", filters=query)\n",
    "    dataframes = [run.history(**kwargs) for run in runs]\n",
    "    if len(runs) == 0:\n",
    "        return [[],[]]\n",
    "    else:\n",
    "        return list(zip(runs, dataframes))\n",
    "\n",
    "\n",
    "def download_files(user=\"anon\", project=\"anon\",\n",
    "                   query={}, save_dir=\".\", **kwargs):\n",
    "    \"\"\"\n",
    "    Download the files of each run into a new directory for the run.\n",
    "    Also saves the config dict of the run.\n",
    "    \"\"\"\n",
    "    if not os.path.isdir(save_dir):\n",
    "        os.mkdir(save_dir)\n",
    "\n",
    "    api = wandb.Api()\n",
    "    runs = api.runs(path=f\"{user}/{project}\", filters=query)\n",
    "    for run in runs:\n",
    "        name = run.name\n",
    "        config = run.config\n",
    "\n",
    "        run_dir = os.path.join(save_dir, name)\n",
    "        if not os.path.isdir(run_dir):\n",
    "            os.mkdir(run_dir)\n",
    "\n",
    "        with open(os.path.join(run_dir, \"config.pkl\"), \"wb\") as h:\n",
    "            pickle.dump(config, h)\n",
    "\n",
    "        files = run.files()\n",
    "        for file in files:\n",
    "            file.download(root=run_dir)\n",
    "    return"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "from pytorch_lightning import Trainer\n",
    "from pl_trainer import DynamicsModel, SaveTestLogCallback\n",
    "import os\n",
    "\n",
    "def load_model_from_run(run, save_dir=\"/tmp\"):\n",
    "    name = run.display_name\n",
    "    ckpt_save_path = os.path.join(save_dir, name)\n",
    "    if not os.path.exists(ckpt_save_path):\n",
    "        os.makedirs(ckpt_save_path)\n",
    "     \n",
    "    ckpts = sorted([f for f in run.files() if \"checkpoints\" in f.name])\n",
    "    if len(ckpts) == 0:\n",
    "        raise RuntimeError(f\"Run {name} has no checkpoints!\")\n",
    "    # pick latest checkpoint if available\n",
    "    last_ckpt = ckpts[-1]\n",
    "    last_ckpt.download(replace=True, root=ckpt_save_path)\n",
    "        \n",
    "    ckpt_path = os.path.join(ckpt_save_path, last_ckpt.name)\n",
    "    # Uncommet if you need the trainer\n",
    "    pl_trainer = Trainer(resume_from_checkpoint=ckpt_path,logger=False)\n",
    "    pl_model = DynamicsModel.load_from_checkpoint(ckpt_path)\n",
    "    \n",
    "    import pprint \n",
    "    pp = pprint.PrettyPrinter(indent=4)\n",
    "    print(\"--------------------------------\")\n",
    "    print(\"Model Hyperparameters:\")\n",
    "    pp.pprint(vars(pl_model.hparams))\n",
    "    print(\"--------------------------------\")\n",
    "    return pl_trainer, pl_model\n",
    "\n",
    "def load_file_from_run(run, filename, save_dir=\"/tmp\", replace=False):\n",
    "    name = run.display_name\n",
    "    save_path = os.path.join(save_dir, name)\n",
    "    if not os.path.exists(save_path):\n",
    "        os.makedirs(save_path)\n",
    "    for f in run.files():\n",
    "        if filename in f.name:\n",
    "            file_path = os.path.join(save_path, f.name)\n",
    "            if not os.path.exists(file_path):\n",
    "                f.download(replace=replace, root=save_path)\n",
    "            else:\n",
    "                print(\"File already exists\")\n",
    "            return file_path\n",
    "    raise ValueError(\"File not found in run!\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "collapsed": true,
    "jupyter": {
     "outputs_hidden": true
    }
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO:lightning:GPU available: True, used: False\n",
      "/home/anon/miniconda3/envs/ham37v2/lib/python3.7/site-packages/torch/serialization.py:657: SourceChangeWarning: source code of class 'pl_trainer.DynamicsModel' has changed. you can retrieve the original source code by accessing the object's source attribute or set `torch.nn.Module.dump_patches = True` and use the patch tool to revert the changes.\n",
      "  warnings.warn(msg, SourceChangeWarning)\n",
      "/home/anon/miniconda3/envs/ham37v2/lib/python3.7/site-packages/torch/serialization.py:657: SourceChangeWarning: source code of class 'biases.models.hnn.HNN' has changed. you can retrieve the original source code by accessing the object's source attribute or set `torch.nn.Module.dump_patches = True` and use the patch tool to revert the changes.\n",
      "  warnings.warn(msg, SourceChangeWarning)\n",
      "Unimplemented OBJ format statement 's' on line 's off'\n",
      "WARNING:pywavefront:Unimplemented OBJ format statement 's' on line 's off'\n",
      "Unimplemented OBJ format statement 's' on line 's off'\n",
      "WARNING:pywavefront:Unimplemented OBJ format statement 's' on line 's off'\n",
      "/mnt/storage1/Documents/repos/hamiltonian-biases/gyro.obj: Load time: 0.003770112991333008\n",
      "INFO:pywavefront:/mnt/storage1/Documents/repos/hamiltonian-biases/gyro.obj: Load time: 0.003770112991333008\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.0028, 0.0028, 0.0048], dtype=torch.float64) tensor([0.0024, 0.0024, 0.0004], dtype=torch.float64)\n",
      "tensor(7.0659e-07, dtype=torch.float64)\n",
      "tensor(5.7775e-07, dtype=torch.float64)\n",
      "tensor(5.3929e-07, dtype=torch.float64)\n",
      "HNN currently assumes potential energy depends only on q\n",
      "HNN currently assumes time independent Hamiltonian\n",
      "--------------------------------\n",
      "Model Hyperparameters:\n",
      "{   'angular_dims': range(0, 3),\n",
      "    'batch_size': 200,\n",
      "    'body_args': [],\n",
      "    'body_class': 'Gyroscope',\n",
      "    'callbacks': [   <pytorch_lightning.callbacks.lr_logger.LearningRateLogger object at 0x7f6cb331b690>,\n",
      "                     <pl_trainer.SaveTestLogCallback object at 0x7f6cb331b850>,\n",
      "                     <pytorch_lightning.callbacks.progress.ProgressBar object at 0x7f6cb331b7d0>],\n",
      "    'check_val_every_n_epoch': 100,\n",
      "    'chunk_len': 5,\n",
      "    'ckpt_dir': '/home/anon/repos/hamiltonian-biases/experiments/Gyroscope/HNN/wandb/run-20200525_000359-150z1xaz/anon/version_150z1xaz/checkpoints',\n",
      "    'dataset_class': 'RigidBodyDataset',\n",
      "    'debug': False,\n",
      "    'dof_ndim': 3,\n",
      "    'dt': 0.02,\n",
      "    'euclidean': False,\n",
      "    'exp_dir': '/home/anon/repos/hamiltonian-biases/experiments/Gyroscope/HNN',\n",
      "    'fast_dev_run': False,\n",
      "    'gpus': 1,\n",
      "    'hidden_size': 256,\n",
      "    'integration_time': 2,\n",
      "    'logger': <pytorch_lightning.loggers.wandb.WandbLogger object at 0x7f6cab8d9d90>,\n",
      "    'lr': 0.003,\n",
      "    'max_epochs': 2000,\n",
      "    'n_epochs': 2000,\n",
      "    'n_epochs_per_val': 100,\n",
      "    'n_gpus': 1,\n",
      "    'n_hidden': 256,\n",
      "    'n_layers': 3,\n",
      "    'n_test': 100,\n",
      "    'n_train': 800,\n",
      "    'n_train_systems': 10000,\n",
      "    'n_val': 100,\n",
      "    'network_class': 'HNN',\n",
      "    'no_lr_sched': False,\n",
      "    'num_layers': 3,\n",
      "    'optimizer_class': 'AdamW',\n",
      "    'regen': False,\n",
      "    'seed': 0,\n",
      "    'tags': ['submission'],\n",
      "    'tol': 1e-07,\n",
      "    'weight_decay': 0.0001,\n",
      "    'wgrad': True}\n",
      "--------------------------------\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO:lightning:GPU available: True, used: False\n",
      "Unimplemented OBJ format statement 's' on line 's off'\n",
      "WARNING:pywavefront:Unimplemented OBJ format statement 's' on line 's off'\n",
      "Unimplemented OBJ format statement 's' on line 's off'\n",
      "WARNING:pywavefront:Unimplemented OBJ format statement 's' on line 's off'\n",
      "/mnt/storage1/Documents/repos/hamiltonian-biases/gyro.obj: Load time: 0.0041179656982421875\n",
      "INFO:pywavefront:/mnt/storage1/Documents/repos/hamiltonian-biases/gyro.obj: Load time: 0.0041179656982421875\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.0028, 0.0028, 0.0048], dtype=torch.float64) tensor([0.0024, 0.0024, 0.0004], dtype=torch.float64)\n",
      "CH ignores angular_dims\n",
      "CH currently assumes potential energy depends only on q\n",
      "CH currently assumes time independent Hamiltonian\n",
      "CH assumes positions q are in Cartesian coordinates\n",
      "--------------------------------\n",
      "Model Hyperparameters:\n",
      "{   'angular_dims': range(0, 3),\n",
      "    'batch_size': 200,\n",
      "    'body_args': [],\n",
      "    'body_class': 'Gyroscope',\n",
      "    'callbacks': [   <pytorch_lightning.callbacks.lr_logger.LearningRateLogger object at 0x7f6ca9074cd0>,\n",
      "                     <pl_trainer.SaveTestLogCallback object at 0x7f6ca9074050>,\n",
      "                     <pytorch_lightning.callbacks.progress.ProgressBar object at 0x7f6ca9074610>],\n",
      "    'check_val_every_n_epoch': 100,\n",
      "    'chunk_len': 5,\n",
      "    'ckpt_dir': '/home/anon/repos/hamiltonian-biases/experiments/Gyroscope/CHNN/wandb/run-20200526_152410-2g4cb4uz/anon/version_2g4cb4uz/checkpoints',\n",
      "    'dataset_class': 'RigidBodyDataset',\n",
      "    'debug': False,\n",
      "    'dof_ndim': 3,\n",
      "    'dt': 0.02,\n",
      "    'euclidean': True,\n",
      "    'exp_dir': '/home/anon/repos/hamiltonian-biases/experiments/Gyroscope/CHNN',\n",
      "    'fast_dev_run': False,\n",
      "    'gpus': 1,\n",
      "    'hidden_size': 256,\n",
      "    'integration_time': 2,\n",
      "    'logger': <pytorch_lightning.loggers.wandb.WandbLogger object at 0x7f6ca8fc4650>,\n",
      "    'lr': 0.003,\n",
      "    'max_epochs': 2000,\n",
      "    'n_epochs': 2000,\n",
      "    'n_epochs_per_val': 100,\n",
      "    'n_gpus': 1,\n",
      "    'n_hidden': 256,\n",
      "    'n_layers': 3,\n",
      "    'n_test': 100,\n",
      "    'n_train': 800,\n",
      "    'n_train_systems': 10000,\n",
      "    'n_val': 100,\n",
      "    'network_class': 'CHNN',\n",
      "    'no_lr_sched': False,\n",
      "    'num_layers': 3,\n",
      "    'optimizer_class': 'AdamW',\n",
      "    'regen': False,\n",
      "    'seed': 0,\n",
      "    'tags': ['submission'],\n",
      "    'tol': 1e-07,\n",
      "    'weight_decay': 0.0001,\n",
      "    'wgrad': True}\n",
      "--------------------------------\n"
     ]
    }
   ],
   "source": [
    "# See https://docs.wandb.com/library/reference/wandb_api for how to write queries\n",
    "\n",
    "query = {\"$and\": [\n",
    "              {\"tags\": \"submission\"},\n",
    "              {\"tags\": {\"$ne\": \"data-efficiency\"}},\n",
    "              {\"state\": \"finished\"},\n",
    "              {\"config.n_train\": 800},\n",
    "              #{\"config.body_args\": [5]},\n",
    "              #{\"config.body_class\": \"ChainPendulum\"},\n",
    "              {\"config.body_class\": \"Gyroscope\"},\n",
    "              {\"config.network_class\": \"HNN\"},\n",
    "             ]}\n",
    "\n",
    "runs, histories = zip(*get_history(query=query))\n",
    "hnn_trainer, hnn_model = load_model_from_run(runs[0])\n",
    "\n",
    "query = {\"$and\": [\n",
    "              {\"tags\": \"submission\"},\n",
    "              {\"tags\": {\"$ne\": \"data-efficiency\"}},\n",
    "              {\"state\": \"finished\"},\n",
    "              {\"config.n_train\": 800},\n",
    "              #{\"config.body_args\": [5]},\n",
    "              #{\"config.body_class\": \"ChainPendulum\"},\n",
    "              {\"config.body_class\": \"Gyroscope\"},\n",
    "              {\"config.network_class\": \"CHNN\"},\n",
    "             ]}\n",
    "\n",
    "runs, histories = zip(*get_history(query=query))\n",
    "chnn_trainer, chnn_model = load_model_from_run(runs[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "sns.set(font_scale=1.25)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Unimplemented OBJ format statement 's' on line 's off'\n",
      "WARNING:pywavefront:Unimplemented OBJ format statement 's' on line 's off'\n",
      "Unimplemented OBJ format statement 's' on line 's off'\n",
      "WARNING:pywavefront:Unimplemented OBJ format statement 's' on line 's off'\n",
      "/mnt/storage1/Documents/repos/hamiltonian-biases/gyro.obj: Load time: 0.0069732666015625\n",
      "INFO:pywavefront:/mnt/storage1/Documents/repos/hamiltonian-biases/gyro.obj: Load time: 0.0069732666015625\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.0028, 0.0028, 0.0048], dtype=torch.float64) tensor([0.0024, 0.0024, 0.0004], dtype=torch.float64)\n"
     ]
    }
   ],
   "source": [
    "from biases.utils import compute_moments\n",
    "from biases.systems import Gyroscope\n",
    "import networkx as nx\n",
    "P = Gyroscope()\n",
    "out = nx.get_edge_attributes(P.body_graph,\"I\")\n",
    "I_3 = out[(0, '0_0')] + out[(0, '0_1')]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 184,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 864x288 with 4 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import torch\n",
    "import math\n",
    "\n",
    "ic_cartesian = P.sample_initial_conditions(1)\n",
    "ic_euler = P.global2bodyCoords(ic_cartesian)\n",
    "\n",
    "#n_dof = runs[0].config[\"body_args\"][0]\n",
    "num_angular_dims = 3\n",
    "n_dof = 4\n",
    "D = 3 # ambient dimension\n",
    "n_per_dim = 50\n",
    "\n",
    "q_vary = torch.linspace(0, math.pi, n_per_dim)\n",
    "qdot_vary = torch.linspace(-3, 3, n_per_dim)\n",
    "\n",
    "q_vary, qdot_vary = torch.meshgrid(q_vary, qdot_vary)\n",
    "#qdot_vary, qdot_vary_ = torch.meshgrid(qdot_vary, qdot_vary)\n",
    "\n",
    "#q_fixed = math.pi / 4 * torch.zeros_like(q_vary)\n",
    "#qdot_fixed = 10 * torch.zeros_like(qdot_vary)\n",
    "\n",
    "# Turn into N1 x N2 x ndof\n",
    "#q_lst = [q_fixed] * num_angular_dims\n",
    "#qdot_lst = [qdot_fixed] * num_angular_dims\n",
    "#qdot_lst = [qdot_vary, qdot_fixed, qdot_vary_]\n",
    "#q_lst = [q_fixed, q_vary, q_fixed]\n",
    "q_lst = [torch.ones_like(q_vary) * ic_euler[0, 0, 0], q_vary, torch.ones_like(q_vary) * ic_euler[0, 0, 2]]\n",
    "\n",
    "#qdot_lst = [qdot_fixed, qdot_vary, qdot_fixed]\n",
    "qdot_lst = [torch.ones_like(q_vary) * ic_euler[0, 1, 0], qdot_vary, torch.ones_like(q_vary) * ic_euler[0, 1, 2]]\n",
    "\n",
    "q = torch.stack(q_lst, dim=-1)\n",
    "qdot = torch.stack(qdot_lst, dim=-1)\n",
    "\n",
    "# Turn into N1*N2 x 2 x num_angular_dims\n",
    "qqdot = torch.stack([q, qdot], dim=-2).reshape(n_per_dim * n_per_dim, 2, num_angular_dims)\n",
    "\n",
    "# Turn into N1*N2 x 2 x ndof x D\n",
    "xxdot = hnn_model.body.body2globalCoords(qqdot)\n",
    "x, xdot = xxdot.chunk(2, dim=1)\n",
    "\n",
    "with torch.no_grad():\n",
    "    # Turn into N1*N2 x num_angular_dims\n",
    "    hnn_q = q.reshape(n_per_dim * n_per_dim, num_angular_dims)\n",
    "    hnn_p = hnn_model.model.M(hnn_q)(\n",
    "        qdot.reshape(n_per_dim * n_per_dim, num_angular_dims)\n",
    "    )\n",
    "    # Turn into N1*N2 x 2*num_angular_dims\n",
    "    hnn_qp = torch.cat([hnn_q, hnn_p], dim=-1)\n",
    "    hnn_energies = hnn_model.model.H(torch.tensor(0.), hnn_qp).reshape(n_per_dim, n_per_dim)\n",
    "    hnn_energies = torch.clamp(hnn_energies, min=-5, max=6)\n",
    "    \n",
    "    hnn_energies -= hnn_energies.mean()\n",
    "    hnn_energies /= hnn_energies.std()\n",
    "    \n",
    "    hnn_v_eff = hnn_energies - hnn_p[:, 0].reshape(n_per_dim, n_per_dim).pow(2).div(2).div(I_3)\n",
    "    #nn_v_eff -= hnn_v_eff.mean()\n",
    "    #nn_v_eff /= hnn_v_eff.std()\n",
    "    \n",
    "    #hnn_energies = torch.clamp(hnn_energies, min=-1000000, max=4)\n",
    "    \n",
    "    \n",
    "    true_p_euc = hnn_model.body.M @ xdot.double()\n",
    "    true_xp_euc = torch.cat([x.double(), true_p_euc], dim=1)\n",
    "    true_xp_euc = true_xp_euc.reshape(n_per_dim * n_per_dim, 2 * n_dof * D)\n",
    "    true_energies = hnn_model.body.hamiltonian(None, true_xp_euc).reshape(n_per_dim, n_per_dim)\n",
    "    \n",
    "    true_energies -= true_energies.mean()\n",
    "    true_energies /= true_energies.std()\n",
    "\n",
    "    true_v_eff = true_energies - I_3 * (qdot[:, :, 0] + qdot[:, :, 2] * torch.cos(q[:, :, 1])).pow(2).div(2)\n",
    "    #true_v_eff -= true_v_eff.mean()\n",
    "    #true_v_eff /= true_v_eff.std()\n",
    "    \n",
    "    chnn_p_euc = chnn_model.model.M(xdot)\n",
    "    # Turn into N1*N2 x 2*ndof*D\n",
    "    chnn_xp_euc = torch.cat([x, chnn_p_euc], dim=1)\n",
    "    chnn_xp_euc = chnn_xp_euc.reshape(n_per_dim * n_per_dim, 2 * n_dof * D)\n",
    "    chnn_energies = chnn_model.model.H(torch.tensor(0.), chnn_xp_euc).reshape(n_per_dim, n_per_dim)\n",
    "\n",
    "    chnn_energies -= chnn_energies.mean()\n",
    "    chnn_energies /= chnn_energies.std()\n",
    "    \n",
    "    chnn_v_eff = chnn_energies - I_3 * (qdot[:, :, 0] + qdot[:, :, 2] * torch.cos(q[:, :, 1])).pow(2).div(2)\n",
    "    #chnn_v_eff -= chnn_v_eff.mean()\n",
    "    #chnn_v_eff /= chnn_v_eff.std()\n",
    "        \n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "fig, axes = plt.subplots(ncols=3, figsize=(3 * 4, 4))\n",
    "\n",
    "#axes[0].contourf(q_vary, qdot_vary, hnn_energies, cmap=\"inferno\", levels=20)\n",
    "axes[0].contourf(q_vary, qdot_vary, hnn_v_eff, cmap=\"inferno\", levels=20)\n",
    "axes[0].set(title=\"HNN (Learned)\")\n",
    "axes[0].set(xlabel=r\"$q_2$\")\n",
    "axes[0].set(ylabel=r\"$dq_2/dt$\")\n",
    "\n",
    "#axes[1].contourf(q_vary, qdot_vary, true_energies, cmap=\"inferno\", levels=20)\n",
    "axes[1].contourf(q_vary, qdot_vary, true_v_eff, cmap=\"inferno\", levels=20)\n",
    "axes[1].set(title=\"True\")\n",
    "axes[1].set(yticklabels=[])\n",
    "axes[1].set(xlabel=r\"$q_2$\")\n",
    "\n",
    "#heatmap = axes[2].contourf(q_vary, qdot_vary, chnn_energies, cmap=\"inferno\", levels=20)\n",
    "heatmap = axes[2].contourf(q_vary, qdot_vary, chnn_v_eff, cmap=\"inferno\", levels=20)\n",
    "axes[2].set(title=\"CHNN (Learned)\")\n",
    "axes[2].set(yticklabels=[])\n",
    "axes[2].set(xlabel=r\"$q_2$\")\n",
    "\n",
    "cbar = fig.colorbar(heatmap, ax=axes.ravel().tolist(), shrink=0.8)\n",
    "#fig.suptitle(\"ChainPendulum(\"+ f\"{n_dof}\" + \") Hamiltonians \" \n",
    "#             + r\"($q_i = \\pi, dq_i/dt = 5$ for $i = 1,\\ldots, 4$)\"\n",
    "#             , fontsize=16)\n",
    "fig.subplots_adjust(top=0.85, right=0.774, wspace=0.05)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 185,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots()\n",
    "\n",
    "ax.contourf(q_vary, qdot_vary, hnn_v_eff, cmap=\"inferno\", levels=20)\n",
    "ax.set(xlabel=r\"$\\theta$\")\n",
    "ax.set(ylabel=r\"$d\\theta/dt$\")\n",
    "fig.savefig(\"gyroscope-hnn-veff.pdf\", bbox_inches=\"tight\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 186,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots()\n",
    "\n",
    "ax.contourf(q_vary, qdot_vary, true_v_eff, cmap=\"inferno\", levels=20)\n",
    "ax.set(xlabel=r\"$\\theta$\")\n",
    "ax.set(ylabel=r\"$d\\theta/dt$\")\n",
    "fig.savefig(\"gyroscope-true-veff.pdf\", bbox_inches=\"tight\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 187,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots()\n",
    "\n",
    "ax.contourf(q_vary, qdot_vary, chnn_v_eff, cmap=\"inferno\", levels=20)\n",
    "ax.set(xlabel=r\"$\\theta$\")\n",
    "ax.set(ylabel=r\"$d\\theta/dt$\")\n",
    "fig.savefig(\"gyroscope-chnn-veff.pdf\", bbox_inches=\"tight\")"
   ]
  },
  {
   "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.7.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
