{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import json\n",
    "import sys\n",
    "from train_cnfqi import run\n",
    "import seaborn as sns\n",
    "import tqdm\n",
    "import matplotlib.pyplot as plt \n",
    "import numpy as np\n",
    "import torch\n",
    "import random\n",
    "import shap"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Interpretability"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\u001b[33mWARN: Box bound precision lowered by casting to float32\u001b[0m\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Fg trained after 466 epochs\n",
      "BG stayed up for steps:  [1000, 1000]\n",
      "FG stayed up for steps:  [153, 66]\n"
     ]
    }
   ],
   "source": [
    "performance_fg, performance_bg, nfq_agent, X, X_test = run(verbose=False, is_contrastive=True, evaluations=2, force_left=6)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Using a non-full backward hook when the forward contains multiple autograd Nodes is deprecated and will be removed in future versions. This hook will be missing some grad_input. Please use register_full_backward_hook to get the documented behavior.\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "array([[-0.01334314,  0.02451875, -0.07314903, -0.24963132,  0.00151554],\n",
       "       [-0.01814637,  0.02094631, -0.07932729, -0.23074524, -0.0020842 ],\n",
       "       [-0.01699225,  0.02166486, -0.076538  , -0.23600948, -0.00188427],\n",
       "       ...,\n",
       "       [-0.00600182,  0.0274763 , -0.0562601 , -0.27659899,  0.00214153],\n",
       "       [-0.01073346,  0.02637872, -0.03978335, -0.24258082,  0.00589564],\n",
       "       [-0.00611308, -0.00245893,  0.01750803,  0.02732145, -0.00860809]])"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "group_0 = []\n",
    "group_1 = []\n",
    "for x, g in zip(X_test[0], X_test[1]):\n",
    "    if g == 0:\n",
    "        group_0.append(x.cpu().detach().numpy())\n",
    "    else:\n",
    "        group_1.append(x.cpu().detach().numpy())\n",
    "group_0 = torch.Tensor(np.asarray(group_0))\n",
    "group_1 = torch.Tensor(np.asarray(group_1))\n",
    "e = shap.DeepExplainer(nfq_agent._nfq_net, X[0])\n",
    "shap_values = e.shap_values(group_1[:300])\n",
    "explanation = shap.Explanation(shap_values)\n",
    "shap_values"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "group_0[:300].shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "shap.plots.beeswarm(explanation, show=False)\n",
    "plt.title(\"Shared SHAP Values\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "#\"Cart Position\", \"Cart Velocity\", \"Pole Angle\", \"Pole Velocity at Tip\", \"Action\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "#shap_shared = np.load('cfqi_shared_shap.npy')\n",
    "#shap_fg = np.load('cfqi_fg_shap.npy')\n",
    "explanation = shap.Explanation(shap_values, data=group_0[:300])\n",
    "shap.plots.beeswarm(explanation, show=False)\n",
    "np.save('cfqi_shared_shap', shap_values)\n",
    "plt.title(\"CFQI Background SHAP Values\")\n",
    "plt.savefig('CFQI_Sim_SHAP_bg.pdf', transparent=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 576x252 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "explanation = shap.Explanation(shap_values, data=group_1[:300])\n",
    "shap.plots.beeswarm(explanation, show=False)\n",
    "np.save('cfqi_fg_shap', shap_values)\n",
    "plt.title(\"CFQI Foreground SHAP Values\")\n",
    "plt.savefig('CFQI_Sim_SHAP_fg.pdf', transparent=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Force left=0"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import json\n",
    "from train_cnfqi import run, warm_start, transfer_learning\n",
    "num_iter=10\n",
    "results = {'fqi': {}, 'cfqi': {}, 'warm_start': {}, 'tl': {}}\n",
    "for i in range(num_iter):\n",
    "    print(str(i))\n",
    "    performance = run(verbose=False, is_contrastive=True, evaluations=2, force_left=0)\n",
    "    results['cfqi'][i] = performance\n",
    "    performance = run(verbose=False, is_contrastive=False, evaluations=2, force_left=0)\n",
    "    results['fqi'][i] = performance\n",
    "    performance = warm_start(verbose=False, is_contrastive=False, evaluations=2, force_left=0)\n",
    "    results['warm_start'][i] = performance\n",
    "    performance = transfer_learning(verbose=False, is_contrastive=False, evaluations=2, force_left=0)\n",
    "    results['tl'][i] = performance\n",
    "    \n",
    "    \n",
    "    with open('force_left=0.json', 'w') as f:\n",
    "        json.dump(results, f)  "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "fqi_results = []\n",
    "cfqi_results = []\n",
    "ws_results = []\n",
    "tl_results = []\n",
    "for alg in ['cfqi', 'fqi', 'warm_start', 'tl']:\n",
    "    for key in results[alg]:\n",
    "        if alg == 'fqi':\n",
    "            fqi_results.extend(results[alg][key])\n",
    "        elif alg == 'cfqi':\n",
    "            cfqi_results.extend(results[alg][key])\n",
    "        elif alg == 'warm_start':\n",
    "            ws_results.extend(results[alg][key])\n",
    "        else:\n",
    "            tl_results.extend(results[alg][key])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "sns.distplot(fqi_results, label='FQI')\n",
    "sns.distplot(cfqi_results, label='CFQI')\n",
    "sns.distplot(ws_results, label='Warm Start')\n",
    "sns.distplot(tl_results, label='Transfer Learning')\n",
    "plt.legend()\n",
    "plt.xlabel(\"Steps survived\")\n",
    "plt.title(\"Force left = 0\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Force Left vs Performance"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import json\n",
    "from train_cnfqi import run, warm_start, transfer_learning\n",
    "import numpy as np\n",
    "import scipy.stats\n",
    "import seaborn as sns\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "num_iter=2\n",
    "results = {}\n",
    "for i in range(0, 11):\n",
    "    results[i] = {}\n",
    "    results[i]['cfqi'] = {}\n",
    "    results[i]['fqi'] = {}\n",
    "    results[i]['warm_start'] = {}\n",
    "    results[i]['tl'] = {}\n",
    "\n",
    "for i in range(num_iter):\n",
    "    for f in range(0, 11):\n",
    "        print(str(i) + str(f))\n",
    "        performance_fg, performance_bg = run(verbose=False, is_contrastive=True, evaluations=2, force_left=f)\n",
    "        results[f]['cfqi'][i] = (performance_fg, performance_bg)\n",
    "        performance_fg, performance_bg = run(verbose=False, is_contrastive=False, evaluations=2, force_left=f)\n",
    "        results[f]['fqi'][i] = (performance_fg, performance_bg)\n",
    "        performance_fg, performance_bg = warm_start(verbose=False, is_contrastive=False, evaluations=2, force_left=f)\n",
    "        results[f]['warm_start'][i] = (performance_fg, performance_bg)\n",
    "        performance_fg, performance_bg = transfer_learning(verbose=False, is_contrastive=False, evaluations=2, force_left=f)\n",
    "        results[f]['tl'][i] = (performance_fg, performance_bg)\n",
    "        \n",
    "    \n",
    "    with open('force_left_v_performance.json', 'w') as f:\n",
    "        json.dump(results, f)  "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "results[0]['cfqi'][0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "def mean_confidence_interval(data, confidence=0.95):\n",
    "    a = 1.0 * np.array(data)\n",
    "    n = len(a)\n",
    "    m, se = np.mean(a), scipy.stats.sem(a)\n",
    "    h = se * scipy.stats.t.ppf((1 + confidence) / 2., n-1)\n",
    "    return m, h\n",
    "\n",
    "def plot_performance(results, ds='bg'):\n",
    "    c_success = []\n",
    "    f_success = []\n",
    "    w_success = []\n",
    "    t_success = []\n",
    "    c_errs = []\n",
    "    f_errs = []\n",
    "    w_errs = []\n",
    "    t_errs = []\n",
    "    if ds == 'bg':\n",
    "        ind = 1\n",
    "    else:\n",
    "        ind = 0\n",
    "    for i in range(0, 11):\n",
    "        cfqi_perf = []\n",
    "        fqi_perf = []\n",
    "        ws_perf = []\n",
    "        tl_perf = []\n",
    "        for key in results[i]['fqi']:\n",
    "            fqi_perf.extend(results[i]['fqi'][key][ind])\n",
    "        for key in results[i]['cfqi']:\n",
    "            cfqi_perf.extend(results[i]['cfqi'][key][ind])\n",
    "        for key in results[i]['warm_start']:\n",
    "            ws_perf.extend(results[i]['warm_start'][key][ind])\n",
    "        for key in results[i]['tl']:\n",
    "            tl_perf.extend(results[i]['tl'][key][ind])\n",
    "\n",
    "        c_success.append(np.mean(cfqi_perf))\n",
    "        f_success.append(np.mean(fqi_perf))\n",
    "        w_success.append(np.mean(ws_perf))\n",
    "        t_success.append(np.mean(tl_perf))\n",
    "        m, h = mean_confidence_interval(cfqi_perf)\n",
    "        c_errs.append(h)\n",
    "        m, h = mean_confidence_interval(fqi_perf)\n",
    "        f_errs.append(h)\n",
    "        m, h = mean_confidence_interval(ws_perf)\n",
    "        w_errs.append(h)\n",
    "        m, h = mean_confidence_interval(tl_perf)\n",
    "        t_errs.append(h) \n",
    "\n",
    "    x = [k for k in range(0, 11)]\n",
    "    plt.figure(figsize=(10, 4))\n",
    "    sns.scatterplot(x, c_success, label='CFQI')\n",
    "    plt.errorbar(x, c_success ,yerr=c_errs, linestyle=\"None\")\n",
    "    sns.scatterplot(x, f_success, label='FQI')\n",
    "    plt.errorbar(x, f_success ,yerr=f_errs, linestyle=\"None\")\n",
    "    sns.scatterplot(x, w_success, label='Warm Start')\n",
    "    plt.errorbar(x, w_success ,yerr=w_errs, linestyle=\"None\")\n",
    "    sns.scatterplot(x, t_success, label='Transfer Learning')\n",
    "    plt.errorbar(x, t_success ,yerr=t_errs, linestyle=\"None\")\n",
    "    if ds == 'bg':\n",
    "        plt.title(\"Background Dataset: Performance of CFQI, FQI, Warm Start, Transfer Learning when force on cart is modified\")\n",
    "    else:\n",
    "        plt.title(\"Foreground Dataset: Performance of CFQI, FQI, Warm Start, Transfer Learning when force on cart is modified\")\n",
    "    plt.xlabel(\"Force Left\")\n",
    "    plt.ylabel(\"Steps Survived\")\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plot_performance(results, ds='bg')\n",
    "plot_performance(results, ds='fg')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Shuffle test"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import configargparse\n",
    "import torch\n",
    "import torch.optim as optim\n",
    "\n",
    "from environments import CartPoleRegulatorEnv\n",
    "from environments import CartEnv\n",
    "from environments import AcrobotEnv\n",
    "from models.agents import NFQAgent\n",
    "from models.networks import NFQNetwork, ContrastiveNFQNetwork\n",
    "# from simulated_fqi import NFQAgent\n",
    "# from simulated_fqi import NFQNetwork, ContrastiveNFQNetwork\n",
    "from util import get_logger, close_logger, load_models, make_reproducible, save_models\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "import itertools\n",
    "import random"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def shuffle_test(verbose=False, epoch=1000, train_env_max_steps=100, eval_env_max_steps=3000, discount=0.95, \n",
    "                 init_experience=200, increment_experience=0, hint_to_goal=0, evaluations=5, force_left=0, random_seed=1234):\n",
    "    # Setup environment\n",
    "    bg_cart_mass = 1.0\n",
    "    fg_cart_mass = 1.0\n",
    "    is_contrastive = True\n",
    "    train_env_bg = CartPoleRegulatorEnv(group=0, masscart=bg_cart_mass, mode=\"train\", force_left=force_left, is_contrastive=is_contrastive)\n",
    "    train_env_fg = CartPoleRegulatorEnv(group=1, masscart=fg_cart_mass, mode=\"train\", force_left=force_left, is_contrastive=is_contrastive)\n",
    "    eval_env_bg = CartPoleRegulatorEnv(group=0, masscart=bg_cart_mass, mode=\"eval\", force_left=force_left, is_contrastive=is_contrastive)\n",
    "    eval_env_fg = CartPoleRegulatorEnv(group=1, masscart=fg_cart_mass, mode=\"eval\", force_left=force_left, is_contrastive=is_contrastive)\n",
    "\n",
    "    # Log to File, Console, TensorBoard, W&B\n",
    "    logger = get_logger()\n",
    "\n",
    "    # Setup agent\n",
    "    nfq_net = ContrastiveNFQNetwork(state_dim=train_env_bg.state_dim, is_contrastive=is_contrastive)\n",
    "    # optimizer = optim.Rprop(nfq_net.parameters())\n",
    "\n",
    "    if is_contrastive:\n",
    "        optimizer = optim.Adam(itertools.chain(nfq_net.layers_shared.parameters(), nfq_net.layers_last_shared.parameters()), lr=1e-1)\n",
    "    else:\n",
    "        optimizer = optim.Adam(nfq_net.parameters(), lr=1e-1)\n",
    "\n",
    "    nfq_agent = NFQAgent(nfq_net, optimizer)\n",
    "\n",
    "    # NFQ Main loop\n",
    "    # A set of transition samples denoted as D\n",
    "    bg_rollouts = []\n",
    "    fg_rollouts = []\n",
    "    if init_experience > 0:\n",
    "        for _ in range(init_experience):\n",
    "            bg_group = random.choice([0, 1])\n",
    "            fg_group = random.choice([0, 1])\n",
    "            rollout_bg, episode_cost = train_env_bg.generate_rollout(\n",
    "                None, render=False, group=bg_group\n",
    "            )\n",
    "            rollout_fg, episode_cost = train_env_fg.generate_rollout(\n",
    "                None, render=False, group=fg_group\n",
    "            )\n",
    "            bg_rollouts.extend(rollout_bg)\n",
    "            fg_rollouts.extend(rollout_fg)\n",
    "    bg_rollouts.extend(fg_rollouts)\n",
    "    all_rollouts = bg_rollouts.copy()\n",
    "\n",
    "    bg_rollouts_test = []\n",
    "    fg_rollouts_test = []\n",
    "    if init_experience > 0:\n",
    "        for _ in range(init_experience):\n",
    "            bg_group = random.choice([0, 1])\n",
    "            fg_group = random.choice([0, 1])\n",
    "            rollout_bg, episode_cost = eval_env_bg.generate_rollout(\n",
    "                None, render=False, group=bg_group\n",
    "            )\n",
    "            rollout_fg, episode_cost = eval_env_fg.generate_rollout(\n",
    "                None, render=False, group=fg_group\n",
    "            )\n",
    "            bg_rollouts_test.extend(rollout_bg)\n",
    "            fg_rollouts_test.extend(rollout_fg)\n",
    "    bg_rollouts_test.extend(fg_rollouts)\n",
    "    all_rollouts_test = bg_rollouts_test.copy()\n",
    "\n",
    "    state_action_b, target_q_values, groups = nfq_agent.generate_pattern_set(all_rollouts_test)\n",
    "    X_test = state_action_b\n",
    "    test_groups = groups\n",
    "\n",
    "    bg_success_queue = [0] * 3\n",
    "    fg_success_queue = [0] * 3\n",
    "    epochs_fg = 0\n",
    "    eval_fg = 0\n",
    "    for epoch in range(epoch + 1):\n",
    "\n",
    "        state_action_b, target_q_values, groups = nfq_agent.generate_pattern_set(all_rollouts)\n",
    "        X = state_action_b\n",
    "        train_groups = groups\n",
    "\n",
    "        if not nfq_net.freeze_shared:\n",
    "            loss = nfq_agent.train((state_action_b, target_q_values, groups))\n",
    "\n",
    "        eval_episode_length_fg, eval_success_fg, eval_episode_cost_fg = 0, 0, 0\n",
    "        if nfq_net.freeze_shared:\n",
    "            eval_fg += 1\n",
    "\n",
    "            if eval_fg > 50:\n",
    "                loss = nfq_agent.train((state_action_b, target_q_values, groups))\n",
    "\n",
    "        if is_contrastive:\n",
    "            # import ipdb; ipdb.set_trace()\n",
    "            if nfq_net.freeze_shared:\n",
    "                eval_episode_length_fg, eval_success_fg, eval_episode_cost_fg = nfq_agent.evaluate(\n",
    "                    eval_env_fg, render=False\n",
    "                )\n",
    "                for param in nfq_net.layers_fg.parameters():\n",
    "                    assert param.requires_grad == True\n",
    "                for param in nfq_net.layers_last_fg.parameters():\n",
    "                    assert param.requires_grad == True\n",
    "                for param in nfq_net.layers_shared.parameters():\n",
    "                    assert param.requires_grad == False\n",
    "                for param in nfq_net.layers_last_shared.parameters():\n",
    "                    assert param.requires_grad == False\n",
    "            else:\n",
    "\n",
    "                for param in nfq_net.layers_fg.parameters():\n",
    "                    assert param.requires_grad == False\n",
    "                for param in nfq_net.layers_last_fg.parameters():\n",
    "                    assert param.requires_grad == False\n",
    "                for param in nfq_net.layers_shared.parameters():\n",
    "                    assert param.requires_grad == True\n",
    "                for param in nfq_net.layers_last_shared.parameters():\n",
    "                    assert param.requires_grad == True\n",
    "                eval_episode_length_bg, eval_success_bg, eval_episode_cost_bg = nfq_agent.evaluate(\n",
    "                    eval_env_bg, render=False\n",
    "                )\n",
    "\n",
    "\n",
    "        else:\n",
    "            eval_episode_length_bg, eval_success_bg, eval_episode_cost_bg = nfq_agent.evaluate(\n",
    "                eval_env_bg, render=False\n",
    "            )\n",
    "            eval_episode_length_fg, eval_success_fg, eval_episode_cost_fg = nfq_agent.evaluate(\n",
    "                eval_env_fg, render=False\n",
    "            )\n",
    "\n",
    "        bg_success_queue = bg_success_queue[1:]\n",
    "        bg_success_queue.append(1 if eval_success_bg else 0)\n",
    "\n",
    "        fg_success_queue = fg_success_queue[1:]\n",
    "        fg_success_queue.append(1 if eval_success_fg else 0)\n",
    "\n",
    "        printed_bg = False\n",
    "        printed_fg = False\n",
    "\n",
    "        if sum(bg_success_queue) == 3 and not nfq_net.freeze_shared == True:\n",
    "            if epochs_fg == 0:\n",
    "                epochs_fg = epoch\n",
    "            printed_bg = True\n",
    "            nfq_net.freeze_shared = True\n",
    "            if verbose:\n",
    "                print(\"FREEZING SHARED\")\n",
    "            if is_contrastive:\n",
    "                for param in nfq_net.layers_shared.parameters():\n",
    "                    param.requires_grad = False\n",
    "                for param in nfq_net.layers_last_shared.parameters():\n",
    "                    param.requires_grad = False\n",
    "                for param in nfq_net.layers_fg.parameters():\n",
    "                    param.requires_grad = True\n",
    "                for param in nfq_net.layers_last_fg.parameters():\n",
    "                    param.requires_grad = True\n",
    "            # else:\n",
    "            #     for param in nfq_net.layers_fg.parameters():\n",
    "            #         param.requires_grad = False\n",
    "            #     for param in nfq_net.layers_last_fg.parameters():\n",
    "            #         param.requires_grad = False\n",
    "\n",
    "                optimizer = optim.Adam(itertools.chain(nfq_net.layers_fg.parameters(), nfq_net.layers_last_fg.parameters()), lr=1e-1)\n",
    "                nfq_agent._optimizer = optimizer\n",
    "            # break\n",
    "\n",
    "        # Print current status\n",
    "        if verbose:\n",
    "            logger.info(\n",
    "                # \"Epoch {:4d} | Eval BG {:4d} / {:4f} | Eval FG {:4d} / {:4f} | Train Loss {:.4f}\".format(\n",
    "                #     epoch, eval_env_bg.success_step, eval_episode_cost_bg, eval_env_fg.success_step, eval_episode_cost_fg, loss\n",
    "                # )\n",
    "                \"Epoch {:4d} | Eval BG {:4d} / {:4f} | Eval FG {:4d} / {:4f} | Train Loss {:.4f}\".format(\n",
    "                    epoch, eval_episode_length_bg, eval_episode_cost_bg, eval_episode_length_fg, eval_episode_cost_fg, loss\n",
    "                )\n",
    "            )\n",
    "        if sum(fg_success_queue) == 3:\n",
    "            printed_fg = True\n",
    "            if verbose:\n",
    "                logger.info(\n",
    "                    \"Epoch {:4d} | Total Cycles {:6d} | Total Cost {:4.2f}\".format(\n",
    "                        epoch, len(all_rollouts), total_cost\n",
    "                    )\n",
    "                )\n",
    "            break\n",
    "\n",
    "    eval_env_bg.step_number = 0\n",
    "    eval_env_fg.step_number = 0\n",
    "\n",
    "    eval_env_bg.max_steps = 1000\n",
    "    eval_env_fg.max_steps = 1000\n",
    "\n",
    "    performance = []\n",
    "    num_steps_bg = []\n",
    "    num_steps_fg = []\n",
    "    total = 0\n",
    "    for it in range(evaluations):\n",
    "\n",
    "        # eval_env_bg.save_gif = True\n",
    "        eval_episode_length_bg, eval_success_bg, eval_episode_cost_bg = nfq_agent.evaluate(eval_env_bg, True)\n",
    "        # eval_env_bg.create_gif()\n",
    "        if verbose:\n",
    "            print(eval_episode_length_bg, eval_success_bg)\n",
    "        num_steps_bg.append(eval_episode_length_bg)\n",
    "        performance.append(eval_episode_length_bg)\n",
    "        total += 1\n",
    "        train_env_bg.close()\n",
    "        eval_env_bg.close()\n",
    "\n",
    "        # eval_env_fg.save_gif = True\n",
    "        eval_episode_length_fg, eval_success_fg, eval_episode_cost_fg = nfq_agent.evaluate(eval_env_fg, True)\n",
    "        # eval_env_fg.create_gif()\n",
    "        if verbose:\n",
    "            print(eval_episode_length_fg, eval_success_fg)\n",
    "        num_steps_fg.append(eval_episode_length_fg)\n",
    "        performance.append(eval_episode_length_fg)\n",
    "        total += 1\n",
    "        train_env_fg.close()\n",
    "        eval_env_fg.close()\n",
    "    \n",
    "    state_action_b, target_q_values, groups = nfq_agent.generate_pattern_set(all_rollouts)\n",
    "    X = (state_action_b, groups)\n",
    "\n",
    "\n",
    "    state_action_b, target_q_values, groups = nfq_agent.generate_pattern_set(all_rollouts_test)\n",
    "    X_test = (state_action_b, groups)\n",
    "    \n",
    "    print(\"Fg trained after \" + str(epochs_fg) + \" epochs\")\n",
    "    print(\"BG stayed up for steps: \", num_steps_bg)\n",
    "    print(\"FG stayed up for steps: \", num_steps_fg)\n",
    "    return nfq_agent, X, X_test"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "nfq_agent, X, X_test = shuffle_test()\n",
    "group_0 = []\n",
    "group_1 = []\n",
    "for x, g in zip(X_test[0], X_test[1]):\n",
    "    if g == 0:\n",
    "        group_0.append(x.cpu().detach().numpy())\n",
    "    else:\n",
    "        group_1.append(x.cpu().detach().numpy())\n",
    "group_0 = torch.Tensor(np.asarray(group_0))\n",
    "group_1 = torch.Tensor(np.asarray(group_1))\n",
    "e = shap.DeepExplainer(nfq_agent._nfq_net, X[0])\n",
    "shap_values = e.shap_values(group_1[:150])\n",
    "explanation = shap.Explanation(shap_values)\n",
    "shap_values"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "shap.plots.beeswarm(explanation, show=False)\n",
    "plt.title(\"Fg SHAP Values\")\n",
    "np.save('cfqi_shuffle_fg_shap', shap_values)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "test = np.random.normal(loc=100, scale=40, size = (1000, 5))\n",
    "explanation = shap.Explanation(test)\n",
    "shap.plots.beeswarm(explanation)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "test = np.random.normal(size = (100, 5))\n",
    "explanation = shap.Explanation(test, data=np.random.normal(size=(100, 5)))\n",
    "shap.plots.beeswarm(explanation)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "shap_shared = np.load('cfqi_shuffle_shared_shap.npy')\n",
    "shap_fg = np.load('cfqi_shuffle_fg_shap.npy')\n",
    "explanation = shap.Explanation(shap_shared)\n",
    "shap.plots.beeswarm(explanation)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "shap_shared = np.load('cfqi_shuffle_shared_shap.npy')\n",
    "shap_fg = np.load('cfqi_shuffle_fg_shap.npy')\n",
    "explanation = shap.Explanation(shap_shared)\n",
    "shap.plots.bar(explanation, show=False)\n",
    "plt.title(\"Group 0 SHAP values\")\n",
    "plt.savefig('CFQI_Shuffle_Group0.pdf', transparent=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "explanation = shap.Explanation(shap_fg)\n",
    "shap.plots.bar(explanation, show=False)\n",
    "plt.title(\"Group 1 SHAP values\")\n",
    "plt.savefig('CFQI_Shuffle_Group1.pdf', transparent=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "new_shap_values = np.concatenate([shap_fg, shap_shared])\n",
    "new_shap_values.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "cohort_g0 = [\"Background\"] * 150\n",
    "cohort_g1 = [\"Foreground\"] * 150\n",
    "cohort_g1.extend(cohort_g0)\n",
    "cohort = cohort_g1\n",
    "len(cohort)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "new_explanation = shap.Explanation(new_shap_values)\n",
    "shap.plots.bar(new_explanation.cohorts(cohort), show=False)\n",
    "plt.title(\"SHAP Values for Shuffle Test\")\n",
    "plt.savefig('CFQI_Shuffle_SHAP.pdf', transparent=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "performance = {'fg': [], 'bg': []}\n",
    "num_iter = 10\n",
    "for i in range(num_iter):\n",
    "    print(str(i))\n",
    "    bg_perf, fg_perf = shuffle_test()\n",
    "    performance['fg'].extend(fg_perf)\n",
    "    performance['bg'].extend(bg_perf)\n",
    "    with open('shuffle_test.json', 'w') as f:\n",
    "        json.dump(performance, f)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.title(\"Shuffle Test\")\n",
    "sns.distplot(performance['fg'], label='Foreground')\n",
    "sns.distplot(performance['bg'], label='Background')\n",
    "plt.legend()\n",
    "plt.xlabel(\"Steps Survived\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "research [~/.conda/envs/research/]",
   "language": "python",
   "name": "conda_research"
  },
  "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.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
