{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 44,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<module 'pred' from '/mnt/c/Users/nir897/Documents/proj/robias/code/lookahead/pred.py'>"
      ]
     },
     "execution_count": 44,
     "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": 45,
   "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": 46,
   "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": 59,
   "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": 48,
   "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 = 1.25\n",
    "n = 25   "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 49,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x7f6029419080>]"
      ]
     },
     "execution_count": 49,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\nEQlw/w/amWO6zF2bSgAAAABJRU5ErkJggg==\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(-3.0, 3.0, 1000)[:, np.newaxis]\n",
    "y_plot = fstar.predict(x_plot)\n",
    "plt.plot(x_plot, y_plot)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 50,
   "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": 51,
   "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.05\n",
    "num_iter_init = 1000 #for initial f\n",
    "num_iter_f = 100 #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.05\n",
    "num_iter_g = 3000 #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": 52,
   "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*: -0.539\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-0.5392\t-0.2610\t-0.4613\t-0.4613\n",
      "imprt\t0.0556\t0.1429\t0.0800\t0.0800\n",
      "\n"
     ]
    }
   ],
   "source": [
    "# train baseline\n",
    "verbose = True\n",
    "\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": 53,
   "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*: -0.428\n",
      "\n",
      "t: 1\n",
      "[h] n_eff: 5.58, w_sum: 1.54\n",
      "[u] loss: 0.0089, norm_reg: 0.0000, obj: 0.0089\n",
      "[u] size: 4.853, contain*: 1.000\n",
      "[f] mse: 0.2113, la_reg: 0.0362, norm_reg: 0.0000, obj: 0.3562\n",
      "[f] improve*: 1.344\n",
      "\n",
      "t: 2\n",
      "[h] n_eff: 4.38, w_sum: 4.60\n",
      "[u] loss: 0.0077, norm_reg: 0.0000, obj: 0.0077\n",
      "[u] size: 1.437, contain*: 1.000\n",
      "[f] mse: 0.1630, la_reg: 0.0233, norm_reg: 0.0000, obj: 0.2563\n",
      "[f] improve*: 1.222\n",
      "\n",
      "t: 3\n",
      "[h] n_eff: 4.51, w_sum: 3.64\n",
      "[u] loss: 0.0078, norm_reg: 0.0000, obj: 0.0078\n",
      "[u] size: 1.835, contain*: 1.000\n",
      "[f] mse: 0.1689, la_reg: 0.0232, norm_reg: 0.0000, obj: 0.2619\n",
      "[f] improve*: 1.239\n",
      "\n",
      "t: 4\n",
      "[h] n_eff: 4.49, w_sum: 3.75\n",
      "[u] loss: 0.0078, norm_reg: 0.0000, obj: 0.0078\n",
      "[u] size: 1.784, contain*: 1.000\n",
      "[f] mse: 0.1666, la_reg: 0.0236, norm_reg: 0.0000, obj: 0.2612\n",
      "[f] improve*: 1.233\n",
      "\n",
      "t: 5\n",
      "[h] n_eff: 4.50, w_sum: 3.71\n",
      "[u] loss: 0.0078, norm_reg: 0.0000, obj: 0.0078\n",
      "[u] size: 1.801, contain*: 1.000\n",
      "[f] mse: 0.1671, la_reg: 0.0236, norm_reg: 0.0000, obj: 0.2614\n",
      "[f] improve*: 1.235\n",
      "\n"
     ]
    }
   ],
   "source": [
    "# train our model\n",
    "verbose = True\n",
    "\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": 54,
   "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-0.5392\t-0.2610\t-0.4613\t-0.4613\n",
      "imprt\t0.0556\t0.1429\t0.0800\t0.0800\n",
      "\n",
      "lookahead:\n",
      "\ttrn\tts\tactv\tall\n",
      "mse\t0.1669\t0.1797\t0.1705\t0.1705\n",
      "mae\t0.3377\t0.3359\t0.3372\t0.3372\n",
      "imprv\t1.2347\t1.0653\t1.1873\t1.1873\n",
      "imprt\t0.7222\t0.7143\t0.7200\t0.7200\n",
      "\n",
      "\ttrn'\ttst'\tactv'\tall'\tall\n",
      "contn\t1.000\t1.000\t1.000\t1.000\t1.000\n",
      "intrsz\t1.797\t1.551\t1.728\t1.728\t0.393\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": 81,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 360x216 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "SAVE_FIG = False\n",
    "fname = 'synth_2_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, 3.3])\n",
    "axins.set_ylim([-7, 1.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": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "collapsed": true
   },
   "outputs": [],
   "source": []
  },
  {
   "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
}
