{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<module 'pred' from '/mnt/c/Users/nir897/Documents/proj/robias/code/lookahead/pred.py'>"
      ]
     },
     "execution_count": 1,
     "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": 2,
   "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": 3,
   "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": 4,
   "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 = 0.75\n",
    "n = 25   "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x7f69087c7f98>]"
      ]
     },
     "execution_count": 5,
     "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": 23,
   "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": 24,
   "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 = 200 #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": 25,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "training baseline:\n",
      "t: 0\n",
      "[f] mse: 0.0403, la_reg: 0.0000, norm_reg: 0.0000, obj: 0.0403\n",
      "[f] improve*: 1.294\n",
      "\n",
      "\ttrn\tts\tactv\tall\n",
      "mse\t0.0403\t0.1220\t0.0632\t0.0632\n",
      "mae\t0.1479\t0.3303\t0.1990\t0.1990\n",
      "imprv\t1.2943\t1.1093\t1.2425\t1.2425\n",
      "imprt\t0.9444\t0.8571\t0.9200\t0.9200\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": 26,
   "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*: 1.327\n",
      "\n",
      "t: 1\n",
      "[h] n_eff: 4.26, w_sum: 3.49\n",
      "[u] loss: 0.0076, norm_reg: 0.0000, obj: 0.0076\n",
      "[u] size: 1.580, contain*: 1.000\n",
      "[f] mse: 0.0405, la_reg: 0.0189, norm_reg: 0.0000, obj: 0.1162\n",
      "[f] improve*: 1.308\n",
      "\n",
      "t: 2\n",
      "[h] n_eff: 4.36, w_sum: 3.87\n",
      "[u] loss: 0.0077, norm_reg: 0.0000, obj: 0.0077\n",
      "[u] size: 1.596, contain*: 1.000\n",
      "[f] mse: 0.0401, la_reg: 0.0191, norm_reg: 0.0000, obj: 0.1164\n",
      "[f] improve*: 1.292\n",
      "\n",
      "t: 3\n",
      "[h] n_eff: 4.39, w_sum: 3.86\n",
      "[u] loss: 0.0077, norm_reg: 0.0000, obj: 0.0077\n",
      "[u] size: 1.639, contain*: 1.000\n",
      "[f] mse: 0.0402, la_reg: 0.0191, norm_reg: 0.0000, obj: 0.1167\n",
      "[f] improve*: 1.296\n",
      "\n",
      "t: 4\n",
      "[h] n_eff: 4.39, w_sum: 3.91\n",
      "[u] loss: 0.0077, norm_reg: 0.0000, obj: 0.0077\n",
      "[u] size: 1.623, contain*: 1.000\n",
      "[f] mse: 0.0402, la_reg: 0.0191, norm_reg: 0.0000, obj: 0.1166\n",
      "[f] improve*: 1.301\n",
      "\n",
      "t: 5\n",
      "[h] n_eff: 4.39, w_sum: 3.95\n",
      "[u] loss: 0.0077, norm_reg: 0.0000, obj: 0.0077\n",
      "[u] size: 1.607, contain*: 1.000\n",
      "[f] mse: 0.0402, la_reg: 0.0191, norm_reg: 0.0000, obj: 0.1166\n",
      "[f] improve*: 1.294\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": 27,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "baseline:\n",
      "\ttrn\tts\tactv\tall\n",
      "mse\t0.0403\t0.1220\t0.0632\t0.0632\n",
      "mae\t0.1479\t0.3303\t0.1990\t0.1990\n",
      "imprv\t1.2943\t1.1093\t1.2425\t1.2425\n",
      "imprt\t0.9444\t0.8571\t0.9200\t0.9200\n",
      "\n",
      "lookahead:\n",
      "\ttrn\tts\tactv\tall\n",
      "mse\t0.0402\t0.1232\t0.0634\t0.0634\n",
      "mae\t0.1471\t0.3310\t0.1986\t0.1986\n",
      "imprv\t1.2936\t1.1083\t1.2417\t1.2417\n",
      "imprt\t0.9444\t0.8571\t0.9200\t0.9200\n",
      "\n",
      "\ttrn'\ttst'\tactv'\tall'\tall\n",
      "contn\t1.000\t1.000\t1.000\t1.000\t1.000\n",
      "intrsz\t1.630\t1.431\t1.575\t1.575\t0.395\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": 40,
   "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, label='_nolegend_')\n",
    "        ax.plot(x_plot.flatten(), l_pred.flatten(), \"tab:green\", linewidth=1.0, alpha=0.5, zorder=0, label='_nolegend_')\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": 54,
   "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_1_v3_legend2'\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.2, 2.2])\n",
    "axins.set_ylim([-4, 1.0])\n",
    "\n",
    "ax.legend([r'$f^*$',r'$f$',r'$g$',r'$(x,y)$',r\"$(x',y')$\"],fontsize=8)\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
}
