{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Defaulting to user installation because normal site-packages is not writeable\n",
      "Requirement already satisfied: scipy in /Users/seokjinkim/Library/Python/3.9/lib/python/site-packages (1.13.0)\n",
      "Requirement already satisfied: numpy<2.3,>=1.22.4 in /Users/seokjinkim/Library/Python/3.9/lib/python/site-packages (from scipy) (1.26.4)\n",
      "\u001b[33mWARNING: You are using pip version 21.2.4; however, version 24.3.1 is available.\n",
      "You should consider upgrading via the '/Library/Frameworks/Python.framework/Versions/3.9/bin/python3.9 -m pip install --upgrade pip' command.\u001b[0m\n",
      "Note: you may need to restart the kernel to use updated packages.\n"
     ]
    }
   ],
   "source": [
    "pip install scipy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Defaulting to user installation because normal site-packages is not writeable\n",
      "Requirement already satisfied: sklearn in /Users/seokjinkim/Library/Python/3.9/lib/python/site-packages (0.0)\n",
      "Requirement already satisfied: scikit-learn in /Users/seokjinkim/Library/Python/3.9/lib/python/site-packages (from sklearn) (1.5.0)\n",
      "Requirement already satisfied: scipy>=1.6.0 in /Users/seokjinkim/Library/Python/3.9/lib/python/site-packages (from scikit-learn->sklearn) (1.13.0)\n",
      "Requirement already satisfied: joblib>=1.2.0 in /Users/seokjinkim/Library/Python/3.9/lib/python/site-packages (from scikit-learn->sklearn) (1.4.2)\n",
      "Requirement already satisfied: threadpoolctl>=3.1.0 in /Users/seokjinkim/Library/Python/3.9/lib/python/site-packages (from scikit-learn->sklearn) (3.5.0)\n",
      "Requirement already satisfied: numpy>=1.19.5 in /Users/seokjinkim/Library/Python/3.9/lib/python/site-packages (from scikit-learn->sklearn) (1.26.4)\n",
      "\u001b[33mWARNING: You are using pip version 21.2.4; however, version 24.3.1 is available.\n",
      "You should consider upgrading via the '/Library/Frameworks/Python.framework/Versions/3.9/bin/python3.9 -m pip install --upgrade pip' command.\u001b[0m\n",
      "Note: you may need to restart the kernel to use updated packages.\n"
     ]
    }
   ],
   "source": [
    "pip install sklearn"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "#!/usr/bin/env python\n",
    "# coding: utf-8\n",
    "\n",
    "import numpy as np\n",
    "import scipy.stats as stats\n",
    "from scipy.stats import norm\n",
    "import matplotlib.pyplot as plt\n",
    "from sklearn import linear_model\n",
    "from numpy.linalg import inv\n",
    "# from models import LassoBandit, DRLassoBandit, SALassoBandit"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "def sample_spherical(N, k):\n",
    "    vec = np.random.randn(k, N)\n",
    "    vec /= np.linalg.norm(vec, axis=0)\n",
    "    return vec\n",
    "\n",
    "def sample_elliptical(N, d, k, mu):\n",
    "    S = sample_spherical(N, k)\n",
    "    A = np.random.rand(d,k)\n",
    "    R = np.random.normal(size=N)\n",
    "    return mu + A.dot(R*S)\n",
    "\n",
    "def SMInv(Ainv, u, v):\n",
    "    return Ainv - np.dot(Ainv, np.dot(np.outer(u,v), Ainv)) / ( 1 + np.dot(v, np.dot(Ainv, u)))\n",
    "\n",
    "\n",
    "def random_ball(num_points, dimension, radius=1):\n",
    "    from numpy import random, linalg\n",
    "    # First generate random directions by normalizing the length of a\n",
    "    # vector of random-normal values (these distribute evenly on ball).\n",
    "    random_directions = random.normal(size=(dimension,num_points))\n",
    "    random_directions /= linalg.norm(random_directions, axis=0)\n",
    "    # Second generate a random radius with probability proportional to\n",
    "    # the surface area of a ball with a given radius.\n",
    "    random_radii = random.random(num_points) ** (1/dimension)\n",
    "    # Return the list of random (direction & length) points.\n",
    "    return radius * (random_directions * random_radii).T, "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "class linUCB:\n",
    "    def __init__(self,sigma,d,lam):\n",
    "        self.d=d\n",
    "        self.x=[]\n",
    "        self.r=[]\n",
    "        self.sigma=sigma\n",
    "        self.lam = lam\n",
    "        self.theta=np.zeros(d)\n",
    "        self.Vinv=1/lam*np.identity(d)\n",
    "        \n",
    "    def choose_a(self,t,x):  # x is N*d matrix\n",
    "        self.alpha = self.sigma*np.sqrt(2*self.d*np.log(t**2+t**3/self.lam)) + np.sqrt(self.lam)\n",
    "        means = np.dot(x,self.theta)\n",
    "        xv = np.sqrt((np.matmul(x, self.Vinv) * x).sum(axis = 1))\n",
    "        u = means + self.alpha * xv\n",
    "        self.action=np.argmax(u)\n",
    "        xt = x[self.action]\n",
    "        self.x.append(xt)\n",
    "        self.Vinv = SMInv(self.Vinv, xt, xt)\n",
    "        return(self.action)\n",
    "\n",
    "    def update_beta(self,rwd,t):\n",
    "        self.r.append(rwd)\n",
    "        if t>3:\n",
    "            model=linear_model.Ridge(alpha=self.lam)\n",
    "            model.fit(self.x,self.r)\n",
    "            self.theta=model.coef_\n",
    "            \n",
    "\n",
    "class linTS:\n",
    "    def __init__(self,sigma,d,lam):\n",
    "        self.d=d\n",
    "        self.x=[]\n",
    "        self.r=[]\n",
    "        self.sigma=sigma\n",
    "        self.lam = lam\n",
    "        self.theta=np.zeros(d)\n",
    "        self.Vinv=1/lam*np.identity(d)\n",
    "        \n",
    "    def choose_a(self,t,x):  # x is N*d matrix\n",
    "        self.alpha = self.sigma*np.sqrt(self.d*np.log(self.lam+t)+d*np.log(self.lam) + 4*np.log(t)) + np.sqrt(self.lam)\n",
    "        theta_tilde = np.random.multivariate_normal(self.theta, np.square(self.alpha)*self.Vinv)\n",
    "        means = np.dot(x,theta_tilde)\n",
    "        self.action=np.argmax(means)\n",
    "        xt = x[self.action]\n",
    "        self.x.append(xt)\n",
    "        self.Vinv = SMInv(self.Vinv, xt, xt)\n",
    "        return(self.action)\n",
    "\n",
    "    def update_beta(self,rwd,t):\n",
    "        self.r.append(rwd)\n",
    "        if t>3:\n",
    "            model=linear_model.Ridge(alpha=self.lam)\n",
    "            model.fit(self.x,self.r)\n",
    "            self.theta=model.coef_\n",
    "\n",
    "\n",
    "class linGreedy:\n",
    "    def __init__(self, sigma, d, lam):\n",
    "        self.d = d\n",
    "        self.x = []\n",
    "        self.r = []\n",
    "        self.theta = np.zeros(d)\n",
    "        self.lam = lam\n",
    "\n",
    "    def choose_a(self, t, x):  # x is N*d matrix\n",
    "        means = np.dot(x, self.theta)\n",
    "        self.action = np.argmax(means)\n",
    "        xt = x[self.action]\n",
    "        self.x.append(xt)\n",
    "        return self.action\n",
    "\n",
    "    def update_beta(self, rwd, t):\n",
    "        self.r.append(rwd)\n",
    "        if t > 3:\n",
    "            X = np.array(self.x)\n",
    "            y = np.array(self.r)\n",
    "\n",
    "            try:\n",
    "                #Attempt OLS estimation\n",
    "                XtX = np.dot(X.T, X)\n",
    "                Xty = np.dot(X.T, y)\n",
    "                self.theta = np.linalg.solve(XtX, Xty)\n",
    "#                 model=linear_model.Ridge(alpha=0.000001)\n",
    "#                 model.fit(self.x,self.r)\n",
    "#                 self.theta=model.coef_\n",
    "            except np.linalg.LinAlgError:\n",
    "                # If XtX is singular, set theta to (1, 0, ..., 0)\n",
    "                self.theta = np.zeros(self.d)\n",
    "                self.theta[0] = 1\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "#simulation settings\n",
    "\n",
    "K=20\n",
    "d=100\n",
    "R=0.2\n",
    "T=1000\n",
    "dist =5\n",
    "\n",
    "sigma_sq=1.\n",
    "rho_sq=0.7\n",
    "V=(sigma_sq-rho_sq)*np.eye(K) + rho_sq*np.ones((K,K))\n",
    "\n",
    "fixed = 0\n",
    "\n",
    "\n",
    "np.random.seed(2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "1000\n",
      "2\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "1000\n",
      "3\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "1000\n",
      "4\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "1000\n",
      "5\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "1000\n",
      "6\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "1000\n",
      "7\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "1000\n",
      "8\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "1000\n",
      "9\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "1000\n",
      "10\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "1000\n",
      "11\n",
      "100\n",
      "200\n",
      "300\n",
      "400\n",
      "500\n",
      "600\n",
      "700\n",
      "800\n",
      "900\n",
      "1000\n"
     ]
    }
   ],
   "source": [
    "simul_n=10\n",
    "\n",
    "def truncated_standard_t(df, size, radius):\n",
    "    samples = []\n",
    "    while len(samples) < size[0]:\n",
    "        sample = np.random.standard_t(df, size=size[1])\n",
    "        if np.linalg.norm(sample) <= radius:\n",
    "            samples.append(sample)\n",
    "    return np.array(samples)\n",
    "\n",
    "if dist == 0:\n",
    "    x = np.random.multivariate_normal(np.zeros(K), V, d).T\n",
    "    x = x / np.sqrt(d)\n",
    "elif dist == 1:\n",
    "    x = np.asarray(random_ball(K, d, radius=np.sqrt(d))).reshape((K, d))\n",
    "elif dist == 2:\n",
    "    x = truncated_standard_t(df=10, size=(K, d), radius=np.sqrt(d))\n",
    "elif dist == 3:\n",
    "    location = 0\n",
    "    scale = 1\n",
    "    x = np.random.laplace(location, scale, K * d).reshape((K, d))\n",
    "elif dist == 4:\n",
    "    x = np.random.multivariate_normal(np.zeros(K), V, d).T\n",
    "\n",
    "elif dist == 5:\n",
    "    loc = 0\n",
    "    scale = 1\n",
    "    size = (K, d)\n",
    "    x = stats.cauchy(loc, scale).rvs(size=size)\n",
    "    x = np.clip(x, -5, 5)\n",
    "    \n",
    "sim_start = 1\n",
    "for simul in range(sim_start, sim_start+simul_n+1):\n",
    "    \n",
    "    beta=np.random.uniform(0.1,1.,d)/np.sqrt(d)\n",
    "    \n",
    "    print(simul)\n",
    "    \n",
    "    if fixed == 1:\n",
    "        savename = \"results/linBandit_N={}_d={}_rho={}_dist={}_id={}_fixed.csv\".format(K, d, rho_sq, dist, simul)\n",
    "    else:\n",
    "        savename = \"results/linBandit_N={}_d={}_rho={}_dist={}_id={}.csv\".format(K, d, rho_sq, dist, simul)\n",
    "    \n",
    "    cumulated_regret_linUCB=[]\n",
    "    cumulated_regret_linTS=[]\n",
    "    cumulated_regret_Greedy=[]\n",
    "\n",
    "    models = []\n",
    "    models.append(linUCB(sigma=R,d=d,lam = 0.5))\n",
    "    models.append(linTS(sigma=R,d=d,lam = 0.5))\n",
    "    models.append(linGreedy(sigma=R,d=d,lam = 0.5))\n",
    "    \n",
    "    num_models = len(models)\n",
    "    \n",
    "    cumulated_regret = [ [] for _ in range(num_models) ]\n",
    "    rewards = [ [] for _ in range(num_models) ]\n",
    "    optRWD=list()\n",
    "    \n",
    "    for t in range(1,T+1):\n",
    "        if (t % 100)==0:\n",
    "            print(t)\n",
    "                \n",
    "        if fixed == 0:\n",
    "            if dist == 0:\n",
    "                x=np.random.multivariate_normal(np.zeros(K),V,d).T\n",
    "            elif dist == 1:\n",
    "                x = np.asarray(random_ball(K,d,radius=np.sqrt(d))).reshape((K, d))\n",
    "            elif dist == 2:\n",
    "                x = np.random.standard_t(10, size=K*d).reshape((K, d))\n",
    "            elif dist == 3:\n",
    "                x = np.random.laplace(location, scale, K*d).reshape((K, d))\n",
    "            elif dist == 4:\n",
    "                x=np.random.multivariate_normal(np.zeros(K),V,d).T\n",
    "            elif dist == 5:\n",
    "                x = np.random.gumbel(0, 1, K*d).reshape((K, d))\n",
    "\n",
    "        err=R*np.random.randn()\n",
    "\n",
    "        for m in range(num_models):\n",
    "            a = models[m].choose_a(t+1,x)\n",
    "            mean_rwd = np.dot(x[a],beta)\n",
    "            rwd= mean_rwd + err\n",
    "            rewards[m].append(mean_rwd)\n",
    "            models[m].update_beta(rwd,t+1)\n",
    "\n",
    "        optRWD.append(np.amax(np.dot(x,beta)))\n",
    "        \n",
    "    for m in range(num_models):\n",
    "        cumulated_regret[m].append(np.cumsum(optRWD)-np.cumsum(rewards[m]))\n",
    "    \n",
    "    regret = np.vstack(cumulated_regret)\n",
    "    np.savetxt(savename, regret, delimiter=\",\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "# d=5\n",
    "# rho_sq=0.0\n",
    "reg_list = []\n",
    "for i in range(1,simul_n+1):\n",
    "    if fixed == 3: \n",
    "        reg_list.append(np.genfromtxt(\n",
    "            'results/linBandit_N={}_d={}_rho={}_dist={}_id={}_fixed.csv'.format(K, d, rho_sq, dist, i), delimiter=','))\n",
    "    else:\n",
    "        reg_list.append(np.genfromtxt(\n",
    "            'results/linBandit_N={}_d={}_rho={}_dist={}_id={}.csv'.format(K, d, rho_sq, dist, i), delimiter=','))\n",
    "        \n",
    "total_reg = np.stack(reg_list)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 374.4x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig = plt.figure(figsize=(5.2, 4))\n",
    "plt.style.use('default')\n",
    "\n",
    "steps=np.arange(1,T+1)\n",
    "freq = int(T/10)\n",
    "\n",
    "avg_reg = total_reg.mean(axis=0)\n",
    "sd_req = total_reg.std(axis=0)\n",
    "\n",
    "# plt.errorbar(steps, avg_reg, sd_req, errorevery=freq, marker='^', markevery=freq, label='linUCB', markersize=4, linewidth=2, elinewidth=1, capsize=3)\n",
    "\n",
    "plt.errorbar(steps, avg_reg[0,:], sd_req[0,:], errorevery=freq, marker='^', markevery=freq, label='LinUCB', markersize=4, linewidth=2, elinewidth=1, capsize=3)\n",
    "plt.errorbar(steps, avg_reg[1,:], sd_req[1,:], errorevery=freq, marker='x', markevery=freq, label='LinTS', linewidth=2, elinewidth=1, capsize=3)\n",
    "plt.errorbar(steps, avg_reg[2,:], sd_req[2,:], errorevery=freq, marker='.', markevery=freq, label='LinGreedy', linewidth=2, elinewidth=1, capsize=3)\n",
    "# plt.errorbar(steps, avg_reg[2,:], sd_req[2,:], errorevery=freq, marker='.', markevery=freq, label=r'$\\epsilon$-Greedy + $\\ell_1 + 1/T$', linewidth=2, elinewidth=1, capsize=3)\n",
    "\n",
    "\n",
    "# plt.grid(True)\n",
    "plt.grid(color='0.85')\n",
    "plt.xlabel('Round ($t$)', size = 14)\n",
    "plt.ylabel('Cumulative Regret', size = 14)\n",
    "if rho_sq == 0.0:\n",
    "    plt.title(r'$d$={}, $K$={}, Gaussian'.format(d, K, int(rho_sq)), size = 14)\n",
    "else:\n",
    "    plt.title(r'$d$={}, $K$={}, $\\rho^2$={}'.format(d, K, rho_sq), size = 14)\n",
    "if dist == 1:\n",
    "    plt.title('$d$={}, $K$={}, Uniform'.format(d, K), size = 14) \n",
    "if dist == 2:\n",
    "    plt.title('$d$={}, $K$={}, Truncated Student t'.format(d, K), size = 14) \n",
    "if dist == 3:\n",
    "    plt.title('$d$={}, $K$={}, Laplace'.format(d, K), size = 14) \n",
    "if dist == 4:    \n",
    "    plt.title('$d$={}, $K$={}, Guassian'.format(d, K), size = 14)\n",
    "if dist == 5:    \n",
    "    plt.title('$d$={}, $K$={}, Truncated Cauchy'.format(d, K), size = 14)     \n",
    "plt.legend(loc='upper left', prop={'size': 12})\n",
    "plt.tick_params(labelsize=12)\n",
    "plt.savefig('linearBandit_d={}_K={}_dist={}.pdf'.format(d,K,dist))"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
