{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 201,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<module 'pred' from '/mnt/c/Users/nir897/Documents/proj/robias/code/lookahead/pred.py'>"
      ]
     },
     "execution_count": 201,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import importlib\n",
    "import numpy as np\n",
    "from matplotlib import pyplot as plt\n",
    "from mpl_toolkits.axes_grid1.inset_locator import inset_axes\n",
    "import inspect\n",
    "import copy\n",
    "from collections import OrderedDict\n",
    "\n",
    "from sklearn.datasets import load_boston\n",
    "from sklearn.preprocessing import StandardScaler\n",
    "from sklearn.model_selection import train_test_split, GridSearchCV\n",
    "from sklearn.svm import SVR\n",
    "from sklearn.linear_model import RidgeCV, LinearRegression\n",
    "from denseratio.core import densratio\n",
    "\n",
    "%matplotlib inline\n",
    "np.set_printoptions(precision=3)\n",
    "\n",
    "import lookahead\n",
    "from lookahead import *\n",
    "import uncert\n",
    "import pred\n",
    "import prop\n",
    "\n",
    "importlib.reload(pred)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 202,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "class Fstar():\n",
    "    def __init__(self, coeffs = []):\n",
    "        self.coeffs = coeffs\n",
    "    def fit(self, x, y):\n",
    "        pass\n",
    "    def predict(self, x):\n",
    "        c = self.coeffs[-1]\n",
    "        for i in range(2, len(self.coeffs) + 1):\n",
    "            c = self.coeffs[-i] + c*x\n",
    "        return c.flatten()\n",
    "    \n",
    "def improve_rate(x, y, eta, mask, model):\n",
    "    xp = model.move_points(x ,eta, mask)\n",
    "    yp = model.f.predict(xp)\n",
    "    return np.mean(yp>y)\n",
    "\n",
    "def make_features_np(X):\n",
    "    return np.concatenate([(X ** i)/fact[i] for i in range(1, degree + 1)], axis = -1)\n",
    "\n",
    "class PolyRegression(torch.nn.Module):\n",
    "    def __init__(self, inputSize, outputSize):\n",
    "        super(PolyRegression, self).__init__()\n",
    "        self.linear = torch.nn.Linear(inputSize, outputSize)\n",
    "        self.inputSize = inputSize\n",
    "        self.linear.weight.data.fill_(0)\n",
    "        self.linear.bias.data.fill_(0)\n",
    "              \n",
    "    def make_features(self, x):\n",
    "        return torch.cat([(x ** i)/fact[i] for i in range(1, self.inputSize + 1)], 1)\n",
    "    \n",
    "    def forward(self, x):\n",
    "        x2 = self.make_features(x)\n",
    "        out = self.linear(x2)\n",
    "        return out\n",
    "\n",
    "    def predict(self, x):\n",
    "        yhat = self(Variable(torch.from_numpy(x).float())).data.numpy().squeeze()\n",
    "        return yhat"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 203,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "def get_perf(model, xs, ys, eta, mask, uncert=False):\n",
    "    perf = {'mse':[], 'mae':[], 'improve':[], 'imprate':[], 'contain':[], 'size':[]}\n",
    "    perf['mse'].append([model.mse(x_,y_) for x_,y_ in zip(xs,ys)])\n",
    "    perf['mae'].append([model.mae(x_,y_) for x_,y_ in zip(xs,ys)])\n",
    "    perf['improve'].append([model.improve(x_,y_,eta,mask) for x_,y_ in zip(xs,ys)])\n",
    "    perf['imprate'].append([improve_rate(x_,y_,eta,mask,model) for x_,y_ in zip(xs,ys)])\n",
    "    if uncert:\n",
    "        xsp = [model.move_points(x_) for x_ in xs]\n",
    "        perf['contain'].append([model.contain(x_)[0] for x_ in [*xsp, x]])\n",
    "        perf['size'].append([model.contain(x_)[1] for x_ in [*xsp, x]])\n",
    "    perf = {k:np.asarray(v) for k,v in zip(perf.keys(),perf.values())}\n",
    "    return perf\n",
    "    \n",
    "def print_perf(perf, idx=0, uncert=False):\n",
    "    print('\\ttrn\\tts\\tactv\\tall')\n",
    "    print(('mse'+'\\t{:.4f}'*4).format(*perf['mse'][idx,:]))\n",
    "    print(('mae'+'\\t{:.4f}'*4).format(*perf['mae'][idx,:]))\n",
    "    print(('imprv'+'\\t{:.4f}'*4).format(*perf['improve'][idx,:]))\n",
    "    print(('imprt'+'\\t{:.4f}'*4).format(*perf['imprate'][idx,:]))\n",
    "    print()\n",
    "    if uncert:\n",
    "        print('\\ttrn\\'\\ttst\\'\\tactv\\'\\tall\\'\\tall')\n",
    "        print(('contn'+'\\t{:.3f}'*5).format(*perf['contain'][idx,:]))\n",
    "        print(('intrsz'+'\\t{:.3f}'*5).format(*perf['size'][idx,:]))\n",
    "        print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 204,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "def plot_quad(ax, M, x, y, x_plot, y_plot, eta, mask, lw=1, osz=50, olw=1):\n",
    "    x_plot_th = torch.from_numpy(x_plot.astype(np.float32))\n",
    "    ax.plot(x_plot, y_plot, '--', color=\"tab:orange\", linewidth =1.0, label =\"gt\", alpha=0.6)\n",
    "\n",
    "    f_pred = M.f.predict(x_plot)\n",
    "    ax.plot(x_plot, f_pred, label=\"f\", linewidth=lw, alpha=0.6)\n",
    "\n",
    "    if M.u is not None:\n",
    "        u_pred, l_pred = M.u.lu(x_plot_th)\n",
    "        u_pred = u_pred.detach().numpy()\n",
    "        l_pred = l_pred.detach().numpy()\n",
    "        ax.fill_between(x_plot.flatten(), u_pred.flatten(), l_pred.flatten(),  color='g', alpha=0.05, zorder=0)\n",
    "        ax.plot(x_plot.flatten(), u_pred.flatten(), \"tab:green\", linewidth=1.0, alpha=0.5, zorder=0)\n",
    "        ax.plot(x_plot.flatten(), l_pred.flatten(), \"tab:green\", linewidth=1.0, alpha=0.5, zorder=0)\n",
    "\n",
    "    xp = M.move_points(x, eta, mask)\n",
    "    ypstar = M.fstar.predict(xp)\n",
    "    ax.scatter(x, y, color=\"white\", edgecolor=\"tab:blue\", alpha = 1, s=osz, zorder=10, linewidth=olw)\n",
    "    ax.scatter(xp, ypstar, color=\"white\", edgecolor=\"tab:red\", alpha = 1, s=osz, zorder=10, linewidth=olw)\n",
    "\n",
    "    ax.set_xticks([])\n",
    "    ax.set_yticks([])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 205,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": [
    "seed = 3\n",
    "ns = 0.25\n",
    "sig = 0.5\n",
    "offset = -0.8\n",
    "trn_sz = 0.75\n",
    "degree = 2\n",
    "eta = 3.5\n",
    "n = 25   "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 206,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x7fe7c0db1cf8>]"
      ]
     },
     "execution_count": 206,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "np.random.seed(seed)\n",
    "\n",
    "fstar = Fstar([0.1, 0.5, -0.8])\n",
    "x = np.random.normal(size = (n, 1), scale=sig)+offset\n",
    "y = fstar.predict(x)\n",
    "y += np.random.normal(scale = ns, size = y.shape)\n",
    "plt.scatter(x, y)\n",
    "\n",
    "n = x.shape[0]\n",
    "d = x.shape[1]\n",
    "\n",
    "plt.rcParams['figure.figsize'] = (5.0, 5.0)\n",
    "x_plot = np.linspace(-6.0, 6.0, 1000)[:, np.newaxis]\n",
    "y_plot = fstar.predict(x_plot)\n",
    "plt.plot(x_plot, y_plot)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 207,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n: 25 , n_active: 25 , n_trn: 18 , n_tst: 7\n"
     ]
    }
   ],
   "source": [
    "# split data\n",
    "trn_sz = 0.75 # working-set relative size\n",
    "#active = np.where(x[:,0] < -0.5)[0]\n",
    "active = np.arange(x.shape[0])\n",
    "\n",
    "x_active = x[active,:]\n",
    "y_active = y[active]\n",
    "n_active = y_active.shape[0]\n",
    "\n",
    "x_trn, x_tst, y_trn, y_tst = train_test_split(x_active, y_active, test_size=1-trn_sz, random_state=seed)\n",
    "n_trn, n_tst = (x_trn.shape[0], x_tst.shape[0])\n",
    "print('n:', n, ', n_active:', n_active, \", n_trn:\", n_trn, \", n_tst:\", n_tst)\n",
    "\n",
    "xs = [x_trn, x_tst, x_active, x]\n",
    "ys = [y_trn, y_tst, y_active, y]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 208,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "mask: [1.]\n"
     ]
    }
   ],
   "source": [
    "alpha = 0.\n",
    "lam = 4.\n",
    "mask = np.ones(d)\n",
    "\n",
    "# lookahead:\n",
    "num_cycles = 5\n",
    "z_score = 1.65 # for confiednce intervals (1.28-90%, 1.65=95%)\n",
    "\n",
    "#models:\n",
    " # f:\n",
    "lr_f = 0.01\n",
    "num_iter_init = 5000 #for initial f\n",
    "num_iter_f = 500 #for training f in cycles\n",
    "num_iter_base = num_iter_init + num_iter_f*num_cycles\n",
    "\n",
    " # g:\n",
    "num_gs = 10 #number of bootstrapped models\n",
    "lr_g = 0.01\n",
    "num_iter_g = 6000 #for training g in cycles\n",
    "\n",
    "# h:\n",
    "    \n",
    "print('mask:', mask)\n",
    "\n",
    "fact = {1:1.0, 2:2.0, 3:6.0, 4:24.0, 5:120.0}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 209,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "training baseline:\n",
      "t: 0\n",
      "[f] mse: 0.0410, la_reg: 0.0000, norm_reg: 0.0000, obj: 0.0410\n",
      "[f] improve*: -35.131\n",
      "\n",
      "\ttrn\tts\tactv\tall\n",
      "mse\t0.0410\t0.1175\t0.0624\t0.0624\n",
      "mae\t0.1473\t0.3236\t0.1966\t0.1966\n",
      "imprv\t-35.1306\t-26.2806\t-32.6526\t-32.6526\n",
      "imprt\t0.0000\t0.0000\t0.0000\t0.0000\n",
      "\n"
     ]
    }
   ],
   "source": [
    "# train baseline\n",
    "print('training baseline:')\n",
    "f_model = PolyRegression(degree, 1)\n",
    "f_base = pred.PredModel(d, model = f_model, reg_type='none', alpha=alpha, lr=lr_f, num_iter_init=num_iter_base)\n",
    "model_base = Lookahead(f_base, None, None, lam=0., eta=eta, mask=mask, ground_truth_model=fstar)\n",
    "_, _ = model_base.train(x_trn, y_trn, num_cycles=0, random_state=seed, verbose=verbose)\n",
    "\n",
    "perf_base = get_perf(model_base, xs, ys, eta, mask)\n",
    "print_perf(perf_base)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 210,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "training lookahead:\n",
      "t: 0\n",
      "[f] mse: 0.0428, la_reg: 0.0000, norm_reg: 0.0000, obj: 0.0428\n",
      "[f] improve*: -33.817\n",
      "\n",
      "t: 1\n",
      "[h] n_eff: 8.75, w_sum: 0.64\n",
      "[u] loss: 0.0210, norm_reg: 0.0000, obj: 0.0210\n",
      "[u] size: 22.874, contain*: 1.000\n",
      "[f] mse: 0.5590, la_reg: 0.0221, norm_reg: 0.0000, obj: 0.6473\n",
      "[f] improve*: 1.223\n",
      "\n",
      "t: 2\n",
      "[h] n_eff: 4.47, w_sum: 3.48\n",
      "[u] loss: 0.0102, norm_reg: 0.0000, obj: 0.0102\n",
      "[u] size: 1.075, contain*: 0.611\n",
      "[f] mse: 0.4492, la_reg: 0.0045, norm_reg: 0.0000, obj: 0.4673\n",
      "[f] improve*: 0.305\n",
      "\n",
      "t: 3\n",
      "[h] n_eff: 5.22, w_sum: 2.03\n",
      "[u] loss: 0.0107, norm_reg: 0.0000, obj: 0.0107\n",
      "[u] size: 1.922, contain*: 0.222\n",
      "[f] mse: 0.4724, la_reg: 0.0104, norm_reg: 0.0000, obj: 0.5139\n",
      "[f] improve*: 0.612\n",
      "\n",
      "t: 4\n",
      "[h] n_eff: 4.97, w_sum: 2.06\n",
      "[u] loss: 0.0108, norm_reg: 0.0000, obj: 0.0108\n",
      "[u] size: 1.736, contain*: 0.167\n",
      "[f] mse: 0.4646, la_reg: 0.0058, norm_reg: 0.0000, obj: 0.4879\n",
      "[f] improve*: 0.484\n",
      "\n",
      "t: 5\n",
      "[h] n_eff: 5.11, w_sum: 2.20\n",
      "[u] loss: 0.0106, norm_reg: 0.0000, obj: 0.0106\n",
      "[u] size: 1.758, contain*: 0.222\n",
      "[f] mse: 0.4750, la_reg: 0.0082, norm_reg: 0.0000, obj: 0.5079\n",
      "[f] improve*: 0.630\n",
      "\n"
     ]
    }
   ],
   "source": [
    "verbose = True\n",
    "\n",
    "# train our model\n",
    "print('training lookahead:')\n",
    "f_model = PolyRegression(degree, 1)\n",
    "f = pred.PredModel(d, model = f_model, reg_type='none', alpha=0., lr=lr_f, num_iter=num_iter_f, num_iter_init=num_iter_init)\n",
    "g_model = PolyRegression(degree, 1)\n",
    "u = uncert.Bootstrap(d, model = g_model, alpha=0., num_gs=num_gs, z_score=z_score, lr=lr_g, num_iter=num_iter_g)\n",
    "#u = uncert.QuantReg(d, alpha = 0., tau = [0.05, 0.95], lr=lr_g, num_iter=num_iter_g)\n",
    "#u = uncert.BootstrapResid(d, f, alpha=0.0, num_gs=num_gs, z_score=z_score, lr=lr_g, num_iter=num_iter_g)\n",
    "h = prop.PropModel(random_state=seed)\n",
    "model = Lookahead(f, u, h, lam=lam, eta=eta, mask=mask, ground_truth_model=fstar)\n",
    "_, _ = model.train(x_trn, y_trn, num_cycles=num_cycles, random_state=seed, verbose=verbose)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 211,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "baseline:\n",
      "\ttrn\tts\tactv\tall\n",
      "mse\t0.0410\t0.1175\t0.0624\t0.0624\n",
      "mae\t0.1473\t0.3236\t0.1966\t0.1966\n",
      "imprv\t-35.1306\t-26.2806\t-32.6526\t-32.6526\n",
      "imprt\t0.0000\t0.0000\t0.0000\t0.0000\n",
      "\n",
      "lookahead:\n",
      "\ttrn\tts\tactv\tall\n",
      "mse\t0.4766\t0.4558\t0.4708\t0.4708\n",
      "mae\t0.5722\t0.5324\t0.5611\t0.5611\n",
      "imprv\t0.6302\t0.6041\t0.6229\t0.6229\n",
      "imprt\t0.5000\t0.5714\t0.5200\t0.5200\n",
      "\n",
      "\ttrn'\ttst'\tactv'\tall'\tall\n",
      "contn\t0.222\t0.429\t0.280\t0.280\t0.800\n",
      "intrsz\t1.654\t1.469\t1.602\t1.602\t0.442\n",
      "\n"
     ]
    }
   ],
   "source": [
    "# evaluate\n",
    "perf_base = get_perf(model_base, xs, ys, eta, mask)\n",
    "perf_la = get_perf(model, xs, ys, eta, mask, uncert=True)\n",
    "\n",
    "print('\\nbaseline:')\n",
    "print_perf(perf_base)\n",
    "print('lookahead:')\n",
    "print_perf(perf_la, uncert=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 212,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\nRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 360x216 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "SAVE_FIG = False\n",
    "fname = 'synth_3_v3'\n",
    "\n",
    "plt.rcParams['figure.figsize'] = (5.0, 3.0)\n",
    "\n",
    "ax = plt.axes()\n",
    "plot_quad(ax, model, x, y, x_plot, y_plot, eta, mask, lw=2)\n",
    "ax.set_xlim([-2.2, 2.2])\n",
    "ax.set_ylim([-4, 1])\n",
    "plt.title(r'$\\eta={}$'.format(eta))\n",
    "\n",
    "axins = inset_axes(ax, width=\"40%\", height=\"40%\", loc=4)\n",
    "plot_quad(axins, model_base, x, y, x_plot, y_plot, eta, mask, osz=10, olw=0.5)\n",
    "axins.set_xlim([-2.5, 6])\n",
    "axins.set_ylim([-30, 4.0])\n",
    "\n",
    "if SAVE_FIG:\n",
    "    plt.draw()\n",
    "    plt.savefig(fname+'.eps', format='eps', bbox_inches='tight')\n",
    "    plt.savefig(fname+'.png', format='png', dpi=300, bbox_inches='tight')\n",
    "    print('saved ' + fname)\n",
    "else:\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "anaconda-cloud": {},
  "kernelspec": {
   "display_name": "Python 3",
   "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.6.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
