{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "import pickle, os, csv, math, time, joblib\n",
    "from joblib import Parallel, delayed\n",
    "import datetime as dt\n",
    "from datetime import date, datetime, timedelta\n",
    "from collections import Counter\n",
    "import copy as cp\n",
    "import tqdm\n",
    "from sklearn.ensemble import ExtraTreesRegressor, ExtraTreesClassifier\n",
    "from lightgbm import LGBMRegressor, LGBMClassifier\n",
    "from sklearn.metrics import mean_absolute_error, mean_squared_error\n",
    "from sklearn.metrics import log_loss, f1_score, precision_score, recall_score, accuracy_score\n",
    "#import matplotlib.pyplot as plt\n",
    "#import matplotlib.ticker as ticker\n",
    "import collections \n",
    "#import shap\n",
    "import seaborn as sns\n",
    "import random\n",
    "from sklearn.linear_modeål import LinearRegression\n",
    "np.seterr(all=\"ignore\")\n",
    "import matplotlib.pyplot as plt\n",
    "import tqdm\n",
    "import math\n",
    "import statsmodels.api as sm\n",
    "import pandas as pd\n",
    "import statsmodels.formula.api as smf\n",
    "import numpy as np"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Simulation\n",
    "* Simulate RL data from two different distributions, generate transition tuples"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "code_folding": []
   },
   "outputs": [],
   "source": [
    "# Generate transition matrices, separate distributions for each one\n",
    "# We have to ensure that these transitions keep the next state calculations within some reasonable range\n",
    "# Make sure that states aren't exploding\n",
    "shape, scale = 2, 10\n",
    "transition_foreground = np.random.gamma(shape, scale, (12, 10))\n",
    "\n",
    "mu, sigma = 0, 4 # mean and standard deviation\n",
    "transition_background = np.random.normal(mu, sigma, (12, 10))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "code_folding": []
   },
   "outputs": [],
   "source": [
    "# Generate reward function\n",
    "mu, sigma = 0, 5\n",
    "reward_function = np.random.normal(mu, sigma, (12, 1))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {
    "code_folding": []
   },
   "outputs": [],
   "source": [
    "# Params\n",
    "exploit = 0.6\n",
    "explore = 1-exploit\n",
    "num_samples = 100\n",
    "num_patients = 100\n",
    "actions = [[0, 0], [0, 1], [1, 0], [1, 1]]\n",
    "#actions = [[0, 0], [0, 2], [3, 1], [4, 4]]\n",
    "mu, sigma = 0, 4"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {
    "code_folding": []
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [00:01<00:00, 75.12it/s]\n"
     ]
    }
   ],
   "source": [
    "transition_tuples = []\n",
    "for k, pat in enumerate(tqdm.tqdm(range(num_patients))):\n",
    "    \n",
    "    flip = np.random.choice(2)\n",
    "    if flip == 0:\n",
    "        ds = 'foreground'\n",
    "    else:\n",
    "        ds = 'background'\n",
    "    # Generate a random initial state\n",
    "    s = np.random.normal(mu, sigma, (10, 1))\n",
    "    \n",
    "    # Generate all of the tuples for this patient\n",
    "    for i in range(num_samples):\n",
    "        flip = random.uniform(0, 1)\n",
    "        # Exploit\n",
    "        if flip < exploit:            \n",
    "            all_rewards = []\n",
    "            for j, a in enumerate(actions):\n",
    "                a = np.asarray(a)\n",
    "                a = np.reshape(a, (2, 1))\n",
    "                s_a = np.concatenate((s, a))\n",
    "                reward = np.dot(reward_function.T, s_a)\n",
    "                all_rewards.append(reward)\n",
    "\n",
    "            noise = np.random.normal(0, 0.05, 1)\n",
    "            all_rewards = np.asarray(all_rewards)\n",
    "            a = actions[np.argmax(all_rewards)]\n",
    "            reward = np.max(all_rewards) + noise\n",
    "            \n",
    "            if ds == 'foreground':\n",
    "                t_m = transition_foreground\n",
    "            else:\n",
    "                t_m = transition_background\n",
    "            ns = np.matmul(s_a.T, t_m) / np.linalg.norm(np.matmul(s_a.T, t_m), ord=2)\n",
    "            ns = np.add(ns, np.random.normal(0, 0.5, (1, 10))) # Add noise\n",
    "            \n",
    "        \n",
    "        # Explore\n",
    "        else:\n",
    "            a = np.asarray(actions[np.random.choice(3)])\n",
    "            a = np.reshape(a, (2, 1))\n",
    "            s_a = np.concatenate((s, a)) # concatenate the state and action\n",
    "\n",
    "            if ds == 'foreground':\n",
    "                t_m = transition_foreground\n",
    "            else:\n",
    "                t_m = transition_background\n",
    "            ns = np.matmul(s_a.T, t_m) / np.linalg.norm(np.matmul(s_a.T, t_m), ord=2)\n",
    "            ns = np.add(ns, np.random.normal(0, 0.5, (1, 10))) # Add noise\n",
    "            \n",
    "            reward = np.dot(reward_function.T, s_a) + np.random.normal(0, 0.5, 1)\n",
    "\n",
    "        # Transition tuple includes state, action, next state, reward, ds\n",
    "        transition_tuples.append((list(s.flatten()), list(a), list(ns.flatten()), list(reward.flatten())[0], ds, i))\n",
    "        s = ns.T"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "split = int(0.8*len(transition_tuples))\n",
    "train_tuples = transition_tuples[:split]\n",
    "test_tuples = transition_tuples[split:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {
    "code_folding": []
   },
   "outputs": [],
   "source": [
    "import json\n",
    "def construct_mixedlm_ds(tuples, fname):\n",
    "    tup_dict = {}\n",
    "    elts = ['a0', 'a1', 'r', 'ds']\n",
    "    for i in range(10):\n",
    "        elts.append('s' +str(i))\n",
    "    for elt in elts:\n",
    "        tup_dict[elt] = []\n",
    "\n",
    "    for j, tup in enumerate(tqdm.tqdm(tuples)):\n",
    "        state = tup[0]\n",
    "        for i in range(len(state)):\n",
    "            tup_dict['s' + str(i)].append(state[i])\n",
    "\n",
    "        try:\n",
    "            a = tup[1]\n",
    "            a = np.concatenate(a).ravel()\n",
    "            a = list(a)\n",
    "            tup_dict['a0'].append(int(a[0]))\n",
    "            tup_dict['a1'].append(int(a[1]))\n",
    "        except:\n",
    "            a = tup[1]\n",
    "            tup_dict['a0'].append(int(a[0]))\n",
    "            tup_dict['a1'].append(int(a[1]))\n",
    "\n",
    "        tup_dict['r'].append(float(tup[3]))\n",
    "        tup_dict['ds'].append(int(tup[4] == 'foreground'))\n",
    "    \n",
    "    with open(fname, 'w') as fp:\n",
    "        json.dump(tup_dict, fp)"
   ]
  },
  {
   "cell_type": "raw",
   "metadata": {},
   "source": [
    "construct_mixedlm_ds(test_tuples, 'test_tuples.json')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<AxesSubplot:>"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "states = [a for (a, b, c, d, e, f) in transition_tuples]\n",
    "states = np.squeeze(states)\n",
    "sns.heatmap(states)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<AxesSubplot:>"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "ns = [c for (a, b, c, d, e, f) in transition_tuples]\n",
    "ns = np.squeeze(ns)\n",
    "sns.heatmap(ns)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/Users/aishwaryamandyam/anaconda3/envs/research/lib/python3.7/site-packages/seaborn/distributions.py:2557: FutureWarning: `distplot` is a deprecated function and will be removed in a future version. Please adapt your code to use either `displot` (a figure-level function with similar flexibility) or `histplot` (an axes-level function for histograms).\n",
      "  warnings.warn(msg, FutureWarning)\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<AxesSubplot:ylabel='Density'>"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "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": [
    "r = [d for (a, b, c, d, e, f) in transition_tuples]\n",
    "sns.distplot(r)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "code_folding": []
   },
   "outputs": [],
   "source": [
    "def construct_dicts(train_tuples, test_tuples):\n",
    "    train = {}\n",
    "    test = {}\n",
    "    elts = ['s', 'a', 'ns', 'r', 'ds', 'vnum']\n",
    "    for elt in elts:\n",
    "        train[elt] = []\n",
    "        test[elt] = []\n",
    "\n",
    "    for tup in train_tuples:\n",
    "        train['s'].append(tup[0])\n",
    "        a = tup[1]\n",
    "        try:\n",
    "            a = np.concatenate(a).ravel()\n",
    "            a = list(a)\n",
    "            train['a'].append(a)\n",
    "        except:\n",
    "            train['a'].append(a)\n",
    "        train['ns'].append(tup[2].flatten())\n",
    "        train['r'].append(tup[3])\n",
    "        train['ds'].append(tup[4])\n",
    "        train['vnum'].append(tup[5])\n",
    "\n",
    "    for tup in test_tuples:\n",
    "        test['s'].append(tup[0].flatten())\n",
    "        try:\n",
    "            a = tup[1]\n",
    "            a = np.concatenate(a).ravel()\n",
    "            a = list(a)\n",
    "            test['a'].append(a)\n",
    "        except:\n",
    "            test['a'].append(tup[1])\n",
    "        test['ns'].append(tup[2].flatten())\n",
    "        test['r'].append(tup[3])\n",
    "        test['ds'].append(tup[4])\n",
    "        test['vnum'].append(tup[5])\n",
    "    return train, test"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# FQI"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## FQI on both datasets"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "code_folding": [
     0
    ]
   },
   "outputs": [],
   "source": [
    "# Define action space, the potential classes of action items. \n",
    "def a2c(action):\n",
    "    #actions = [[0, 0], [0, 1], [1, 0], [1, 1]]\n",
    "    actions = [[0, 0], [0, 2], [3, 1], [4, 4]]\n",
    "    classes = []\n",
    "    for a in action:\n",
    "        a = list(a)\n",
    "        for c in range(len(actions)):\n",
    "            if actions[c] == a:\n",
    "                classes.append(c) \n",
    "    return classes\n",
    "\n",
    "def p2c(pred):\n",
    "    if pred <= 0.25:\n",
    "        action = [0, 0]\n",
    "    elif pred <= 0.5:\n",
    "        action = [0, 1]\n",
    "    elif pred <= 0.75:\n",
    "        action = [1, 0]\n",
    "    else:\n",
    "        action = [1, 1]\n",
    "        \n",
    "# Mapping states to actions?        \n",
    "def c2a(c):\n",
    "    d = {0: [0, 0], 1: [0, 1], 2: [1, 0], 3: [1, 1]}\n",
    "    return np.array([d[k] for k in c])\n",
    "\n",
    "def random_weights(size=5):\n",
    "    \n",
    "    #w = 2*np.random.uniform(size=size) - 1\n",
    "    w = norm(np.random.uniform(size=size))\n",
    "    #w / np.sum(np.abs(w))\n",
    "    \n",
    "    return w\n",
    "\n",
    "def norm(vec):\n",
    "    return vec/np.sum(np.abs(vec))\n",
    "\n",
    "def learnBehaviour(training_set, test_set):  \n",
    "    floc = \"simulated_fqi/behavior.pkl\"\n",
    "    #if os.path.exists(floc):\n",
    "    #    behaviour_pi = pickle.load(open(floc, 'rb'))\n",
    "    #else:\n",
    "    # Use a linear regression to predict behavior\n",
    "    behaviour_pi = LinearRegression()\n",
    "    X = np.vstack((training_set['s'], test_set['s']))\n",
    "    X = np.reshape(X, (-1, 10))\n",
    "    y = a2c(np.vstack((training_set['a'], test_set['a'])))\n",
    "    behaviour_pi.fit(X,y)\n",
    "    pickle.dump(behaviour_pi, open(floc, 'wb'))    \n",
    "    \n",
    "    return behaviour_pi"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "code_folding": [
     0
    ]
   },
   "outputs": [],
   "source": [
    "class FQIagent():\n",
    "    def __init__(self, train_tuples, test_tuples, iters=150, gamma=0.99, batch_size=100, prioritize=False, estimator='lin',\n",
    "                 weights=np.array([1, 1, 1, 1, 1])/5., maxT=36):\n",
    "        \n",
    "        self.iters = iters\n",
    "        self.gamma = gamma\n",
    "        self.batch_size = batch_size\n",
    "        self.prioritize_a = prioritize\n",
    "        self.training_set, self.test_set = construct_dicts(train_tuples, test_tuples)\n",
    "        self.raw_test = test_tuples\n",
    "        \n",
    "        self.visits = {'train': len(train_tuples), 'test': len(test_tuples)}\n",
    "        self.NV = {'train': len(train_tuples), 'test': len(test_tuples)}\n",
    "        self.n_samples = len(self.training_set['s'])\n",
    "        _, self.unique_actions, self.action_counts, _ = self.sub_actions()\n",
    "        self.state_feats = [str(x) for x in range(10)]\n",
    "        self.n_features = len(self.state_feats)\n",
    "        self.reward_weights = weights\n",
    "        self.maxT = maxT\n",
    "        self.piB = learnBehaviour(self.training_set, self.test_set)\n",
    "        self.n_actions = 4\n",
    "        \n",
    "        if estimator == 'tree':\n",
    "            self.q_est = ExtraTreesRegressor(n_estimators=50, max_depth=None, min_samples_leaf=10, min_samples_split=2,\n",
    "                                             random_state=0)\n",
    "        elif estimator == 'gbm':\n",
    "            self.q_est = LGBMRegressor(n_estimators=50, silent=True)\n",
    "\n",
    "        elif estimator == 'nn':\n",
    "            self.q_est = None\n",
    "        \n",
    "        elif estimator == 'lin':\n",
    "            self.q_est = LinearRegression()\n",
    "            \n",
    "        self.piE = LinearRegression()#LGBMClassifier(n_estimators=50, silent=True)\n",
    "        \n",
    "        self.eval_est = LGBMRegressor(n_estimators=50, silent=True)\n",
    "\n",
    "    def sub_actions(self):\n",
    "        \n",
    "        a = self.training_set['a']\n",
    "        a = list(a)\n",
    "        \n",
    "        unique_actions = 0\n",
    "        action_counts = 0\n",
    "        n_actions = 0\n",
    "        \n",
    "        unique_actions, action_counts = np.unique(a, axis=0, return_counts=True)\n",
    "        n_actions = len(unique_actions)\n",
    "                \n",
    "        return a, unique_actions, action_counts, n_actions\n",
    "    \n",
    "    def sampleTuples(self):\n",
    "        \n",
    "        # Get a batch of unprioritized samples:\n",
    "        \n",
    "        ids = list(np.random.choice(np.arange(self.n_samples), self.batch_size, replace=False))\n",
    "        batch = {}\n",
    "        for k in self.training_set.keys():\n",
    "            batch[k] = np.asarray(self.training_set[k], dtype=object)[ids]\n",
    "        batch['r'] = np.dot(batch['r'] * [1, 1, 10, 10, 100], self.reward_weights)\n",
    "        batch['s_ids'] = np.asarray(ids, dtype=int)\n",
    "        batch['ns_ids'] = np.asarray(ids, dtype=int) + 1\n",
    "            \n",
    "    \n",
    "        return batch\n",
    "    \n",
    "    def fitQ(self, batch, Q):\n",
    "        \n",
    "        # input = [state action]\n",
    "        x =  np.hstack((batch['s'], batch['a']))\n",
    "        \n",
    "        # target = r + gamma * max_a(Q(s', a))      == r for first iteration\n",
    "        y = batch['r'] + (self.gamma * np.max(Q[batch['ns_ids'], :], axis=1))\n",
    "        \n",
    "        self.q_est.fit(x, y)   \n",
    "    \n",
    "    def updateQtable(self, Qtable, batch):\n",
    "        \n",
    "        for i, a in enumerate(self.unique_actions):\n",
    "            #print(a, i)\n",
    "            Qtable[batch['s_ids'], i] = self.q_est.predict(np.hstack((batch['ns'], np.tile(a, (self.batch_size,1)))))\n",
    "        return Qtable\n",
    "    \n",
    "    def runFQI(self, repeats=10):\n",
    "        \n",
    "        print('Learning policy')\n",
    "        meanQtable = np.zeros((self.n_samples + 1, self.n_actions))\n",
    "        \n",
    "        for r in range(repeats):\n",
    "            print('Run', r, ':')\n",
    "            print('Initialize: get batch, set initial Q')\n",
    "            Qtable = np.zeros((self.n_samples + 1, self.n_actions))\n",
    "            Qdist = []\n",
    "\n",
    "            #print('Run FQI')\n",
    "            for iteration in range(self.iters):\n",
    "\n",
    "                # copy q-table\n",
    "                Qold = cp.deepcopy(Qtable)\n",
    "\n",
    "                # sample batch  \n",
    "                batch = self.sampleTuples()\n",
    "\n",
    "                # learn q_est with samples, targets from batch\n",
    "                self.fitQ(batch, Qtable)\n",
    "\n",
    "                # update Q table for all s given new estimator\n",
    "                self.updateQtable(Qtable, batch)\n",
    "\n",
    "                # check divergence from last estimate\n",
    "                Qdist.append(mean_absolute_error(Qold, Qtable))\n",
    "         \n",
    "            #plt.plot(Qdist)\n",
    "            meanQtable += Qtable\n",
    "        \n",
    "        meanQtable = meanQtable / repeats\n",
    "        print('Learn policy')\n",
    "        self.getPi(meanQtable)\n",
    "        return Qdist\n",
    "                    \n",
    "    \n",
    "    def getPi(self, Qtable):\n",
    "        optA = np.argmax(Qtable, axis=1)\n",
    "        print(\"Opta: \", optA)\n",
    "        #print(\"Fitting to training set\")\n",
    "        #print(\"Optimal actions: \", optA)\n",
    "        self.piE.fit(self.training_set['s'], optA[:-1])\n",
    "        #print(\"Done Fitting\")\n",
    "    \n",
    "    def testPi(self, behavior):\n",
    "        accurate = 0\n",
    "        total = 0\n",
    "        \n",
    "        for tup in self.raw_test:\n",
    "            s = tup[0]\n",
    "            try:\n",
    "                a = tup[1]\n",
    "                a = np.concatenate(a).ravel()\n",
    "                a = list(a)\n",
    "            except:\n",
    "                a = tup[1]\n",
    "            # actions based on policy we learn\n",
    "            s = s.reshape((1, 10))\n",
    "            evalA = self.piE.predict(s)\n",
    "            \n",
    "            # predicted actions based on historical actions model\n",
    "            behavB = behavior.predict(s)\n",
    "            \n",
    "            if behavB <= 0.25:\n",
    "                behavB = 0\n",
    "            elif behavB <= 0.5:\n",
    "                behavB = 1\n",
    "            elif behavB <= 0.75:\n",
    "                behavB = 2\n",
    "            else:\n",
    "                behavB = 3\n",
    "            \n",
    "            # actual historical actions\n",
    "            actions = [[0, 0], [0, 1], [1, 0], [1, 1]]\n",
    "            behavA = actions.index(a)\n",
    "            \n",
    "            if behavA == behavB:\n",
    "                accurate += 1\n",
    "            total += 1\n",
    "        \n",
    "        return float(accurate)/total"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "fqi_agent = FQIagent(train_tuples=train_tuples, test_tuples=test_tuples)\n",
    "Q_dist = fqi_agent.runFQI(repeats=1)\n",
    "plt.plot(Q_dist, label= \"Vanilla FQI\")\n",
    "plt.xlabel(\"Iteration\")\n",
    "plt.ylabel(\"Q Estimate\")\n",
    "plt.legend()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# CFQI"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "code_folding": [
     0
    ]
   },
   "outputs": [],
   "source": [
    "class CFQIagent():\n",
    "    def __init__(self, train_tuples, test_tuples, iters=150, gamma=0.99, batch_size=100, prioritize=False, estimator='lin',\n",
    "                 weights=np.array([1, 1, 1, 1, 1])/5., maxT=36):\n",
    "        \n",
    "        self.iters = iters\n",
    "        self.gamma = gamma\n",
    "        self.batch_size = batch_size\n",
    "        self.prioritize_a = prioritize\n",
    "        self.training_set, self.test_set = construct_dicts(train_tuples, test_tuples)\n",
    "        self.raw_test = test_tuples\n",
    "        \n",
    "        self.visits = {'train': len(train_tuples), 'test': len(test_tuples)}\n",
    "        self.NV = {'train': len(train_tuples), 'test': len(test_tuples)}\n",
    "        self.n_samples = len(self.training_set['s'])\n",
    "        _, self.unique_actions, self.action_counts, _ = self.sub_actions()\n",
    "        self.state_feats = [str(x) for x in range(10)]\n",
    "        self.n_features = len(self.state_feats)\n",
    "        self.reward_weights = weights\n",
    "        self.maxT = maxT\n",
    "        self.piB = learnBehaviour(self.training_set, self.test_set)\n",
    "        self.n_actions = 4\n",
    "        \n",
    "        if estimator == 'tree':\n",
    "            self.q_est = ExtraTreesRegressor(n_estimators=50, max_depth=None, min_samples_leaf=10, min_samples_split=2,\n",
    "                                             random_state=0)\n",
    "        elif estimator == 'gbm':\n",
    "            self.q_est = LGBMRegressor(n_estimators=50, silent=True)\n",
    "\n",
    "        elif estimator == 'nn':\n",
    "            self.q_est = None\n",
    "        \n",
    "        elif estimator == 'lin':\n",
    "            self.q_est_shared = LinearRegression()\n",
    "            self.q_est_fg = LinearRegression()\n",
    "            \n",
    "        self.piE = LinearRegression()#LGBMClassifier(n_estimators=50, silent=True)\n",
    "        \n",
    "        self.eval_est = LGBMRegressor(n_estimators=50, silent=True)\n",
    "\n",
    "    def sub_actions(self):\n",
    "        \n",
    "        a = self.training_set['a']\n",
    "        a = list(a)\n",
    "        \n",
    "        unique_actions = 0\n",
    "        action_counts = 0\n",
    "        n_actions = 0\n",
    "        \n",
    "        unique_actions, action_counts = np.unique(a, axis=0, return_counts=True)\n",
    "        n_actions = len(unique_actions)\n",
    "                \n",
    "        return a, unique_actions, action_counts, n_actions\n",
    "    \n",
    "    def sampleTuples(self):\n",
    "        ids = list(np.random.choice(np.arange(self.n_samples), self.batch_size, replace=False))\n",
    "        batch = {}\n",
    "        for k in self.training_set.keys():\n",
    "            batch[k] = np.asarray(self.training_set[k], dtype=object)[ids]\n",
    "        batch['r'] = np.dot(batch['r'] * [1, 1, 10, 10, 100], self.reward_weights)\n",
    "        batch['s_ids'] = np.asarray(ids, dtype=int)\n",
    "        batch['ns_ids'] = np.asarray(ids, dtype=int) + 1\n",
    "            \n",
    "    \n",
    "        return batch\n",
    "    \n",
    "    def fitQ(self, batch, Q):\n",
    "        \n",
    "        # Divide into foreground and background batches. \n",
    "        batch_foreground = {}\n",
    "        batch_background = {}\n",
    "        \n",
    "        elts = ['s', 'a', 'ns', 'r', 'ds', 'vnum', 's_ids', 'ns_ids']\n",
    "        for el in elts:\n",
    "            batch_foreground[el] = []\n",
    "            batch_background[el] = []\n",
    "        \n",
    "        for i in range(len(batch['s_ids'])):\n",
    "            if batch['ds'][i] == 'foreground':\n",
    "                for k in batch.keys():\n",
    "                    batch_foreground[k].append(batch[k][i])\n",
    "            else:\n",
    "                for k in batch.keys():\n",
    "                    batch_background[k].append(batch[k][i])\n",
    "            \n",
    "        # input = [state action]\n",
    "        x_fg =  np.hstack((batch_foreground['s'], batch_foreground['a']))\n",
    "        x_shared = np.hstack((batch['s'], batch['a'])) \n",
    "        \n",
    "        # target = r + gamma * max_a(Q(s', a))      == r for first iteration\n",
    "        y_fg = batch_foreground['r'] + (self.gamma * np.max(Q[batch_foreground['ns_ids'], :], axis=1))\n",
    "        y_shared = batch['r'] + (self.gamma * np.max(Q[batch['ns_ids'], :], axis=1))\n",
    "        \n",
    "        # Used mixed model here\n",
    "        self.q_est_shared.fit(x_shared, y_shared)\n",
    "        self.q_est_fg.fit(x_fg, y_fg)\n",
    "        \n",
    "        return batch_foreground, batch_background\n",
    "    \n",
    "    def updateQtable(self, Qtable, batch_fg, batch_bg):\n",
    "        # Update for foregound using just foreground\n",
    "        # Update for background using shared\n",
    "        \n",
    "        bg_size = len(batch_bg['s'])\n",
    "        fg_size = len(batch_fg['s'])\n",
    "        for i, a in enumerate(self.unique_actions):\n",
    "            Qtable[batch_bg['s_ids'], i] = self.q_est_shared.predict(np.hstack((batch_bg['ns'], np.tile(a, (bg_size,1)))))\n",
    "            Qtable[batch_fg['s_ids'], i] = self.q_est_fg.predict(np.hstack((batch_fg['ns'], np.tile(a, (fg_size,1)))))\n",
    "        return Qtable\n",
    "    \n",
    "    def runFQI(self, repeats=10):\n",
    "        \n",
    "        print('Learning policy')\n",
    "        meanQtable = np.zeros((self.n_samples + 1, self.n_actions))\n",
    "        \n",
    "        for r in range(repeats):\n",
    "            print('Run', r, ':')\n",
    "            print('Initialize: get batch, set initial Q')\n",
    "            Qtable = np.zeros((self.n_samples + 1, self.n_actions))\n",
    "            Qdist = []\n",
    "\n",
    "            #print('Run FQI')\n",
    "            for iteration in range(self.iters):\n",
    "\n",
    "                # copy q-table\n",
    "                Qold = cp.deepcopy(Qtable)\n",
    "\n",
    "                # sample batch  \n",
    "                batch = self.sampleTuples()\n",
    "\n",
    "                # learn q_est with samples, targets from batch\n",
    "                batch_foreground, batch_background = self.fitQ(batch, Qtable)\n",
    "\n",
    "                # update Q table for all s given new estimator\n",
    "                self.updateQtable(Qtable, batch_foreground, batch_background)\n",
    "\n",
    "                # check divergence from last estimate\n",
    "                Qdist.append(mean_absolute_error(Qold, Qtable))\n",
    "         \n",
    "            #plt.plot(Qdist)\n",
    "            meanQtable += Qtable\n",
    "        \n",
    "        meanQtable = meanQtable / repeats\n",
    "        print('Learn policy')\n",
    "        \n",
    "        # Since the Q table is constructed contrastively, the policy is contrastive?\n",
    "        self.getPi(meanQtable)\n",
    "        return Qdist\n",
    "                    \n",
    "    \n",
    "    def getPi(self, Qtable):\n",
    "        optA = np.argmax(Qtable, axis=1)\n",
    "        print(\"Opta: \", optA)\n",
    "        #print(\"Fitting to training set\")\n",
    "        #print(\"Optimal actions: \", optA)\n",
    "        self.piE.fit(self.training_set['s'], optA[:-1])\n",
    "        #print(\"Done Fitting\")\n",
    "    \n",
    "    def testPi(self, behavior):\n",
    "        accurate = 0\n",
    "        total = 0\n",
    "        \n",
    "        for tup in self.raw_test:\n",
    "            s = tup[0]\n",
    "            try:\n",
    "                a = tup[1]\n",
    "                a = np.concatenate(a).ravel()\n",
    "                a = list(a)\n",
    "            except:\n",
    "                a = tup[1]\n",
    "            # actions based on policy we learn\n",
    "            s = s.reshape((1, 10))\n",
    "            evalA = self.piE.predict(s)\n",
    "            \n",
    "            # predicted actions based on historical actions model\n",
    "            behavB = behavior.predict(s)\n",
    "            \n",
    "            if behavB <= 0.25:\n",
    "                behavB = 0\n",
    "            elif behavB <= 0.5:\n",
    "                behavB = 1\n",
    "            elif behavB <= 0.75:\n",
    "                behavB = 2\n",
    "            else:\n",
    "                behavB = 3\n",
    "            \n",
    "            # actual historical actions\n",
    "            actions = [[0, 0], [0, 1], [1, 0], [1, 1]]\n",
    "            behavA = actions.index(a)\n",
    "            \n",
    "            if behavA == behavB:\n",
    "                accurate += 1\n",
    "            total += 1\n",
    "        \n",
    "        return float(accurate)/total"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "cfqi_agent = CFQIagent(train_tuples=train_tuples, test_tuples=test_tuples)\n",
    "Q_dist = cfqi_agent.runFQI(repeats=1)\n",
    "plt.plot(Q_dist, label= \"Contrastive FQI\")\n",
    "plt.xlabel(\"Iteration\")\n",
    "plt.ylabel(\"Q Estimate\")\n",
    "plt.legend()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "cfqi_agent.q_est_shared.coef_"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "cfqi_agent.q_est_fg.coef_"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Statsmodels MixedLM\n",
    "* Use the statsmodels api to create mixed model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "code_folding": []
   },
   "outputs": [],
   "source": [
    "class SMFQIagent():\n",
    "    def __init__(self, train_tuples, test_tuples, iters=150, gamma=0.99, batch_size=100, prioritize=False, estimator='lin',\n",
    "                 weights=np.array([1, 1, 1, 1, 1])/5., maxT=36):\n",
    "        \n",
    "        self.iters = iters\n",
    "        self.gamma = gamma\n",
    "        self.batch_size = batch_size\n",
    "        self.prioritize_a = prioritize\n",
    "        self.training_set, self.test_set = construct_dicts(train_tuples, test_tuples)\n",
    "        self.raw_test = test_tuples\n",
    "        \n",
    "        self.visits = {'train': len(train_tuples), 'test': len(test_tuples)}\n",
    "        self.NV = {'train': len(train_tuples), 'test': len(test_tuples)}\n",
    "        self.n_samples = len(self.training_set['s'])\n",
    "        _, self.unique_actions, self.action_counts, _ = self.sub_actions()\n",
    "        self.state_feats = [str(x) for x in range(10)]\n",
    "        self.n_features = len(self.state_feats)\n",
    "        self.reward_weights = weights\n",
    "        self.maxT = maxT\n",
    "        self.piB = learnBehaviour(self.training_set, self.test_set)\n",
    "        self.n_actions = 4\n",
    "        \n",
    "        self.q_est_shared = LinearRegression()\n",
    "        self.q_est_fg = LinearRegression()\n",
    "            \n",
    "        self.piE = LinearRegression()\n",
    "        self.eval_est = LGBMRegressor(n_estimators=50, silent=True)\n",
    "\n",
    "    def sub_actions(self):\n",
    "        \n",
    "        a = self.training_set['a']\n",
    "        a = list(a)\n",
    "        \n",
    "        unique_actions = 0\n",
    "        action_counts = 0\n",
    "        n_actions = 0\n",
    "        \n",
    "        unique_actions, action_counts = np.unique(a, axis=0, return_counts=True)\n",
    "        n_actions = len(unique_actions)\n",
    "                \n",
    "        return a, unique_actions, action_counts, n_actions\n",
    "    \n",
    "    def sampleTuples(self):\n",
    "        ids = list(np.random.choice(np.arange(self.n_samples), self.batch_size, replace=False))\n",
    "        batch = {}\n",
    "        for k in self.training_set.keys():\n",
    "            batch[k] = np.asarray(self.training_set[k], dtype=object)[ids]\n",
    "        batch['r'] = np.dot(batch['r'] * [1, 1, 10, 10, 100], self.reward_weights)\n",
    "        batch['s_ids'] = np.asarray(ids, dtype=int)\n",
    "        batch['ns_ids'] = np.asarray(ids, dtype=int) + 1\n",
    "            \n",
    "    \n",
    "        return batch\n",
    "    \n",
    "    def fitQ(self, batch, Q):\n",
    "        \n",
    "        # Divide into foreground and background batches. \n",
    "        batch_foreground = {}\n",
    "        batch_background = {}\n",
    "        \n",
    "        elts = ['s', 'a', 'ns', 'r', 'ds', 'vnum', 's_ids', 'ns_ids']\n",
    "        for el in elts:\n",
    "            batch_foreground[el] = []\n",
    "            batch_background[el] = []\n",
    "        \n",
    "        for i in range(len(batch['s_ids'])):\n",
    "            if batch['ds'][i] == 'foreground':\n",
    "                for k in batch.keys():\n",
    "                    batch_foreground[k].append(batch[k][i])\n",
    "            else:\n",
    "                for k in batch.keys():\n",
    "                    batch_background[k].append(batch[k][i])\n",
    "            \n",
    "        # input = [state action]\n",
    "        x_fg =  np.hstack((batch_foreground['s'], batch_foreground['a']))\n",
    "        x_shared = np.hstack((batch['s'], batch['a'])) \n",
    "        \n",
    "        # target = r + gamma * max_a(Q(s', a))      == r for first iteration\n",
    "        y_fg = batch_foreground['r'] + (self.gamma * np.max(Q[batch_foreground['ns_ids'], :], axis=1))\n",
    "        y_shared = batch['r'] + (self.gamma * np.max(Q[batch['ns_ids'], :], axis=1))\n",
    "        \n",
    "        self.q_est_shared.fit(x_shared, y_shared)\n",
    "        self.q_est_fg.fit(x_fg, y_fg)\n",
    "        \n",
    "        return batch_foreground, batch_background\n",
    "    \n",
    "    def updateQtable(self, Qtable, batch_fg, batch_bg):\n",
    "        # Update for foregound using just foreground\n",
    "        # Update for background using shared\n",
    "        \n",
    "        bg_size = len(batch_bg['s'])\n",
    "        fg_size = len(batch_fg['s'])\n",
    "        for i, a in enumerate(self.unique_actions):\n",
    "            Qtable[batch_bg['s_ids'], i] = self.q_est_shared.predict(np.hstack((batch_bg['ns'], np.tile(a, (bg_size,1)))))\n",
    "            Qtable[batch_fg['s_ids'], i] = self.q_est_fg.predict(np.hstack((batch_fg['ns'], np.tile(a, (fg_size,1)))))\n",
    "        return Qtable\n",
    "    \n",
    "    def runFQI(self, repeats=10):\n",
    "        \n",
    "        print('Learning policy')\n",
    "        meanQtable = np.zeros((self.n_samples + 1, self.n_actions))\n",
    "        \n",
    "        for r in range(repeats):\n",
    "            print('Run', r, ':')\n",
    "            print('Initialize: get batch, set initial Q')\n",
    "            Qtable = np.zeros((self.n_samples + 1, self.n_actions))\n",
    "            Qdist = []\n",
    "\n",
    "            #print('Run FQI')\n",
    "            for iteration in range(self.iters):\n",
    "\n",
    "                # copy q-table\n",
    "                Qold = cp.deepcopy(Qtable)\n",
    "\n",
    "                # sample batch  \n",
    "                batch = self.sampleTuples()\n",
    "\n",
    "                # learn q_est with samples, targets from batch\n",
    "                batch_foreground, batch_background = self.fitQ(batch, Qtable)\n",
    "\n",
    "                # update Q table for all s given new estimator\n",
    "                self.updateQtable(Qtable, batch_foreground, batch_background)\n",
    "\n",
    "                # check divergence from last estimate\n",
    "                Qdist.append(mean_absolute_error(Qold, Qtable))\n",
    "         \n",
    "            #plt.plot(Qdist)\n",
    "            meanQtable += Qtable\n",
    "        \n",
    "        meanQtable = meanQtable / repeats\n",
    "        print('Learn policy')\n",
    "        \n",
    "        # Since the Q table is constructed contrastively, the policy is contrastive?\n",
    "        self.getPi(meanQtable)\n",
    "        return Qdist\n",
    "                    \n",
    "    \n",
    "    def getPi(self, Qtable):\n",
    "        optA = np.argmax(Qtable, axis=1)\n",
    "        print(\"Opta: \", optA)\n",
    "        #print(\"Fitting to training set\")\n",
    "        #print(\"Optimal actions: \", optA)\n",
    "        self.piE.fit(self.training_set['s'], optA[:-1])\n",
    "        #print(\"Done Fitting\")\n",
    "    \n",
    "    def testPi(self, behavior):\n",
    "        accurate = 0\n",
    "        total = 0\n",
    "        \n",
    "        for tup in self.raw_test:\n",
    "            s = tup[0]\n",
    "            try:\n",
    "                a = tup[1]\n",
    "                a = np.concatenate(a).ravel()\n",
    "                a = list(a)\n",
    "            except:\n",
    "                a = tup[1]\n",
    "            # actions based on policy we learn\n",
    "            s = s.reshape((1, 10))\n",
    "            evalA = self.piE.predict(s)\n",
    "            \n",
    "            # predicted actions based on historical actions model\n",
    "            behavB = behavior.predict(s)\n",
    "            \n",
    "            if behavB <= 0.25:\n",
    "                behavB = 0\n",
    "            elif behavB <= 0.5:\n",
    "                behavB = 1\n",
    "            elif behavB <= 0.75:\n",
    "                behavB = 2\n",
    "            else:\n",
    "                behavB = 3\n",
    "            \n",
    "            # actual historical actions\n",
    "            actions = [[0, 0], [0, 1], [1, 0], [1, 1]]\n",
    "            behavA = actions.index(a)\n",
    "            \n",
    "            if behavA == behavB:\n",
    "                accurate += 1\n",
    "            total += 1\n",
    "        \n",
    "        return float(accurate)/total"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "mmfqi_agent = MMFQIagent(train_tuples=train_tuples, test_tuples=test_tuples)\n",
    "Q_dist = mmfqi_agent.runFQI(repeats=1)\n",
    "plt.plot(Q_dist, label= \"Mixed Model FQI\")\n",
    "plt.xlabel(\"Iteration\")\n",
    "plt.ylabel(\"Q Estimate\")\n",
    "plt.legend()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Custom LMM\n",
    "* Using Andy's code"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "code_folding": []
   },
   "outputs": [],
   "source": [
    "class LMMagent():\n",
    "    def __init__(self, train_tuples, test_tuples, iters=150, gamma=0.99, batch_size=100, prioritize=False, estimator='lin',\n",
    "                 weights=np.array([1, 1, 1, 1, 1])/5., maxT=36):\n",
    "        \n",
    "        self.iters = iters\n",
    "        self.gamma = gamma\n",
    "        self.batch_size = batch_size\n",
    "        self.prioritize_a = prioritize\n",
    "        self.training_set, self.test_set = construct_dicts(train_tuples, test_tuples)\n",
    "        self.raw_test = test_tuples\n",
    "        \n",
    "        self.visits = {'train': len(train_tuples), 'test': len(test_tuples)}\n",
    "        self.NV = {'train': len(train_tuples), 'test': len(test_tuples)}\n",
    "        self.n_samples = len(self.training_set['s'])\n",
    "        _, self.unique_actions, self.action_counts, _ = self.sub_actions()\n",
    "        self.state_feats = [str(x) for x in range(10)]\n",
    "        self.n_features = len(self.state_feats)\n",
    "        self.reward_weights = weights\n",
    "        self.maxT = maxT\n",
    "        self.piB = learnBehaviour(self.training_set, self.test_set)\n",
    "        self.n_actions = 4\n",
    "        \n",
    "        self.q_est_shared = LinearRegression()\n",
    "        self.q_est_fg = LinearRegression()\n",
    "            \n",
    "        self.piE = LinearRegression()\n",
    "        self.eval_est = LGBMRegressor(n_estimators=50, silent=True)\n",
    "\n",
    "    def sub_actions(self):\n",
    "        \n",
    "        a = self.training_set['a']\n",
    "        a = list(a)\n",
    "        \n",
    "        unique_actions = 0\n",
    "        action_counts = 0\n",
    "        n_actions = 0\n",
    "        \n",
    "        unique_actions, action_counts = np.unique(a, axis=0, return_counts=True)\n",
    "        n_actions = len(unique_actions)\n",
    "                \n",
    "        return a, unique_actions, action_counts, n_actions\n",
    "    \n",
    "    def sampleTuples(self):\n",
    "        ids = list(np.random.choice(np.arange(self.n_samples), self.batch_size, replace=False))\n",
    "        batch = {}\n",
    "        for k in self.training_set.keys():\n",
    "            batch[k] = np.asarray(self.training_set[k], dtype=object)[ids]\n",
    "        batch['r'] = np.dot(batch['r'] * [1, 1, 10, 10, 100], self.reward_weights)\n",
    "        batch['s_ids'] = np.asarray(ids, dtype=int)\n",
    "        batch['ns_ids'] = np.asarray(ids, dtype=int) + 1\n",
    "            \n",
    "    \n",
    "        return batch\n",
    "    \n",
    "    def fitQ(self, batch, Q):\n",
    "        \n",
    "        # Divide into foreground and background batches. \n",
    "        batch_foreground = {}\n",
    "        batch_background = {}\n",
    "        \n",
    "        elts = ['s', 'a', 'ns', 'r', 'ds', 'vnum', 's_ids', 'ns_ids']\n",
    "        for el in elts:\n",
    "            batch_foreground[el] = []\n",
    "            batch_background[el] = []\n",
    "        \n",
    "        for i in range(len(batch['s_ids'])):\n",
    "            if batch['ds'][i] == 'foreground':\n",
    "                for k in batch.keys():\n",
    "                    batch_foreground[k].append(batch[k][i])\n",
    "            else:\n",
    "                for k in batch.keys():\n",
    "                    batch_background[k].append(batch[k][i])\n",
    "            \n",
    "        # input = [state action]\n",
    "        x_fg =  np.hstack((batch_foreground['s'], batch_foreground['a']))\n",
    "        x_shared = np.hstack((batch['s'], batch['a'])) \n",
    "        \n",
    "        # target = r + gamma * max_a(Q(s', a))      == r for first iteration\n",
    "        y_fg = batch_foreground['r'] + (self.gamma * np.max(Q[batch_foreground['ns_ids'], :], axis=1))\n",
    "        y_shared = batch['r'] + (self.gamma * np.max(Q[batch['ns_ids'], :], axis=1))\n",
    "        \n",
    "        self.q_est_shared.fit(x_shared, y_shared)\n",
    "        self.q_est_fg.fit(x_fg, y_fg)\n",
    "        \n",
    "        return batch_foreground, batch_background\n",
    "    \n",
    "    def updateQtable(self, Qtable, batch_fg, batch_bg):\n",
    "        # Update for foregound using just foreground\n",
    "        # Update for background using shared\n",
    "        \n",
    "        bg_size = len(batch_bg['s'])\n",
    "        fg_size = len(batch_fg['s'])\n",
    "        for i, a in enumerate(self.unique_actions):\n",
    "            Qtable[batch_bg['s_ids'], i] = self.q_est_shared.predict(np.hstack((batch_bg['ns'], np.tile(a, (bg_size,1)))))\n",
    "            Qtable[batch_fg['s_ids'], i] = self.q_est_fg.predict(np.hstack((batch_fg['ns'], np.tile(a, (fg_size,1)))))\n",
    "        return Qtable\n",
    "    \n",
    "    def runFQI(self, repeats=10):\n",
    "        \n",
    "        print('Learning policy')\n",
    "        meanQtable = np.zeros((self.n_samples + 1, self.n_actions))\n",
    "        \n",
    "        for r in range(repeats):\n",
    "            print('Run', r, ':')\n",
    "            print('Initialize: get batch, set initial Q')\n",
    "            Qtable = np.zeros((self.n_samples + 1, self.n_actions))\n",
    "            Qdist = []\n",
    "\n",
    "            #print('Run FQI')\n",
    "            for iteration in range(self.iters):\n",
    "\n",
    "                # copy q-table\n",
    "                Qold = cp.deepcopy(Qtable)\n",
    "\n",
    "                # sample batch  \n",
    "                batch = self.sampleTuples()\n",
    "\n",
    "                # learn q_est with samples, targets from batch\n",
    "                batch_foreground, batch_background = self.fitQ(batch, Qtable)\n",
    "\n",
    "                # update Q table for all s given new estimator\n",
    "                self.updateQtable(Qtable, batch_foreground, batch_background)\n",
    "\n",
    "                # check divergence from last estimate\n",
    "                Qdist.append(mean_absolute_error(Qold, Qtable))\n",
    "         \n",
    "            #plt.plot(Qdist)\n",
    "            meanQtable += Qtable\n",
    "        \n",
    "        meanQtable = meanQtable / repeats\n",
    "        print('Learn policy')\n",
    "        \n",
    "        # Since the Q table is constructed contrastively, the policy is contrastive?\n",
    "        self.getPi(meanQtable)\n",
    "        return Qdist\n",
    "                    \n",
    "    \n",
    "    def getPi(self, Qtable):\n",
    "        optA = np.argmax(Qtable, axis=1)\n",
    "        print(\"Opta: \", optA)\n",
    "        #print(\"Fitting to training set\")\n",
    "        #print(\"Optimal actions: \", optA)\n",
    "        self.piE.fit(self.training_set['s'], optA[:-1])\n",
    "        #print(\"Done Fitting\")\n",
    "    \n",
    "    def testPi(self, behavior):\n",
    "        accurate = 0\n",
    "        total = 0\n",
    "        \n",
    "        for tup in self.raw_test:\n",
    "            s = tup[0]\n",
    "            try:\n",
    "                a = tup[1]\n",
    "                a = np.concatenate(a).ravel()\n",
    "                a = list(a)\n",
    "            except:\n",
    "                a = tup[1]\n",
    "            # actions based on policy we learn\n",
    "            s = s.reshape((1, 10))\n",
    "            evalA = self.piE.predict(s)\n",
    "            \n",
    "            # predicted actions based on historical actions model\n",
    "            behavB = behavior.predict(s)\n",
    "            \n",
    "            if behavB <= 0.25:\n",
    "                behavB = 0\n",
    "            elif behavB <= 0.5:\n",
    "                behavB = 1\n",
    "            elif behavB <= 0.75:\n",
    "                behavB = 2\n",
    "            else:\n",
    "                behavB = 3\n",
    "            \n",
    "            # actual historical actions\n",
    "            actions = [[0, 0], [0, 1], [1, 0], [1, 1]]\n",
    "            behavA = actions.index(a)\n",
    "            \n",
    "            if behavA == behavB:\n",
    "                accurate += 1\n",
    "            total += 1\n",
    "        \n",
    "        return float(accurate)/total"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Validation (FQI, CFQI, Oracle, Random)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "code_folding": [
     0
    ]
   },
   "outputs": [],
   "source": [
    "def cumulative_reward(rewards):\n",
    "    c_reward = [rewards[0]]\n",
    "    for i in range(1, len(rewards)):\n",
    "        c_reward.append(rewards[i] + c_reward[i-1])\n",
    "    return c_reward"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "code_folding": [
     0
    ]
   },
   "outputs": [],
   "source": [
    "# Test out on each test tuple\n",
    "# FQI, CFQI, Oracle, Random\n",
    "algos = ['fqi', 'cfqi', 'oracle', 'random']\n",
    "overall_reward = {}\n",
    "for alg in algos:\n",
    "    overall_reward[alg] = []\n",
    "for k, pat in enumerate(tqdm.tqdm(range(num_patients))):\n",
    "    \n",
    "    flip = np.random.choice(2)\n",
    "    if flip == 0:\n",
    "        ds = 'foreground'\n",
    "    else:\n",
    "        ds = 'background'\n",
    "    # Generate a random initial state\n",
    "    s = np.random.normal(mu, sigma, (10, 1))\n",
    "    \n",
    "    val_rewards = {}\n",
    "    for alg in algos:\n",
    "        val_rewards[alg] = []\n",
    "    \n",
    "    \n",
    "    # Generate all of the tuples for this patient\n",
    "    for i in range(num_samples):\n",
    "        s = s.T\n",
    "        # FQI agent\n",
    "        fqi_action = fqi_agent.piE.predict(s)\n",
    "        if fqi_action[0] > 3:\n",
    "            fqi_action[0] = 3\n",
    "        fqi_action = actions[round(fqi_action[0])]\n",
    "        fqi_action = np.reshape(fqi_action, (2, 1))\n",
    "        s_a = np.concatenate((s.T, fqi_action))\n",
    "        val_rewards['fqi'].append(np.dot(reward_function.T, s_a)[0])\n",
    "\n",
    "\n",
    "        # CFQI agent\n",
    "        cfqi_action = cfqi_agent.piE.predict(s)\n",
    "        if cfqi_action[0] > 3:\n",
    "            cfqi_action[0] = 3\n",
    "        cfqi_action = actions[round(cfqi_action[0])]\n",
    "        cfqi_action = np.reshape(cfqi_action, (2, 1))\n",
    "        s_a = np.concatenate((s.T, cfqi_action))\n",
    "        val_rewards['cfqi'].append(np.dot(reward_function.T, s_a)[0])\n",
    "\n",
    "\n",
    "        # Oracle\n",
    "        all_rewards = []\n",
    "        for j, a in enumerate(actions):\n",
    "            a = np.asarray(a)\n",
    "            a = np.reshape(a, (2, 1))\n",
    "            s_a = np.concatenate((s.T, a))\n",
    "            reward = np.dot(reward_function.T, s_a)\n",
    "            all_rewards.append(reward)\n",
    "\n",
    "        all_rewards = np.asarray(all_rewards)\n",
    "        oracle_action = actions[np.argmax(all_rewards)]\n",
    "        val_rewards['oracle'].append(np.max(all_rewards))\n",
    "\n",
    "\n",
    "        # Random action\n",
    "        random_action = np.asarray(actions[np.random.choice(3)])\n",
    "        random_action = np.reshape(random_action, (2, 1))\n",
    "        s_a = np.concatenate((s.T, random_action))\n",
    "        val_rewards['random'].append(np.dot(reward_function.T, s_a)[0])\n",
    "        \n",
    "        if ds == 'foreground':\n",
    "            t_m = transition_foreground\n",
    "        else:\n",
    "            t_m = transition_background\n",
    "        ns = np.matmul(s_a.T, t_m) / np.linalg.norm(np.matmul(s_a.T, t_m), ord=2)\n",
    "        ns = np.add(ns, np.random.normal(0, 0.5, (1, 10))) # Add noise\n",
    "        s = ns.T\n",
    "    \n",
    "    plt.title(\"Rewards for trajectory: \" + str(k))\n",
    "    plt.xlabel(\"Step\")\n",
    "    plt.ylabel(\"Cumulative Reward\")\n",
    "    x = [i for i in range(num_samples)]\n",
    "    rewards_fqi = cumulative_reward(val_rewards['fqi'])\n",
    "    overall_reward['fqi'].append(rewards_fqi[-1])\n",
    "    rewards_cfqi = cumulative_reward(val_rewards['cfqi'])\n",
    "    overall_reward['cfqi'].append(rewards_cfqi[-1])\n",
    "    rewards_oracle = cumulative_reward(val_rewards['oracle'])\n",
    "    overall_reward['oracle'].append(rewards_oracle[-1])\n",
    "    rewards_random = cumulative_reward(val_rewards['random'])\n",
    "    overall_reward['random'].append(rewards_random[-1])\n",
    "    \n",
    "    \n",
    "    plt.plot(x, rewards_fqi, label=\"FQI\")\n",
    "    plt.plot(x, rewards_cfqi, label='CFQI')\n",
    "    plt.plot(x, rewards_oracle, label='Oracle')\n",
    "    plt.plot(x, rewards_random, label='Random')\n",
    "    plt.legend()\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.title(\"Cumulative Reward across patients and algorithms\")\n",
    "sns.stripplot(overall_reward['fqi'], color='r', label='FQI')\n",
    "sns.stripplot(overall_reward['cfqi'], color='g', label='CFQI')\n",
    "sns.stripplot(overall_reward['random'], color='b', label='Random')\n",
    "sns.stripplot(overall_reward['oracle'], color='m', label=\"Oracle\")\n",
    "plt.legend()\n",
    "plt.xlabel(\"Cumulative Reward\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "reward_fqi = np.asarray(overall_reward['fqi'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "#_, axs = plt.subplots(2, 2)\n",
    "fig = plt.figure(figsize=(10,10))\n",
    "#sns.set_theme(style=\"whitegrid\")\n",
    "plt.boxplot(x=[np.asarray(overall_reward['fqi']).squeeze().tolist(), np.asarray(overall_reward['cfqi']).squeeze().tolist(), np.asarray(overall_reward['random']).squeeze().tolist(), np.asarray(overall_reward['oracle']).squeeze().tolist()])# axs[0, 1].boxplot(x=np.asarray(overall_reward['cfqi']).squeeze().tolist())\n",
    "# axs[1, 0].set_title(\"Random\")\n",
    "# axs[1, 0].boxplot(x=np.asarray(overall_reward['random']).squeeze().tolist())\n",
    "# axs[1, 1].set_title(\"Oracle\")\n",
    "# axs[1, 1].boxplot(x=np.asarray(overall_reward['oracle']).squeeze().tolist())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "research",
   "language": "python",
   "name": "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.7.9"
  },
  "toc": {
   "base_numbering": 1,
   "nav_menu": {},
   "number_sections": true,
   "sideBar": true,
   "skip_h1_title": false,
   "title_cell": "Table of Contents",
   "title_sidebar": "Contents",
   "toc_cell": false,
   "toc_position": {},
   "toc_section_display": true,
   "toc_window_display": false
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
