{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "3bc0d1e3",
   "metadata": {},
   "source": [
    "# Notebook to run Facebook news experiment\n",
    "\n",
    "### This notebook implements IGL-P(3) and evaluates its performance on a simulator using Facebook news data for three latent reward states"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "cdf8fc66",
   "metadata": {},
   "source": [
    "### Preprocessed data\n",
    "\n",
    "For ease of experimentation, we include preprocessed data files with our code (action_set.csv). We use 3 files from the Martinchek dataset corresponding to all posts made by (1) Fox News, (2) The Huffington Post and (3) TIME Magazine. Each row of the original dataset included a post_id (this is preserved in our preprocessed data set), information about the type of post, date of post, text information, total number of interactions for (1) likes, (2) comments, (3) shares, (4) love, (5) wow, (6) haha, (7) sad and (8) angry.\n",
    "\n",
    "For each data set we compute a correlation matrix and then use PCA to obtain a low dimensional representation of the structure between different signals. We use clustering on the results to determine how feedback signals are related and then assign the cluster containing the 'like' signal to correspond to r=1 signals, the cluster containing the angry signal to correspond to r=-1 and the remaining cluster contains signals that correspond to the r=0 state. For the probabilities of a user emitting a given signal while in a state, we use aggregate statistics for the given news source across 6 months of data. Finally, we introduce the 'none' signal for r=0 and assign it a probability of 0.9 to account for the fact that the vast majority of people do not interact with posts.\n",
    "\n",
    "Using the embedding of signals, we compute a latent score for each post. We then look at all the posts from a given day and chose the post with the highest latent score as being the action with r=1. For the remaining actions we chose 14 actions with low engagement rates, implying that they have r!=1. Of the 14 chosen actions, we select the action with the lowest latent score as having r=-1. We do this for all days in the 6 month period from May to November 2016. This gives us 175 \"action sets\" that each consist of 15 actions, where 1 has reward r=1, 1 has reward r=-1 and 13 have reward r=0.\n",
    "\n",
    "For the action features of the posts, we use the 'paraphrase-MiniLM-L6-v2' from SentenceTransformers to summarize the text content of the post, as well as information about the type of post (link, video, photo, status or event), encoded as a one-hot. The resulting action features are nearly 400 dimensional vectors, and all action features in the preprocessed data set were generated in the same way."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "93efc097",
   "metadata": {},
   "outputs": [],
   "source": [
    "def from_np_array(array_string): # otherwise have issues reading in numpy arrays from pd csv\n",
    "    import numpy as np\n",
    "    import ast\n",
    "    array_string = ','.join(array_string.replace('[ ', '[').split())\n",
    "    return np.array(ast.literal_eval(array_string))\n",
    "\n",
    "def get_cooked_data(file_name):\n",
    "    import pandas as pd\n",
    "    data = pd.read_csv('action_sets.csv',index_col=0,\n",
    "                   converters={'action_feat': from_np_array})\n",
    "    return data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "f6fb735a",
   "metadata": {},
   "outputs": [],
   "source": [
    "data = get_cooked_data('action_sets.csv')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7ce0ca26",
   "metadata": {},
   "source": [
    "## Accuracy helper function"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "infectious-albuquerque",
   "metadata": {
    "code_folding": [
     2,
     31,
     53,
     72,
     83,
     112,
     150,
     151
    ]
   },
   "outputs": [],
   "source": [
    "import torch\n",
    "import pandas as pd\n",
    "\n",
    "class EasyAcc:\n",
    "    def __init__(self):\n",
    "        self.n = 0\n",
    "        self.sum = 0\n",
    "        self.sumsq = 0\n",
    "\n",
    "    def __iadd__(self, other):\n",
    "        self.n += 1\n",
    "        self.sum += other\n",
    "        self.sumsq += other*other\n",
    "        return self\n",
    "\n",
    "    def __isub__(self, other):\n",
    "        self.n += 1\n",
    "        self.sum -= other\n",
    "        self.sumsq += other*other\n",
    "        return self\n",
    "\n",
    "    def mean(self):\n",
    "        return self.sum / max(self.n, 1)\n",
    "\n",
    "    def var(self):\n",
    "        from math import sqrt\n",
    "        return sqrt(self.sumsq / max(self.n, 1) - self.mean()**2)\n",
    "\n",
    "    def semean(self):\n",
    "        from math import sqrt\n",
    "        return self.var() / sqrt(max(self.n, 1))\n",
    "\n",
    "class IK(torch.nn.Module):\n",
    "    def __init__(self, dim_x, dim_a, dim_y):\n",
    "        super().__init__()\n",
    "        self.bilinear = torch.nn.Bilinear(in1_features=dim_x*dim_a, in2_features=dim_y, out_features=1)\n",
    "        self.softmax = torch.nn.Softmax(dim=-1)\n",
    "        \n",
    "    # Xs --> [batch, dim_x] \n",
    "    # As --> [batch, num_a, dim_a] \n",
    "    # Ys --> [batch, dim_y]\n",
    "    def prelogits(self, Xs, As, Ys):\n",
    "        batch, numa, dima = As.shape\n",
    "        _, dimx = Xs.shape\n",
    "        \n",
    "        xview = Xs.unsqueeze(1).unsqueeze(3)            # [ batch, 1, dim_x, 1 ]\n",
    "        aview = As.unsqueeze(2)                         # [ batch, num_a, 1, dim_a ]\n",
    "        xaouter = torch.matmul(xview, aview)            # [ batch, num_a, dim_x, dim_a ]\n",
    "        yview = Ys.unsqueeze(1).expand(-1, numa, -1)    # [ batch, num_a, dim_y ]\n",
    "        return self.bilinear(xaouter.view(-1, numa, dimx*dima), yview).view(-1, numa)\n",
    "    \n",
    "    def predictions(self, prelogits):\n",
    "        return self.softmax(prelogits)\n",
    "\n",
    "class BilinearPredictor(torch.nn.Module):\n",
    "    def __init__(self, dim_x, dim_a):\n",
    "        super().__init__()\n",
    "        if dim_x * dim_a > 0:\n",
    "            self.bilinear = torch.nn.Bilinear(in1_features=dim_x, in2_features=dim_a, out_features=1)\n",
    "        else:\n",
    "            self.bilinear = None\n",
    "            self.b = torch.nn.Parameter(torch.zeros(1))\n",
    "        self.sigmoid = torch.nn.Sigmoid()\n",
    "        \n",
    "    def prelogits(self, Xs, As):\n",
    "        if self.bilinear is not None:\n",
    "            return self.bilinear(Xs, As).squeeze(1)\n",
    "        else:\n",
    "            return self.b.expand(Xs.shape[0])\n",
    "    \n",
    "    def prediction(self, prelogits):\n",
    "        return self.sigmoid(prelogits)\n",
    "\n",
    "class Policy(torch.nn.Module):\n",
    "    def __init__(self):\n",
    "        super().__init__()\n",
    "        pass\n",
    "    \n",
    "    def sample(self, Fs):\n",
    "        raise NotImplementedError\n",
    "        \n",
    "    def update(self, dt):\n",
    "        raise NotImplementedError\n",
    "        \n",
    "class EpsilonGreedy(Policy):\n",
    "    def __init__(self, epsilon, epsilon_t0):\n",
    "        super().__init__()\n",
    "        self.epsilon = epsilon\n",
    "        self.epsilon_t0 = epsilon_t0\n",
    "        self.t = epsilon_t0\n",
    "        \n",
    "    def myeps(self):\n",
    "        return self.epsilon * (self.epsilon_t0 / self.t)**(1/3)\n",
    "        \n",
    "    def sample(self, Fs, As):\n",
    "        myeps = self.myeps()\n",
    "        \n",
    "        unif = torch.rand(size=(Fs.shape[0], 1))\n",
    "        should_explore = (unif < myeps).long()\n",
    "        \n",
    "        explore = torch.randint(low=0, high=Fs.shape[1], size=(Fs.shape[0], 1))\n",
    "        exploit = torch.max(Fs, dim=1, keepdim=True).indices\n",
    "        \n",
    "        aindex = exploit + should_explore * (explore - exploit)\n",
    "        action = torch.gather(input=As, dim=1, index=aindex.unsqueeze(2).expand(-1, -1, As.shape[2])).squeeze(1)\n",
    "        \n",
    "        isgreedy = (aindex == exploit).long()\n",
    "        \n",
    "        paction = myeps * torch.ones(size=(Fs.shape[0],1))/Fs.shape[1] + isgreedy * (1 - myeps)\n",
    "        return action, paction, aindex\n",
    "    \n",
    "    def update(self, dt):\n",
    "        self.t += dt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "ab5b3f10",
   "metadata": {},
   "outputs": [],
   "source": [
    "class NuThreeStateSimulator(torch.utils.data.Dataset):\n",
    "    def __init__(self, *, T, data, seed):\n",
    "        import random\n",
    "        import pandas as pd\n",
    "        \n",
    "        super().__init__()\n",
    "\n",
    "        self.data = data\n",
    "        self.random = random.Random(seed)\n",
    "        self.nusers = len(pd.unique(data['news_id']))\n",
    "        self.nactionsets = len(pd.unique(data['set_id']))\n",
    "        self.actdim = len(data['action_feat'].iloc[0])\n",
    "        self.nactions = 15\n",
    "        self.probs = {'fox':{0:[0.9,0,0,0,0,0,0,0.1,0],\n",
    "                             1:[0,0.744328,0,0.238383,0.017289,0,0,0,0],\n",
    "                             -1:[0,0,0.826858,0,0,0.023984,0.040857,0,0.108301]},\n",
    "                      'huffpost':{0:[0.9,0,0.016804,0.080953,0,0,0.002243,0,0],\n",
    "                                  1:[0,0.977411,0,0,0.022589,0,0,0,0],\n",
    "                                  -1:[0,0,0,0,0,0.236086,0,0.347774,0.416139]},\n",
    "                      'time':{0:[0.9,0,0,0,0,0.038336,0.061664,0,0],\n",
    "                              1:[0,0.723603,0.060473,0.186107,0.029816,0,0,0,0],\n",
    "                              -1:[0,0,0,0,0,0,0,0.633414,0.366586]}}\n",
    "    \n",
    "        self.feedbacks = ['none','like','comment','share','love','wow','haha','sad','angry']\n",
    "        self.cmap = {0:'fox',1:'huffpost',2:'time'}\n",
    "        \n",
    "        self.contexts = torch.Tensor([ self.random.randint(0, self.nusers-1) for _ in range(T) ]).long()\n",
    "        self.actionsets = torch.Tensor([ self.random.randint(0, self.nactionsets-1) for _ in range(T) ]).long()\n",
    "        self.random = random.Random(1664525*seed + 1013904223)\n",
    "\n",
    "    def __len__(self):\n",
    "        return len(self.contexts)\n",
    "    \n",
    "    def __getitem__(self, idx):\n",
    "        context = torch.nn.functional.one_hot(self.contexts[idx], num_classes=self.nusers).float()\n",
    "        data_set = self.data[(self.data['news_id']==self.cmap[self.contexts[idx].item()])& (self.data['set_id']==self.actionsets[idx].item())]\n",
    "        action_set = data_set['action_feat']\n",
    "        action_set.reset_index(drop=True, inplace=True)\n",
    "        actions = torch.zeros(self.nactions,self.actdim)\n",
    "        for count, value in enumerate(action_set):\n",
    "            actions[count,:] = torch.Tensor(value).float()\n",
    "        return context, actions\n",
    "\n",
    "    def first_nonzero(self, x, axis=0):\n",
    "        nonz = (x != 0)\n",
    "        return ((nonz.cumsum(axis) == 1) & nonz).max(axis, keepdim=True)\n",
    "    \n",
    "    def get_one_hot_id(self, onehot):\n",
    "        return self.first_nonzero(onehot, axis=-1)[1].squeeze().item()\n",
    "    \n",
    "    def sample_feedback(self, Xs, Rs):\n",
    "        results = []\n",
    "        for context, reward in zip(Xs, Rs):\n",
    "            probs = self.probs[self.cmap[self.get_one_hot_id(context)]][reward.item()]\n",
    "        \n",
    "            feedback = self.random.choices(list(range(len(self.feedbacks))), weights=probs)[0]\n",
    "\n",
    "            results.append(torch.nn.functional.one_hot(torch.LongTensor([feedback]), num_classes = len(self.feedbacks)).float())\n",
    "        return torch.cat(results, dim=0)\n",
    "    \n",
    "    def is_definitely_negative(self, Xs, Ys):\n",
    "        return self.first_nonzero(Ys, axis=-1)[1].squeeze(1) == 8\n",
    "\n",
    "    def trueReward(self, Xs, Ainds):\n",
    "        reward_all = torch.Tensor([ (aind.item() == 0)-(aind.item() == self.nactions-1)for aind in Ainds ])\n",
    "        reward_fox = []\n",
    "        reward_huffpost = []\n",
    "        reward_time = []\n",
    "        for x, aind in zip(Xs,Ainds):\n",
    "            if self.get_one_hot_id(x) == 0:\n",
    "                reward_fox.append((aind.item() == 0)-(aind.item()== self.nactions-1))\n",
    "            elif self.get_one_hot_id(x) == 1:\n",
    "                reward_huffpost.append((aind.item() == 0)-(aind.item()== self.nactions-1))\n",
    "            else:\n",
    "                reward_time.append((aind.item() == 0)-(aind.item()== self.nactions-1))            \n",
    "        return reward_all, reward_fox, reward_huffpost, reward_time"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "derived-partnership",
   "metadata": {},
   "source": [
    "# Epsilon-Greedy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "located-duplicate",
   "metadata": {
    "code_folding": [
     0,
     4,
     139
    ]
   },
   "outputs": [],
   "source": [
    "def noglobalz_iglp3(*, seed, T, data):\n",
    "    from math import sqrt\n",
    "    import itertools\n",
    "    \n",
    "    nusers = len(pd.unique(data['news_id']))\n",
    "    nactions = len(data['action_feat'].iloc[0])\n",
    "    numr = 3\n",
    "    batch_size = 100\n",
    "    lr = 0.001\n",
    "    epsilon = 1\n",
    "    epsilon_t0 = 100\n",
    "     \n",
    "    true_rewards = []\n",
    "    positive_rewards_counter = []\n",
    "    negative_rewards_counter = []\n",
    "    neutral_rewards_counter = []\n",
    "    \n",
    "    true_rewards_fox = []\n",
    "    true_rewards_huffpost = []\n",
    "    true_rewards_time = []\n",
    "\n",
    "    sim = NuThreeStateSimulator(T=T,data=data,seed=seed); seed += 1\n",
    "    \n",
    "    torch.manual_seed(seed); seed += 1\n",
    "    loader = torch.utils.data.DataLoader(sim, batch_size=batch_size, shuffle=False)\n",
    "\n",
    "    policy = EpsilonGreedy(epsilon, epsilon_t0)\n",
    "\n",
    "    ik = IK(dim_x=nusers, dim_a=nactions, dim_y=len(sim.feedbacks))\n",
    "    negpredictor = BilinearPredictor(dim_x=nusers, dim_a=nactions)\n",
    "    constpredictor = BilinearPredictor(dim_x=0, dim_a=0)\n",
    "    rewardpredictor = BilinearPredictor(dim_x=nusers, dim_a=nactions)\n",
    "\n",
    "    actionloss = torch.nn.CrossEntropyLoss(reduction='none')\n",
    "    actionoptimizer = torch.optim.Adam(ik.parameters(), lr=lr*sqrt(batch_size))\n",
    "    negloss = torch.nn.BCEWithLogitsLoss(reduction='none')\n",
    "    negoptimizer = torch.optim.Adam(itertools.chain(negpredictor.parameters(), constpredictor.parameters()), lr=lr*sqrt(batch_size))\n",
    "    regloss = torch.nn.MSELoss(reduction='none')\n",
    "    regoptimizer = torch.optim.Adam(rewardpredictor.parameters(), lr=lr*sqrt(batch_size))\n",
    "\n",
    "    print('{:<5s}\\t{:<8s} {:<8s}\\t{:<8s} {:<8s}\\t{:<8s} {:<8s}\\t{:<8s} {:<8s}\\t{:<8s} {:<8s}\\t{:<8s} {:<8s}\\t{:<8s} {:<8s}\\t{:<8s}'.format(\n",
    "        'bno',  'loss', 'since', 'acc', 'since', 'neg', 'since', 'const', 'since', 'reg', 'since', 'pol', 'since', 'greedy', 'since', 'eps'), flush=True) \n",
    "    avloss, avacc, avnegloss, avconstloss, avregloss, avpol, avgreedy = [ EasyAcc() for _ in range(7) ]\n",
    "    avlosssincelast, avaccsincelast, avneglosssincelast, avconstlosssincelast, avreglosssincelast, avpolsincelast, avgreedysincelast = [ EasyAcc() for _ in range(7) ]\n",
    "    \n",
    "    for bno, minibatch in enumerate(loader):\n",
    "        Xs, As = minibatch\n",
    "\n",
    "        with torch.no_grad():\n",
    "            rhat = rewardpredictor.prelogits(Xs.unsqueeze(1).expand(-1, As.shape[1], -1), As).squeeze(2)\n",
    "            action, paction, aindex = policy.sample(rhat, As)\n",
    "            truerewards, foxrewards, huffpostrewards, timerewards = sim.trueReward(Xs,aindex)\n",
    "            avpol += torch.mean(truerewards).item()\n",
    "            avpolsincelast += torch.mean(truerewards).item()\n",
    "            \n",
    "            true_rewards_fox.append(foxrewards)\n",
    "            true_rewards_huffpost.append(huffpostrewards)\n",
    "            true_rewards_time.append(timerewards)\n",
    "            \n",
    "            true_rewards.append(torch.mean(truerewards).item())\n",
    "\n",
    "            greedy = torch.max(rhat, dim=1, keepdim=True).indices\n",
    "            greedyaction = torch.gather(input=As, dim=1, index=greedy.unsqueeze(2).expand(-1, -1, As.shape[2])).squeeze(1)\n",
    "            greedyrewards,_,_,_ = sim.trueReward(Xs,greedy)\n",
    "            avgreedy += torch.mean(greedyrewards).item()\n",
    "            avgreedysincelast += torch.mean(greedyrewards).item()\n",
    "\n",
    "            Ys = sim.sample_feedback(Xs, truerewards)\n",
    "            \n",
    "        actionoptimizer.zero_grad()\n",
    "        prepapred = ik.prelogits(Xs, As, Ys)\n",
    "        \n",
    "        rawactionloss = actionloss(prepapred, aindex.squeeze(1)).unsqueeze(1)\n",
    "        actioniw = 1/(As.shape[1]*paction)\n",
    "        batchactionloss = torch.sum(torch.mul(actioniw, rawactionloss)) / torch.sum(actioniw)\n",
    "        batchactionloss.backward() #learn\n",
    "        actionoptimizer.step()\n",
    "                \n",
    "        with torch.no_grad():\n",
    "            papred = torch.gather(input=ik.predictions(prepapred), index=aindex, dim=1).squeeze()\n",
    "            \n",
    "            isExtreme = papred > (2 / As.shape[1])\n",
    "            \n",
    "        if torch.any(isExtreme):\n",
    "            extremeXs = torch.atleast_2d(Xs[isExtreme])\n",
    "            extremeAs = torch.atleast_2d(action[isExtreme])\n",
    "            extremeYs = torch.atleast_2d(Ys[isExtreme])\n",
    "            extremepaction = torch.atleast_2d(paction[isExtreme])\n",
    "            with torch.no_grad():\n",
    "                isNeg = sim.is_definitely_negative(extremeXs, extremeYs).float()\n",
    "                \n",
    "            negoptimizer.zero_grad()\n",
    "            negpred = negpredictor.prelogits(extremeXs, extremeAs)\n",
    "            constpred = constpredictor.prelogits(torch.empty(extremeXs.shape[0], 0), None)\n",
    "            extremeactioniw = 1/(As.shape[1]*extremepaction)\n",
    "            batchnegloss = torch.sum(torch.mul(extremeactioniw, negloss(negpred, isNeg).unsqueeze(1))) / torch.sum(extremeactioniw)\n",
    "            batchconstloss = torch.sum(torch.mul(extremeactioniw, negloss(constpred, isNeg).unsqueeze(1))) / torch.sum(extremeactioniw)\n",
    "            (batchnegloss + batchconstloss).backward()\n",
    "            negoptimizer.step()\n",
    "            \n",
    "            with torch.no_grad():\n",
    "                fakereward = torch.zeros(Xs.shape[0])\n",
    "                fakereward[isExtreme] = 1 - 2 * (negpred > constpred).float()\n",
    "\n",
    "                states, counts = fakereward.unique(return_counts=True)\n",
    "                \n",
    "                if -1 not in states:\n",
    "                    negative_rewards_counter.append(0)\n",
    "                \n",
    "                if 1 not in states:\n",
    "                    positive_rewards_counter.append(0)\n",
    "    \n",
    "                for latent_reward, count in zip(states, counts):\n",
    "                    if latent_reward.item() == -1:\n",
    "                        negative_rewards_counter.append(count.item())\n",
    "                    elif latent_reward.item() == 1:\n",
    "                        positive_rewards_counter.append(count.item())\n",
    "                    else:\n",
    "                        neutral_rewards_counter.append(count.item())\n",
    "            \n",
    "            regoptimizer.zero_grad()\n",
    "            rewardpred = rewardpredictor.prelogits(Xs, action)\n",
    "            batchregloss = torch.sum(torch.mul(actioniw, regloss(rewardpred, fakereward).unsqueeze(1))) / torch.sum(actioniw)\n",
    "            batchregloss.backward()\n",
    "            regoptimizer.step()\n",
    "            \n",
    "            with torch.no_grad():\n",
    "                policy.update(Xs.shape[0])\n",
    "        else:\n",
    "            batchnegloss = None\n",
    "\n",
    "        with torch.no_grad():\n",
    "            avloss += batchactionloss.item()\n",
    "            avlosssincelast += batchactionloss.item()\n",
    "            if batchnegloss is not None:\n",
    "                avnegloss += batchnegloss.item()\n",
    "                avneglosssincelast += batchnegloss.item()\n",
    "                avconstloss += batchconstloss.item()\n",
    "                avconstlosssincelast += batchconstloss.item()\n",
    "                avregloss += batchregloss.item()\n",
    "                avreglosssincelast += batchregloss.item()\n",
    "                \n",
    "            amaxpred = torch.max(prepapred, dim=1, keepdim=True)\n",
    "            avacc += torch.mean(torch.mul(actioniw, (aindex == amaxpred.indices).float())).item()\n",
    "            avaccsincelast += torch.mean(torch.mul(actioniw, (aindex == amaxpred.indices).float())).item()\n",
    "\n",
    "            if bno & (bno - 1) == 0:\n",
    "                print('{:<5d}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f}'.format( \n",
    "                    bno,  \n",
    "                    avloss.mean(), avlosssincelast.mean(),\n",
    "                    avacc.mean(), avaccsincelast.mean(),\n",
    "                    avnegloss.mean(), avneglosssincelast.mean(),\n",
    "                    avconstloss.mean(), avconstlosssincelast.mean(),\n",
    "                    avregloss.mean(), avreglosssincelast.mean(),\n",
    "                    avpol.mean(), avpolsincelast.mean(),\n",
    "                    avgreedy.mean(), avgreedysincelast.mean(),\n",
    "                    policy.myeps(),\n",
    "                ), \n",
    "                      flush=True)\n",
    "                avlosssincelast, avaccsincelast, avneglosssincelast, avconstlosssincelast, avreglosssincelast, avpolsincelast, avgreedysincelast = [ EasyAcc() for _ in range(7) ]\n",
    "\n",
    "    print('{:<5d}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f}'.format( \n",
    "        bno,  \n",
    "        avloss.mean(), avlosssincelast.mean(),\n",
    "        avacc.mean(), avaccsincelast.mean(),\n",
    "        avnegloss.mean(), avneglosssincelast.mean(),\n",
    "        avconstloss.mean(), avconstlosssincelast.mean(),\n",
    "        avregloss.mean(), avreglosssincelast.mean(),\n",
    "        avpol.mean(), avpolsincelast.mean(),\n",
    "        avgreedy.mean(), avgreedysincelast.mean(),\n",
    "        policy.myeps()\n",
    "    ), flush=True)\n",
    "    \n",
    "    return true_rewards, positive_rewards_counter, negative_rewards_counter, neutral_rewards_counter, true_rewards_fox, true_rewards_huffpost, true_rewards_time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "66d48d83",
   "metadata": {
    "code_folding": [
     0,
     4,
     139
    ]
   },
   "outputs": [],
   "source": [
    "def noglobalz_cb(*, seed, T, data, cb_reward_mapping):\n",
    "    from math import sqrt\n",
    "    import itertools\n",
    "    \n",
    "    \n",
    "    nusers = len(pd.unique(data['news_id']))\n",
    "    nactions = len(data['action_feat'].iloc[0])\n",
    "    numr = 3\n",
    "    batch_size = 100\n",
    "    lr = 0.001\n",
    "    epsilon = 1\n",
    "    epsilon_t0 = 100\n",
    "        \n",
    "    true_rewards = []\n",
    "    \n",
    "    true_rewards_fox = []\n",
    "    true_rewards_huffpost = []\n",
    "    true_rewards_time = []\n",
    "    \n",
    "    positive_rewards_counter = []\n",
    "    negative_rewards_counter = []\n",
    "    neutral_rewards_counter = []\n",
    "\n",
    "    sim = NuThreeStateSimulator(T=T,data=data,seed=seed); seed += 1\n",
    "    \n",
    "    torch.manual_seed(seed); seed += 1\n",
    "    loader = torch.utils.data.DataLoader(sim, batch_size=batch_size, shuffle=False)\n",
    "\n",
    "    policy = EpsilonGreedy(epsilon, epsilon_t0)\n",
    "\n",
    "    ik = IK(dim_x=nusers, dim_a=nactions, dim_y=len(sim.feedbacks))\n",
    "    negpredictor = BilinearPredictor(dim_x=nusers, dim_a=nactions)\n",
    "    constpredictor = BilinearPredictor(dim_x=0, dim_a=0)\n",
    "    rewardpredictor = BilinearPredictor(dim_x=nusers, dim_a=nactions)\n",
    "\n",
    "    actionloss = torch.nn.CrossEntropyLoss(reduction='none')\n",
    "    actionoptimizer = torch.optim.Adam(ik.parameters(), lr=lr*sqrt(batch_size))\n",
    "    negloss = torch.nn.BCEWithLogitsLoss(reduction='none')\n",
    "    negoptimizer = torch.optim.Adam(itertools.chain(negpredictor.parameters(), constpredictor.parameters()), lr=lr*sqrt(batch_size))\n",
    "    regloss = torch.nn.BCEWithLogitsLoss(reduction='none')\n",
    "    regoptimizer = torch.optim.Adam(rewardpredictor.parameters(), lr=lr*sqrt(batch_size))\n",
    "\n",
    "    print('{:<5s}\\t{:<8s} {:<8s}\\t{:<8s} {:<8s}\\t{:<8s} {:<8s}\\t{:<8s} {:<8s}\\t{:<8s} {:<8s}\\t{:<8s} {:<8s}\\t{:<8s} {:<8s}\\t{:<8s}'.format(\n",
    "        'bno',  'loss', 'since', 'acc', 'since', 'neg', 'since', 'const', 'since', 'reg', 'since', 'pol', 'since', 'greedy', 'since', 'eps'), flush=True) \n",
    "    avloss, avacc, avnegloss, avconstloss, avregloss, avpol, avgreedy = [ EasyAcc() for _ in range(7) ]\n",
    "    avlosssincelast, avaccsincelast, avneglosssincelast, avconstlosssincelast, avreglosssincelast, avpolsincelast, avgreedysincelast = [ EasyAcc() for _ in range(7) ]\n",
    "    \n",
    "    for bno, minibatch in enumerate(loader):\n",
    "        Xs, As = minibatch\n",
    "\n",
    "        with torch.no_grad():\n",
    "            rhat = rewardpredictor.prelogits(Xs.unsqueeze(1).expand(-1, As.shape[1], -1), As).squeeze(2)\n",
    "            action, paction, aindex = policy.sample(rhat, As)\n",
    "            truerewards, foxrewards, huffpostrewards, timerewards = sim.trueReward(Xs,aindex)\n",
    "            avpol += torch.mean(truerewards).item()\n",
    "            avpolsincelast += torch.mean(truerewards).item()\n",
    "            \n",
    "            true_rewards_fox.append(foxrewards)\n",
    "            true_rewards_huffpost.append(huffpostrewards)\n",
    "            true_rewards_time.append(timerewards)\n",
    "            \n",
    "            true_rewards.append(torch.mean(truerewards).item())\n",
    "\n",
    "            greedy = torch.max(rhat, dim=1, keepdim=True).indices\n",
    "            greedyaction = torch.gather(input=As, dim=1, index=greedy.unsqueeze(2).expand(-1, -1, As.shape[2])).squeeze(1)\n",
    "            greedyrewards,_,_,_ = sim.trueReward(Xs,greedy)\n",
    "            avgreedy += torch.mean(greedyrewards).item()\n",
    "            avgreedysincelast += torch.mean(greedyrewards).item()\n",
    "\n",
    "            Ys = sim.sample_feedback(Xs, truerewards)\n",
    "            \n",
    "            # fakereward based on the feedback type\n",
    "            fakereward = torch.zeros(Xs.shape[0])\n",
    "            for jj in range(len(sim.feedbacks)):\n",
    "                fakereward += cb_reward_mapping[sim.feedbacks[jj]]*torch.Tensor(Ys[:,jj])\n",
    "\n",
    "            states, counts = greedyrewards.unique(return_counts=True)\n",
    "            \n",
    "            if -1 not in states:\n",
    "                negative_rewards_counter.append(0)\n",
    "            \n",
    "            if 1 not in states:\n",
    "                positive_rewards_counter.append(0)\n",
    "\n",
    "            for latent_reward, count in zip(states, counts):\n",
    "                if latent_reward.item() == -1:\n",
    "                    negative_rewards_counter.append(count.item())\n",
    "                elif latent_reward.item() == 1:\n",
    "                    positive_rewards_counter.append(count.item())\n",
    "                else:\n",
    "                    neutral_rewards_counter.append(count.item())\n",
    "            \n",
    "        regoptimizer.zero_grad()\n",
    "        actioniw = 1/(As.shape[1]*paction)\n",
    "        rewardpred = rewardpredictor.prelogits(Xs, action)\n",
    "        batchregloss = torch.sum(torch.mul(actioniw, regloss(rewardpred, fakereward).unsqueeze(1))) / torch.sum(actioniw)\n",
    "        batchregloss.backward()\n",
    "        regoptimizer.step()\n",
    "        \n",
    "        with torch.no_grad():\n",
    "            policy.update(Xs.shape[0])\n",
    "\n",
    "        with torch.no_grad():\n",
    "\n",
    "            if bno & (bno - 1) == 0:\n",
    "                print('{:<5d}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f}'.format( \n",
    "                    bno,  \n",
    "                    avloss.mean(), avlosssincelast.mean(),\n",
    "                    avacc.mean(), avaccsincelast.mean(),\n",
    "                    avnegloss.mean(), avneglosssincelast.mean(),\n",
    "                    avconstloss.mean(), avconstlosssincelast.mean(),\n",
    "                    avregloss.mean(), avreglosssincelast.mean(),\n",
    "                    avpol.mean(), avpolsincelast.mean(),\n",
    "                    avgreedy.mean(), avgreedysincelast.mean(),\n",
    "                    policy.myeps(),\n",
    "                ), \n",
    "                      flush=True)\n",
    "                avlosssincelast, avaccsincelast, avneglosssincelast, avconstlosssincelast, avreglosssincelast, avpolsincelast, avgreedysincelast = [ EasyAcc() for _ in range(7) ]\n",
    "\n",
    "    print('{:<5d}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f} {:<8.5f}\\t{:<8.5f}'.format( \n",
    "        bno,  \n",
    "        avloss.mean(), avlosssincelast.mean(),\n",
    "        avacc.mean(), avaccsincelast.mean(),\n",
    "        avnegloss.mean(), avneglosssincelast.mean(),\n",
    "        avconstloss.mean(), avconstlosssincelast.mean(),\n",
    "        avregloss.mean(), avreglosssincelast.mean(),\n",
    "        avpol.mean(), avpolsincelast.mean(),\n",
    "        avgreedy.mean(), avgreedysincelast.mean(),\n",
    "        policy.myeps()\n",
    "    ), flush=True)\n",
    "    \n",
    "    return true_rewards, positive_rewards_counter, negative_rewards_counter, neutral_rewards_counter, true_rewards_fox, true_rewards_huffpost, true_rewards_time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "825d1f94",
   "metadata": {},
   "outputs": [],
   "source": [
    "def separate_rewards(rewards):\n",
    "    neut, pos, neg = [], [], []\n",
    "\n",
    "    for item in rewards:\n",
    "        neut.append(sum([i==0 for i in item])/len(item))\n",
    "        pos.append(sum([i==1 for i in item])/len(item))\n",
    "        neg.append(sum([i==-1 for i in item])/len(item))\n",
    "    \n",
    "    return neut, pos, neg\n",
    "    \n",
    "\n",
    "def run_experiment(*, seed, T, data, cb_reward_mapping_1, cb_reward_mapping_2):\n",
    "    _, _, _, _, igl_fox, igl_hp, igl_time = noglobalz_iglp3(seed=seed, T=T, data=data)\n",
    "    _, _, _, _, cb1_fox, cb1_hp, cb1_time = noglobalz_cb(seed=seed, T=T, data=data, cb_reward_mapping = cb_reward_mapping_1)\n",
    "    _, _, _, _, cb2_fox, cb2_hp, cb2_time = noglobalz_cb(seed=seed, T=T, data=data, cb_reward_mapping = cb_reward_mapping_2)\n",
    "\n",
    "    return igl_fox, igl_hp, igl_time, cb1_fox, cb1_hp, cb1_time, cb2_fox, cb2_hp, cb2_time\n",
    "\n",
    "def gen_plots(igl_fox, igl_hp, igl_time, cb1_fox, cb1_hp, cb1_time, cb2_fox, cb2_hp, cb2_time):\n",
    "    from matplotlib import pyplot as plt\n",
    "    from scipy.stats import sem\n",
    "    \n",
    "    import numpy as np\n",
    "\n",
    "    r1_color = (120/255, 94/255, 240/255)\n",
    "    r_1_color = (220/255, 38/255, 127/255)\n",
    "    r0_color = (255/255, 176/255, 0/255)\n",
    "    \n",
    "    _, igl_f_p, igl_f_n = separate_rewards(igl_fox)\n",
    "    _, igl_h_p, igl_h_n = separate_rewards(igl_hp)\n",
    "    _, igl_t_p, igl_t_n = separate_rewards(igl_time)\n",
    "    \n",
    "    _, cb1_f_p, cb1_f_n = separate_rewards(cb1_fox)\n",
    "    _, cb1_h_p, cb1_h_n = separate_rewards(cb1_hp)\n",
    "    _, cb1_t_p, cb1_t_n = separate_rewards(cb1_time)\n",
    "    \n",
    "    _, cb2_f_p, cb2_f_n = separate_rewards(cb2_fox)\n",
    "    _, cb2_h_p, cb2_h_n = separate_rewards(cb2_hp)\n",
    "    _, cb2_t_p, cb2_t_n = separate_rewards(cb2_time)\n",
    "    \n",
    "    \n",
    "    igl_means_p = [np.mean(igl_f_p),np.mean(igl_h_p),np.mean(igl_t_p)]\n",
    "    igl_err_p = [sem(igl_f_p),sem(igl_h_p),sem(igl_t_p)]\n",
    "    print('sem of averages for igl positive reward are'+str(igl_err_p))\n",
    "    cb1_means_p = [np.mean(cb1_f_p),np.mean(cb1_h_p),np.mean(cb1_t_p)]\n",
    "    cb1_err_p = [sem(cb1_f_p),sem(cb1_h_p),sem(cb1_t_p)]\n",
    "    print('sem of averages for cb1 positive reward are'+str(cb1_err_p))\n",
    "    cb2_means_p = [np.mean(cb2_f_p),np.mean(cb2_h_p),np.mean(cb2_t_p)]\n",
    "    cb2_err_p = [sem(cb2_f_p),sem(cb2_h_p),sem(cb2_t_p)]\n",
    "    print('sem of averages for cb2 positive reward are'+str(cb2_err_p))\n",
    "    \n",
    "    igl_means_n = [np.mean(igl_f_n),np.mean(igl_h_n),np.mean(igl_t_n)]\n",
    "    igl_err_n = [sem(igl_f_n),sem(igl_h_n),sem(igl_t_n)]\n",
    "    print('sem of averages for igl negative reward are'+str(igl_err_n))\n",
    "    cb1_means_n = [np.mean(cb1_f_n),np.mean(cb1_h_n),np.mean(cb1_t_n)]\n",
    "    cb1_err_n = [sem(cb1_f_n),sem(cb1_h_n),sem(cb1_t_n)]\n",
    "    print('sem of averages for cb1 negative reward are'+str(cb1_err_n))\n",
    "    cb2_means_n = [np.mean(cb2_f_n),np.mean(cb2_h_n),np.mean(cb2_t_n)]\n",
    "    cb2_err_n = [sem(cb2_f_n),sem(cb2_h_n),sem(cb2_t_n)]\n",
    "    print('sem of averages for cb2 positive reward are'+str(cb2_err_n))\n",
    "    \n",
    "    N = 3\n",
    "    ind_x = np.arange(N) \n",
    "    width = 0.25\n",
    "\n",
    "    plt.figure(figsize=(6, 3.5))\n",
    "    plt.rcParams.update({'font.size': 14})\n",
    "    bar1 = plt.bar(ind_x, igl_means_p, width, color = r1_color, label='IGL-P(3)')\n",
    "    bar2 = plt.bar(ind_x+width, cb1_means_p, width, color=r0_color, label='CB-emoji')\n",
    "    bar3 = plt.bar(ind_x+width*2, cb2_means_p, width, color = r_1_color, label='CB-comment')\n",
    "    plt.ylabel('positive rewards')\n",
    "    plt.xticks(ind_x+width,['Fox News', 'The Huffington Post', 'TIME'])\n",
    "    plt.legend(bbox_to_anchor=(0, 1.02, 1, 0.2),loc=\"lower left\",ncol=3,prop={'size': 11.5})\n",
    "    \n",
    "    plt.figure(figsize=(6, 3.5))\n",
    "    plt.rcParams.update({'font.size': 14})\n",
    "    bar1 = plt.bar(ind_x, igl_means_n, width, color = r1_color, label='IGL-P(3)')\n",
    "    bar2 = plt.bar(ind_x+width, cb1_means_n, width, color=r0_color, label='CB-emoji')\n",
    "    bar3 = plt.bar(ind_x+width*2, cb2_means_n, width, color = r_1_color, label='CB-comment')\n",
    "    plt.ylabel('negative rewards')\n",
    "    plt.xticks(ind_x+width,['Fox News', 'The Huffington Post', 'TIME'])\n",
    "    plt.legend(bbox_to_anchor=(0, 1.02, 1, 0.2),loc=\"lower left\",ncol=3,prop={'size': 11.5})"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "0ad0b6ff",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "bno  \tloss     since   \tacc      since   \tneg      since   \tconst    since   \treg      since   \tpol      since   \tgreedy   since   \teps     \n",
      "0    \t2.72008  2.72008 \t0.05000  0.05000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.01000  0.01000 \t-0.01000 -0.01000\t1.00000 \n",
      "1    \t2.69531  2.67053 \t0.06500  0.08000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.04000 -0.09000\t-0.03000 -0.05000\t1.00000 \n",
      "2    \t2.69906  2.70658 \t0.06667  0.07000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.03667 -0.03000\t-0.02667 -0.02000\t1.00000 \n",
      "4    \t2.70011  2.70168 \t0.07969  0.09923 \t0.65186  0.65186 \t0.69065  0.69065 \t3.12523  3.12523 \t-0.03400 -0.03000\t-0.01200 0.01000 \t0.69336 \n",
      "8    \t2.70158  2.70341 \t0.08427  0.08998 \t0.88543  1.00222 \t0.68376  0.68032 \t2.81409  2.65853 \t-0.01778 0.00250 \t-0.01556 -0.02000\t0.52276 \n",
      "16   \t2.66601  2.62599 \t0.10472  0.12774 \t0.70911  0.57686 \t0.66758  0.65544 \t2.24059  1.81047 \t-0.02118 -0.02500\t-0.02529 -0.03625\t0.40548 \n",
      "32   \t2.61715  2.56524 \t0.12491  0.14635 \t0.75203  0.78959 \t0.65139  0.63723 \t1.67713  1.18410 \t-0.02515 -0.02937\t-0.03636 -0.04813\t0.31833 \n",
      "64   \t2.58774  2.55742 \t0.13148  0.13827 \t0.60505  0.46726 \t0.60640  0.56421 \t1.15751  0.67037 \t-0.03431 -0.04375\t-0.04477 -0.05344\t0.25132 \n",
      "128  \t2.55977  2.53136 \t0.14876  0.16631 \t0.48573  0.37013 \t0.54360  0.48276 \t0.77171  0.39797 \t-0.04364 -0.05312\t-0.05775 -0.07094\t0.19894 \n",
      "256  \t2.52913  2.49825 \t0.16363  0.17861 \t0.40253  0.32062 \t0.48000  0.41739 \t0.54286  0.31758 \t-0.01685 0.01016 \t-0.02412 0.00977 \t0.15770 \n",
      "512  \t2.49772  2.46619 \t0.18345  0.20335 \t0.33180  0.26163 \t0.38525  0.29125 \t0.42337  0.30482 \t0.02747  0.07195 \t0.02955  0.08344 \t0.12508 \n",
      "1024 \t2.48784  2.47794 \t0.18969  0.19593 \t0.31580  0.29985 \t0.35775  0.33035 \t0.36495  0.30676 \t0.12442  0.22156 \t0.13879  0.24824 \t0.09924 \n",
      "1999 \t2.47974  2.47123 \t0.19504  0.20066 \t0.27811  0.23860 \t0.31854  0.27745 \t0.33049  0.29437 \t0.19287  0.26483 \t0.21291  0.29083 \t0.07940 \n",
      "bno  \tloss     since   \tacc      since   \tneg      since   \tconst    since   \treg      since   \tpol      since   \tgreedy   since   \teps     \n",
      "0    \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.01000  0.01000 \t-0.01000 -0.01000\t0.79370 \n",
      "1    \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.01000 -0.03000\t-0.03000 -0.05000\t0.69336 \n",
      "2    \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.01667 -0.03000\t-0.01667 0.01000 \t0.62996 \n",
      "4    \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.02000 -0.02500\t-0.01800 -0.02000\t0.55032 \n",
      "8    \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.02333 -0.02750\t-0.03667 -0.06000\t0.46416 \n",
      "16   \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.02824 -0.03375\t-0.04824 -0.06125\t0.38157 \n",
      "32   \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.07091 -0.11625\t-0.11000 -0.17563\t0.30868 \n",
      "64   \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.13446 -0.20000\t-0.18723 -0.26688\t0.24745 \n",
      "128  \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.26380 -0.39516\t-0.34194 -0.49906\t0.19740 \n",
      "256  \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.39401 -0.52523\t-0.49140 -0.64203\t0.15708 \n",
      "512  \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.47491 -0.55613\t-0.56862 -0.64613\t0.12484 \n",
      "1024 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.55736 -0.63996\t-0.64240 -0.71633\t0.09915 \n",
      "1999 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.59639 -0.63743\t-0.66947 -0.69793\t0.07936 \n",
      "bno  \tloss     since   \tacc      since   \tneg      since   \tconst    since   \treg      since   \tpol      since   \tgreedy   since   \teps     \n",
      "0    \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.01000  0.01000 \t-0.01000 -0.01000\t0.79370 \n",
      "1    \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.01000 -0.03000\t-0.03000 -0.05000\t0.69336 \n",
      "2    \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.01667 -0.03000\t-0.01667 0.01000 \t0.62996 \n",
      "4    \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.01200 -0.00500\t-0.00400 0.01500 \t0.55032 \n",
      "8    \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.00889 -0.00500\t-0.01222 -0.02250\t0.46416 \n",
      "16   \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.01412 -0.02000\t-0.01706 -0.02250\t0.38157 \n",
      "32   \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.01212 -0.01000\t-0.01333 -0.00938\t0.30868 \n",
      "64   \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.02092 -0.03000\t-0.02477 -0.03656\t0.24745 \n",
      "128  \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.08147 -0.14297\t-0.10225 -0.18094\t0.19740 \n",
      "256  \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.13451 -0.18797\t-0.16191 -0.22203\t0.15708 \n",
      "512  \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.16470 -0.19500\t-0.19513 -0.22848\t0.12484 \n",
      "1024 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.17581 -0.18695\t-0.20229 -0.20947\t0.09915 \n",
      "1999 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t0.00000  0.00000 \t-0.15928 -0.14189\t-0.17994 -0.15645\t0.07936 \n"
     ]
    }
   ],
   "source": [
    "data_some = data[data['set_id']<22] # subsample set of all possible action sets to reduce training time\n",
    "\n",
    "cb_rm1 =  {'none': 0, 'like':0.2, 'comment':0, 'share':0, 'love': 1 ,'wow':1 ,'haha':1, 'sad':1, 'angry':1} \n",
    "cb_rm2 =  {'none': 0, 'like':0, 'comment':1, 'share':0, 'love':0 ,'wow':0 ,'haha':0, 'sad':0, 'angry':0} \n",
    "\n",
    "igl_fox, igl_hp, igl_time, cb1_fox, cb1_hp, cb1_time, cb2_fox, cb2_hp, cb2_time = run_experiment(seed=31, T=200_000, data=data_some, cb_reward_mapping_1=cb_rm1, cb_reward_mapping_2=cb_rm2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "9cfc77fa",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "sem of averages for igl positive reward are[0.0030042166790878607, 0.003506949686628027, 0.0031863003772975825]\n",
      "sem of averages for cb1 positive reward are[0.0014243262627260877, 0.0004655334243035469, 0.000415943837769066]\n",
      "sem of averages for cb2 positive reward are[0.000642949971692472, 0.0010457488586185608, 0.004112688050288712]\n",
      "sem of averages for igl negative reward are[0.0027077535837508066, 0.0016039471780844678, 0.0014159489920677578]\n",
      "sem of averages for cb1 negative reward are[0.0025078019142782265, 0.0032612020041720424, 0.0029660489783103587]\n",
      "sem of averages for cb2 positive reward are[0.004174710238312498, 0.0013101985314351052, 0.0011071811294002317]\n"
     ]
    },
    {
     "data": {
      "image/png": "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kAClUokTJ04gMjISQUFB+OWXXwC8mn+iSlI6duyIWrVqaTVgIiIiqrgkJSZ2dnbw9fUVb5oXHx+PEydOICoqCkqlEhs2bMCGDRugUCjES4qpcHJdwhmQ9zLORERUMWll8qujoyNq164NW1tbWFlZYdOmTbh//742miYiIqJKRHJi8uzZM5w+fRonTpyAUqnEpUuXkJeXB0NDQ7i6uorDOURERESakpSYuLu74/z583j58iX09fXh4uKCBQsWwNPTEy4uLtDT09N2nERERFQJSEpMTp06BYVCAS8vLyxYsABubm5qN/UjIiIikkJSNjF16lQ4Ozvj+PHj8PT0hIWFBT766COsXr0aV69e1XaMREREVElI6jFZvXo1AODRo0dQKpVQKpWIjIzEwYMHoVAoYGFhAQ8PD3h5eWHSpEnajJeIiIgqsBKNv1hYWKB///74/vvvce3aNdy7dw8rVqyAjo4O9u7di8mTJ2srTiIiIqoESny5cHp6uthrolQqcevWLQiCAACoU6dOiQMkIiKiykNSYrJnzx4xEfnzzz8BAIIgoHbt2hg0aJB4qXDDhg21GiwRERFVbJISkwEDBgAAatSogX79+omJSJMmTbQaHBEREVUukie/enp6wtnZWdvxEBERUSUmKTGZOnWqtuMgIiIiKtnk17S0NOzZswc3b97E06dPERQUBAC4f/8+EhIS4OzsjKpVq2olUCIiIqr4JCcmP/zwA2bOnImcnBwAgEKhEBOT9PR0tG/fHj/99BPGjx+vnUiJiIiowpO0jklYWBgmT54MZ2dnHDhwoMAiak2bNkXz5s2xb98+bcRIRERElYSkHpMVK1agXr16UCqVMDIywqVLlwrUcXZ2xsmTJ0scIBEREVUeknpMrly5gp49e8LIyKjIOnXq1MG9e/ckB0ZERESVj6TEJD8/H3p6em+tk56eDgMDA0lBERERUeUkKTFxcnJ66zBNbm4uoqOjuc4JERERFYukxGT48OG4fPkyFi1aVGBfXl4eZs2ahb///hujRo0qcYBERERUeUia/DplyhSEhYVh8eLF2LJlCwwNDQEAgwYNwsWLF5GYmIiuXbvCz89Pq8ESERFRxSapx0RPTw8RERGYO3cuHj58iGvXrkEQBOzatQuPHj3CnDlzcODAASgUCm3HS0RERBWY5AXW9PX1sXTpUixZsgSxsbF49OgR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\n",
      "text/plain": [
       "<Figure size 600x350 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 600x350 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "gen_plots(igl_fox, igl_hp, igl_time, cb1_fox, cb1_hp, cb1_time, cb2_fox, cb2_hp, cb2_time)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "aafb9547",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "igl",
   "language": "python",
   "name": "igl"
  },
  "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.10.4"
  },
  "vscode": {
   "interpreter": {
    "hash": "8e044b4ce504636100ab2819b5683209b72d0d015e5b8f9f916ab95125ee1e19"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
