{
  "nbformat": 4,
  "nbformat_minor": 0,
  "metadata": {
    "colab": {
      "name": "synethic2.ipynb",
      "provenance": [],
      "collapsed_sections": []
    },
    "kernelspec": {
      "display_name": "Python 3",
      "name": "python3"
    }
  },
  "cells": [
    {
      "cell_type": "code",
      "metadata": {
        "id": "GJqwQsjvHRjb"
      },
      "source": [
        "import numpy as np \n",
        "from time import time\n",
        "import math\n",
        "import matplotlib.pyplot as plt \n",
        "from sklearn.linear_model import LogisticRegression as LogReg\n",
        "import matplotlib.pyplot as plt \n",
        "#plt.style.use('seaborn-whitegrid')\n",
        "plt.style.use('ggplot')"
      ],
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "metadata": {
        "id": "TtGRGVkPivWb"
      },
      "source": [
        "import torch\n",
        "from torch.distributions import Uniform, Categorical, MultivariateNormal as mvnorm\n",
        "class Oracle_2():\n",
        "    def __init__(self, mu00, cov00, mu01, cov01, mix00, mix01, unif0, unif1, rot_angle=math.pi/6):\n",
        "        '''\n",
        "        11 is the left dumbbell, 12 the right dumbbell, unif1 is the interval for the \n",
        "        left uniform, unif2 for the right uniform, hard coding p(y)=.5 for everything for now\n",
        "        mu and cov list of 3 tensors each, for the 3 components of the mixture\n",
        "        '''\n",
        "        #default to 30 degrees ccw, pass 0 if you don't want a rotation\n",
        "        angle = torch.tensor(rot_angle)\n",
        "        self.rotation = torch.tensor([[torch.cos(angle), -torch.sin(angle)],\n",
        "                                        [torch.sin(angle), torch.cos(angle)]])\n",
        "        self.mixture = (Categorical(mix00), Categorical(mix01))\n",
        "        self.gmm = []\n",
        "        dist = []\n",
        "        for m,v in zip(mu00, cov00):\n",
        "            dist.append(mvnorm(m,v))\n",
        "        self.gmm.append(dist)\n",
        "        dist = []\n",
        "        for m,v in zip(mu01, cov01):\n",
        "            dist.append(mvnorm(m,v))\n",
        "        self.gmm.append(dist)\n",
        "        self.unif = (Uniform(unif0[0], unif0[1]), Uniform(unif1[0],unif1[1]))\n",
        "    \n",
        "    def __call__(self, att=0, n_samples=2):\n",
        "        features = []\n",
        "        labels = []\n",
        "        for _ in range(n_samples):\n",
        "            # choose the Y label with 50/50 odds\n",
        "            p = torch.rand((1,)).item()\n",
        "            if p<.5:\n",
        "                y=0\n",
        "            else:\n",
        "                y=1\n",
        "                #label y==0 case: for attribute 0, pick which Gaussian to sample \n",
        "                #according to the mixture distribution, then sample from that Gaussian\n",
        "            labels.append(y)\n",
        "            if att==0:\n",
        "                idx = self.mixture[y].sample()\n",
        "                x = self.gmm[y][idx].sample()\n",
        "                features.append((x @ self.rotation.T).numpy())\n",
        "            #for attribute 1, sample from the associated uniform distribution\n",
        "            #for the x coordinate and set y-coord=0\n",
        "            elif att==1:\n",
        "                x = self.unif[y].sample().item()                   \n",
        "                features.append(np.array([x,0]))\n",
        "        \n",
        "        return features, labels"
      ],
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "c2zcTE6xQGEC"
      },
      "source": [
        "### Visualization\n",
        "This plots a sampling of the distributions specified by the parameters in lines 1-7 and can be used to tune the subjective difficulty of the different distributions. "
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 323
        },
        "id": "PcFPxxpsDU8U",
        "outputId": "9252fed6-6812-4b64-e33d-4be94c0caadc"
      },
      "source": [
        "means1 = [torch.tensor([-2.5,0.0]),torch.tensor([-0.25,3.0]),torch.tensor([-1.5,-3.0])]\n",
        "var1 = [torch.tensor([[0.25,0.0],[0.0,2.0]]),torch.tensor([[1.0,0.0],[0.0,1.0]]),torch.tensor([[1.0,0.0],[0.0,1.0]])]\n",
        "means2 = [torch.tensor([2.5,0.0]),torch.tensor([0.75,3.0]),torch.tensor([0.75,-3.0])]\n",
        "var2 = [torch.tensor([[0.25,0.0],[0.0,2.0]]),torch.tensor([[1.0,0.0],[0.0,1.0]]),torch.tensor([[1.0,0.0],[0.0,1.0]])]\n",
        "mix = torch.tensor([0.5,0.25,0.25])\n",
        "unif1 = [-1.5, 0.7]\n",
        "unif2 = [-0.7, 1.5]\n",
        "\n",
        "oracle = Oracle_2(means1, var1, means2, var2, mix, mix, unif1, unif2)\n",
        "\n",
        "X,y = oracle(att=0, n_samples=100)\n",
        "for x,y in zip(X,y):\n",
        "    if y==0:\n",
        "        plt.scatter(x[0],x[1], c='b', marker='o', s=70)\n",
        "    else:\n",
        "        plt.scatter(x[0],x[1], c='b', marker='x', s=70)\n",
        "\n",
        "X,y = oracle(att=1, n_samples=25)\n",
        "for x,y in zip(X,y):\n",
        "    if y==0:\n",
        "        plt.scatter(x[0],x[1], c='r', marker='o', s=40)\n",
        "    else:\n",
        "        plt.scatter(x[0],x[1], c='r', marker='x', s=100)\n",
        "\n",
        "plt.title('Synthetic Model II', fontsize=21, fontweight='bold')\n",
        "plt.xticks(fontsize=16, fontweight='bold')\n",
        "plt.yticks(fontsize=16, fontweight='bold')\n",
        "plt.xlabel(' ', fontsize=10)\n",
        "plt.ylabel(' ', fontsize=10)\n"
      ],
      "execution_count": null,
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "Text(0, 0.5, ' ')"
            ]
          },
          "metadata": {
            "tags": []
          },
          "execution_count": 29
        },
        {
          "output_type": "display_data",
          "data": {
            "image/png": "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\n",
            "text/plain": [
              "<Figure size 432x288 with 1 Axes>"
            ]
          },
          "metadata": {
            "tags": [],
            "needs_background": "light"
          }
        }
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "WQBCzZQ6QWRo"
      },
      "source": [
        "### Mann-Kendall Statistic\n",
        "This class tracks the Mann-Kendall statistic for the accuracy of each attribute on the round after they were sampled.  `window` controls how far back-looking the statistic is: how many previous accuracies it remembers, and `param` is just a scaling parameter.  Update adds a new accuracy to the specified attribute, bumping off the oldest one if the window has been filled, and re-calculates the normalized statistic.  The call function returns the normalized, scaled statistic for the specified attribute.  "
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "id": "VyJ1Vi5qJ0KC"
      },
      "source": [
        "#tracks losses after an attribute has been selected and calculates the associated Mann-Kendall statistic\n",
        "# window controls how far back-looking it is and param is just a scaling parameter\n",
        "from collections import deque\n",
        "class MKStat():\n",
        "    def __init__(self, num_att=2, window=20, param=.1):\n",
        "        self.num_att = num_att\n",
        "        self.window = window\n",
        "        self.param = param\n",
        "        self.accuracies = [deque() for i in range(self.num_att)]\n",
        "        self.stats = [0]*self.num_att\n",
        "    def update(self, idx, new_acc):\n",
        "        #idx is which attribute you're updating, new_acc is the new accuracy to add\n",
        "        if torch.is_tensor(new_acc):\n",
        "            new_acc = new_acc.item()\n",
        "        self.accuracies[idx].append(new_acc)\n",
        "        #if we've exceeded our tracking window, get rid of the oldest accuracy\n",
        "        m = len(self.accuracies[idx])\n",
        "        if m > self.window:\n",
        "            self.accuracies[idx].popleft()\n",
        "            m -= 1\n",
        "        #update the statistic for this attribute\n",
        "        S = 0\n",
        "        for i in range(m-1):\n",
        "            for j in range(i+1,m):\n",
        "                S += np.sign(self.accuracies[idx][j]-self.accuracies[idx][i])\n",
        "        #normalize and store the statistic\n",
        "        std = np.sqrt((1/18)*m*(m-1)*(2*m+5))\n",
        "        if S < 0:\n",
        "            S -= 1\n",
        "        elif S > 0:\n",
        "            S += 1\n",
        "        if m>1:\n",
        "            self.stats[idx] = (S/std)\n",
        "    def __call__(self, idx):\n",
        "        return self.param * self.stats[idx]"
      ],
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "metadata": {
        "id": "fsZhtMVTGqoU"
      },
      "source": [
        "\n",
        "def helper1(features, labels, atts,  att=0, return_idx=False):\n",
        "  Idx = [i for i,z in enumerate(atts) if z==att]\n",
        "  if return_idx:\n",
        "    return Idx\n",
        "  if isinstance(features, list):\n",
        "    Xa = np.array(features)[Idx]\n",
        "    Ya = np.array(labels)[Idx]\n",
        "  else: #np array \n",
        "    Xa = features[Idx]\n",
        "    Ya = features[Idx]\n",
        "  return Xa, Ya\n",
        "  \n",
        "def Experiment0(model, oracle, mkstat, features1, labels1, features2, labels2, \n",
        "                train_attributes, val_attributes, n0, TT=500, rule='UCB', param=None, \n",
        "                forced=True, verbose=True):\n",
        "  '''\n",
        "  rule  :string   possible options are\n",
        "                  ucb0 -- standard UCB\n",
        "                  emp  -- empirical, i.e., greedy \n",
        "                  eps   -- epsilon greedy (with epsilon=param)\n",
        "  '''\n",
        "  # Initialize the Pi_a Vector \n",
        "  Pi_a = np.zeros((TT,))\n",
        "  Pi_a[:n0] = 0.5 # set the first $n0$ elements to 0.5 (technically not correct!!)\n",
        "\n",
        "  # start the main loop\n",
        "  acc = []\n",
        "  for t in range(2*n0, TT):\n",
        "    # fit the model to the current data\n",
        "    model.fit(features1, labels1)\n",
        "    # Obtain the validation set for attributes\n",
        "    Xa, Ya = helper1(features2, labels2, atts=val_attributes, att=0)\n",
        "    Xb, Yb = helper1(features2, labels2, atts=val_attributes, att=1)\n",
        "    # Predict labels on the two validation sets using model\n",
        "    Yap, Ybp = model.predict(Xa), model.predict(Xb)\n",
        "    # Compute the accuracy \n",
        "    acc_a = np.array([1 if ypi==yi else 0 for ypi, yi in zip(Yap, Ya)]).mean()\n",
        "    acc_b = np.array([1 if ypi==yi else 0 for ypi, yi in zip(Ybp, Yb)]).mean()\n",
        "    num_a = len(helper1(features1, labels1, train_attributes, att=0, return_idx=True))\n",
        "    num_b = len(helper1(features1, labels1, train_attributes, att=1, return_idx=True))\n",
        "\n",
        "    # Point Selection Rule\n",
        "    if rule=='ucb0':# Optimistic Rule\n",
        "      if param is None:\n",
        "        param=0.1\n",
        "      U_a, U_b = acc_a - param/np.sqrt(num_a), acc_b - param/np.sqrt(num_b)\n",
        "    elif rule=='emp':# Empirical Rule\n",
        "      U_a, U_b = acc_a, acc_b\n",
        "    elif rule=='eps':\n",
        "      if param is None:\n",
        "        param=0.1 # epsilon value for epsilon-greedy sampling\n",
        "      if np.random.random()<param: # randomly choose one of two attributes\n",
        "        if np.random.random()>0.5:\n",
        "          U_a, U_b = 1, 0\n",
        "        else:\n",
        "          U_a, U_b = 0, 1\n",
        "      else: # empirical sampling otherwise\n",
        "        U_a, U_b = acc_a, acc_b\n",
        "    elif rule=='ucb+': #optimistic rule with trend statistic\n",
        "        stat_a = mkstat(0)\n",
        "        stat_b = mkstat(1)\n",
        "        if param is None:\n",
        "            param=0.1\n",
        "        U_a = acc_a - param/np.sqrt(num_a) - stat_a\n",
        "        U_b = acc_b - param/np.sqrt(num_b) - stat_b\n",
        "  \n",
        "    # choose the attribute with lower value of the index function U \n",
        "    if num_a < np.sqrt(t) and forced:\n",
        "      at=0\n",
        "    elif num_b<np.sqrt(t) and forced:\n",
        "      at=1\n",
        "    else:\n",
        "      at = 0 if U_a<U_b else 1\n",
        "    \n",
        "    #add the accuracy for the chosen attribute to the trend statistic\n",
        "    if at==0: \n",
        "        mkstat.update(idx=at, new_acc=acc_a)\n",
        "    elif at==1:\n",
        "        mkstat.update(idx=at, new_acc=acc_b)\n",
        "    \n",
        "    # Store the updated Pi_a value \n",
        "    Pi_a[t] = (num_a + 1)/(t+1) if at==0 else num_a/(t+1)\n",
        "\n",
        "    # Print the values \n",
        "    if t%100==49 and verbose:\n",
        "      print('U_a = {:.2f}, acc_a = {:.2f}, \\t U_b = {:.2f}, acc_b = {:.2f}, \\t pi_a = {}, \\t at = {}\\t t = {}'.format(U_a, acc_a, U_b, acc_b, num_a/t, at, t))\n",
        "      print('\\n')\n",
        "\n",
        "    if t%(TT//100)==(TT//100 - 1):\n",
        "        xt,yt = oracle(att=0, n_samples=5000)\n",
        "        pred = model.predict(xt)\n",
        "        ac_a = (pred==yt).mean()\n",
        "        xt,yt = oracle(att=1, n_samples=5000)\n",
        "        pred = model.predict(xt)\n",
        "        ac_b = (pred==yt).mean()\n",
        "        acc.append([ac_a,ac_b])\n",
        "    # draw a sample from the chosen attribute \n",
        "    xt, yt = oracle(att=at, n_samples=2)\n",
        "    features1.append(xt[0])\n",
        "    features2.append(xt[1])\n",
        "    labels1.append(yt[0])\n",
        "    labels2.append(yt[1])\n",
        "    val_attributes.append(at)\n",
        "    train_attributes.append(at)\n",
        "  return acc, Pi_a\n",
        "  \n"
      ],
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "metadata": {
        "id": "Ao7duETiQ8yS"
      },
      "source": [
        "# Run the experiment \n",
        "def runExperiment0(\n",
        "    n0=1,\n",
        "    rule='ucb0',\n",
        "    forced='True',\n",
        "    param=0.1,\n",
        "    n_trials=100,\n",
        "    TT=500,\n",
        "    verbose=True,\n",
        "    window = 20,\n",
        "    monoparam = 0.2\n",
        "):\n",
        "\n",
        "  # Initialize the vectors.\n",
        "  start_time = time()\n",
        "  # track the accuracy over time and the learned mixture distribution\n",
        "  total_acc = []\n",
        "  all_dist = []\n",
        "  for i in range(n_trials):\n",
        "    print('Starting trial {}/{} with rule = {}, forced = {}'.format(\n",
        "       i+1, n_trials, rule, forced \n",
        "    )\n",
        "    )\n",
        "    # Initialize the model\n",
        "    model = LogReg(C=.5)\n",
        "    # Set the parameters for each of the distributions and initialize the oracle\n",
        "    means1 = [torch.tensor([-2.5,0.0]),torch.tensor([-0.25,3.0]),torch.tensor([-1.5,-3.0])]\n",
        "    var1 = [torch.tensor([[0.25,0.0],[0.0,2.0]]),torch.tensor([[1.0,0.0],[0.0,1.0]]),torch.tensor([[1.0,0.0],[0.0,1.0]])]\n",
        "    means2 = [torch.tensor([2.5,0.0]),torch.tensor([0.75,3.0]),torch.tensor([0.75,-3.0])]\n",
        "    var2 = [torch.tensor([[0.25,0.0],[0.0,2.0]]),torch.tensor([[1.0,0.0],[0.0,1.0]]),torch.tensor([[1.0,0.0],[0.0,1.0]])]\n",
        "    mix = torch.tensor([0.5,0.25,0.25])\n",
        "    unif1 = [-1.2, 1.0]\n",
        "    unif2 = [-1.0, 1.2]\n",
        "\n",
        "    oracle = Oracle_2(means1, var1, means2, var2, mix, mix, unif1, unif2, rot_angle = math.pi/6)\n",
        "    #initialize M-K stat tracker\n",
        "    mkstat = MKStat(num_att=2, window=window, param=monoparam)\n",
        "    # Get the initial training datasets (2*n0)\n",
        "    while True:\n",
        "      X_temp0, Y_temp0 = oracle(att=0, n_samples=n0)\n",
        "      X_temp1, Y_temp1 = oracle(att=1, n_samples=n0)\n",
        "      features1, labels1 = X_temp0+X_temp1, Y_temp0+Y_temp1\n",
        "      if len(set(labels1))>1: # ensures at least one element of each label\n",
        "        break\n",
        "    # Get the initial validation dataset (of size 10*n0) \n",
        "    while True:\n",
        "      X_temp0, Y_temp0 = oracle(att=0, n_samples=10*n0)\n",
        "      X_temp1, Y_temp1 = oracle(att=1, n_samples=10*n0)\n",
        "      features2, labels2 = X_temp0+X_temp1, Y_temp0+Y_temp1\n",
        "      if len(set(labels2))>1: # ensures at least one element of each label\n",
        "        break     \n",
        "    # Set the attributes selected so far\n",
        "    train_attributes = [0]*n0+[1]*n0\n",
        "    val_attributes = [0]*10*n0+[1]*10*n0\n",
        "    # Run the experiment to get one pi vector\n",
        "    acc_temp, pi_temp = Experiment0(model=model, oracle=oracle, mkstat=mkstat,\n",
        "        features1=features1, labels1=labels1, features2=features2, labels2=labels2,\n",
        "        train_attributes=train_attributes, val_attributes=val_attributes, n0=n0, TT=TT,\n",
        "        rule=rule, param=param, forced=forced, verbose=verbose\n",
        "    )\n",
        "    total_acc.append(acc_temp)\n",
        "    all_dist.append(pi_temp)\n",
        "  end_time=time()\n",
        "  print('Completed {} trials in {:.2f} seconds \\n \\n'.format(n_trials, -start_time+end_time)) \n",
        "\n",
        "  return total_acc, all_dist\n",
        "  \n",
        "\n",
        "rule = 'ucb0'\n",
        "TT = 200\n",
        "acc, dist = runExperiment0(rule=rule, TT=TT, forced=False, param=0.1)\n",
        "\n",
        "torch.save(np.array(acc), rule+'_synthetic_acc.pt')\n",
        "torch.save(np.array(dist), rule+'_synthetic_mix.pt')\n"
      ],
      "execution_count": null,
      "outputs": []
    }
  ]
}