{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "%matplotlib inline"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# One"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# ZOJ"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dataset 20newsgroup, train_samples=5657, val_samples=5657, test_samples=7532, n_features=130107, n_classes=20\n",
      "logistic_20news with ZOJ, mu = 0.01, beta = 0.02 and T = 5\n",
      "o_step=0 (8.21e-02s) Val loss: 2.1360e+00, Val Acc: 48.65%\n",
      "          Test loss: 2.2243e+00, Test Acc: 43.26%\n",
      "          l2_hp norm: 3.1284e+00\n",
      "          l2_hp: -3.1284e+00\n",
      "o_step=50 (7.82e-02s) Val loss: 5.3442e-01, Val Acc: 85.35%\n",
      "          Test loss: 8.4850e-01, Test Acc: 75.86%\n",
      "          l2_hp norm: 1.6243e+00\n",
      "          l2_hp: -1.6243e+00\n",
      "o_step=100 (7.81e-02s) Val loss: 5.2221e-01, Val Acc: 85.75%\n",
      "          Test loss: 8.4593e-01, Test Acc: 76.08%\n",
      "          l2_hp norm: 1.6282e+00\n",
      "          l2_hp: -1.6282e+00\n",
      "o_step=150 (7.81e-02s) Val loss: 5.1886e-01, Val Acc: 85.86%\n",
      "          Test loss: 8.4775e-01, Test Acc: 76.23%\n",
      "          l2_hp norm: 1.6295e+00\n",
      "          l2_hp: -1.6295e+00\n",
      "o_step=200 (7.80e-02s) Val loss: 5.1756e-01, Val Acc: 85.95%\n",
      "          Test loss: 8.4976e-01, Test Acc: 76.30%\n",
      "          l2_hp norm: 1.6300e+00\n",
      "          l2_hp: -1.6300e+00\n",
      "o_step=250 (7.81e-02s) Val loss: 5.1693e-01, Val Acc: 86.07%\n",
      "          Test loss: 8.5143e-01, Test Acc: 76.38%\n",
      "          l2_hp norm: 1.6299e+00\n",
      "          l2_hp: -1.6299e+00\n",
      "o_step=300 (7.81e-02s) Val loss: 5.1657e-01, Val Acc: 86.14%\n",
      "          Test loss: 8.5271e-01, Test Acc: 76.45%\n",
      "          l2_hp norm: 1.6300e+00\n",
      "          l2_hp: -1.6300e+00\n",
      "o_step=350 (7.80e-02s) Val loss: 5.1631e-01, Val Acc: 86.23%\n",
      "          Test loss: 8.5366e-01, Test Acc: 76.42%\n",
      "          l2_hp norm: 1.6299e+00\n",
      "          l2_hp: -1.6299e+00\n",
      "o_step=400 (7.81e-02s) Val loss: 5.1608e-01, Val Acc: 86.30%\n",
      "          Test loss: 8.5432e-01, Test Acc: 76.53%\n",
      "          l2_hp norm: 1.6299e+00\n",
      "          l2_hp: -1.6299e+00\n",
      "o_step=450 (7.81e-02s) Val loss: 5.1586e-01, Val Acc: 86.35%\n",
      "          Test loss: 8.5476e-01, Test Acc: 76.55%\n",
      "          l2_hp norm: 1.6298e+00\n",
      "          l2_hp: -1.6298e+00\n",
      "o_step=499 (7.81e-02s) Val loss: 5.1563e-01, Val Acc: 86.37%\n",
      "          Test loss: 8.5501e-01, Test Acc: 76.63%\n",
      "          l2_hp norm: 1.6297e+00\n",
      "          l2_hp: -1.6297e+00\n",
      "HPO ended in 3.91e+01 seconds\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from itertools import repeat\n",
    "\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import hypergrad as hg\n",
    "import numpy as np\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.preprocessing import MinMaxScaler\n",
    "from sklearn.datasets import fetch_20newsgroups_vectorized\n",
    "import time\n",
    "import pdb\n",
    "\n",
    "import torch.nn.functional as F\n",
    "\n",
    "experiment_name = 'logistic_20news'\n",
    "algs = {'ZOJ': hg.hgvzoj, 'HOZOG': hg.hozog}\n",
    "\n",
    "# Helper functions to deal with cuda\n",
    "cuda = True and torch.cuda.is_available()\n",
    "\n",
    "default_tensor_str = 'torch.cuda.FloatTensor' if cuda else 'torch.FloatTensor'\n",
    "\n",
    "kwargs = {'num_workers': 4, 'pin_memory': True} if cuda else {}\n",
    "torch.set_default_tensor_type(default_tensor_str)\n",
    "\n",
    "\n",
    "# torch.multiprocessing.set_start_method('forkserver')\n",
    "\n",
    "def frnp(x): return torch.from_numpy(x).cuda().float() if cuda else torch.from_numpy(x).float()\n",
    "\n",
    "\n",
    "def tonp(x, cuda=cuda): return x.detach().cpu().numpy() if cuda else x.detach().numpy()\n",
    "\n",
    "\n",
    "seed = 0\n",
    "torch.manual_seed(seed)\n",
    "np.random.seed(seed)\n",
    "\n",
    "# load twentynews and preprocess\n",
    "val_size = 0.5\n",
    "X, y = fetch_20newsgroups_vectorized(subset='train', return_X_y=True,\n",
    "                                     # remove=('headers', 'footers', 'quotes')\n",
    "                                     )\n",
    "x_test, y_test = fetch_20newsgroups_vectorized(subset='test', return_X_y=True,\n",
    "                                               # remove=('headers', 'footers', 'quotes')\n",
    "                                               )\n",
    "\n",
    "x_train, x_val, y_train, y_val = train_test_split(X, y, stratify=y, test_size=val_size)\n",
    "\n",
    "train_samples, n_features = x_train.shape\n",
    "test_samples, n_features = x_test.shape\n",
    "val_samples, n_features = x_val.shape\n",
    "n_classes = np.unique(y_train).shape[0]\n",
    "\n",
    "print('Dataset 20newsgroup, train_samples=%i, val_samples=%i, test_samples=%i, n_features=%i, n_classes=%i'\n",
    "      % (train_samples, val_samples, test_samples, n_features, n_classes))\n",
    "\n",
    "ys = [frnp(y_train).long(), frnp(y_val).long(), frnp(y_test).long()] # long tensors used for ys\n",
    "xs = [x_train, x_val, x_test]\n",
    "\n",
    "\n",
    "def from_sparse(x):\n",
    "    x = x.tocoo() # tocoo() is a numpy function?\n",
    "    values = x.data\n",
    "    indices = np.vstack((x.row, x.col))\n",
    "\n",
    "    i = torch.LongTensor(indices)\n",
    "    v = torch.FloatTensor(values)\n",
    "    shape = x.shape\n",
    "\n",
    "    return torch.sparse.FloatTensor(i, v, torch.Size(shape)) # float tensors used for xs\n",
    "\n",
    "\n",
    "if cuda:\n",
    "    xs = [from_sparse(x).cuda() for x in xs]\n",
    "else:\n",
    "    xs = [from_sparse(x) for x in xs]\n",
    "\n",
    "x_train, x_val, x_test = xs\n",
    "y_train, y_val, y_test = ys\n",
    "\n",
    "\n",
    "class CustomTensorIterator:\n",
    "    def __init__(self, tensor_list, batch_size, **loader_kwargs):\n",
    "        self.loader = DataLoader(TensorDataset(*tensor_list), batch_size=batch_size, **loader_kwargs)\n",
    "        self.iterator = iter(self.loader)\n",
    "\n",
    "    def __next__(self, *args):\n",
    "        try:\n",
    "            idx = next(self.iterator)\n",
    "        except StopIteration:\n",
    "            self.iterator = iter(self.loader)\n",
    "            idx = next(self.iterator)\n",
    "        return idx\n",
    "\n",
    "\n",
    "# torch.DataLoader has problems with sparse tensor on GPU\n",
    "train_batch_size = len(y_train)\n",
    "val_batch_size = len(y_val)\n",
    "\n",
    "iterators = []\n",
    "for bs, x, y in [(train_batch_size, x_train, y_train), (val_batch_size, x_val, y_val)]:\n",
    "    if bs < len(y):\n",
    "        print('making iterator with batch size ', bs)\n",
    "        iterators.append(CustomTensorIterator([x, y], batch_size=bs, shuffle=True, **kwargs))\n",
    "    else: # only else statement will be executed because of the way they defined train_batch_size and val_batch_size\n",
    "        iterators.append(repeat([x, y]))\n",
    "\n",
    "train_iterator, val_iterator = iterators\n",
    "\n",
    "# HPO set up\n",
    "prefix = 'ZOJ'\n",
    "reg_type = 'ONE' # or EACH\n",
    "mu = 0.01\n",
    "n_steps = 500\n",
    "outer_lr, outer_mu = 0.02, 0.0  # nice with 100.0, 0.0 (torch.SGD) tested with T, K = 5, 10 and CG\n",
    "inner_lr, inner_mu = 100.0, 0.9  # nice with 100., 0.9 (HeavyBall) tested with T, K = 5, 10 and CG\n",
    "T, K = 5, 10\n",
    "tol = 1e-12\n",
    "warm_start = True\n",
    "bias = False  # without bias outer_lr can be bigger (much faster convergence)\n",
    "\n",
    "hyperAlgo = algs[prefix]\n",
    "\n",
    "train_log_interval = 100\n",
    "val_log_interval = 50\n",
    "print(experiment_name + ' with ' + prefix + ', mu = ' + str(mu) + ', beta = ' + str(outer_lr) + ' and T = ' + str(T))\n",
    "\n",
    "if reg_type=='EACH':\n",
    "    l2_reg_params = torch.zeros(n_features).requires_grad_(True)  # one hp per feature\n",
    "    l1_reg_params = (-1. * torch.ones(n_features)).requires_grad_(True)\n",
    "else:\n",
    "    l2_reg_params = (-1.*torch.ones(1)).requires_grad_(True)  # one l2 hp only (best when really low)\n",
    "    l1_reg_params = (0. * torch.ones(1)).requires_grad_(True)  # one l1 hp only (best when really low)\n",
    "\n",
    "\n",
    "hparams = [l2_reg_params]\n",
    "\n",
    "ones_dxc = torch.ones(n_features, n_classes)\n",
    "\n",
    "\n",
    "def reg_f(params, l2_reg_params, l1_reg_params=None):\n",
    "    r = 0.5 * ((params[0] ** 2) * torch.exp(l2_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    if l1_reg_params is not None:\n",
    "        r += (params[0].abs() * torch.exp(l1_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    return r\n",
    "\n",
    "\n",
    "outer_opt = torch.optim.SGD(lr=outer_lr, momentum=outer_mu, params=hparams)\n",
    "# outer_opt = torch.optim.Adam(lr=0.01, params=hparams)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "val_losses, val_accs = [], []\n",
    "test_losses, test_accs = [], []\n",
    "\n",
    "w = torch.zeros(n_features, n_classes).requires_grad_(True)\n",
    "parameters = [w]\n",
    "\n",
    "if bias:\n",
    "    b = torch.zeros(n_classes).requires_grad_(True)\n",
    "    parameters.append(b)\n",
    "\n",
    "initial_parameters = []\n",
    "for param in parameters:\n",
    "    initial_parameters.append(param.clone())\n",
    "\n",
    "def out_f(x, params):\n",
    "    out = x @ params[0]\n",
    "    out += params[1] if len(params) == 2 else 0\n",
    "    return out\n",
    "\n",
    "\n",
    "def train_loss(params, hparams, data):\n",
    "    x_mb, y_mb = data\n",
    "    out = out_f(x_mb, params)\n",
    "    return F.cross_entropy(out, y_mb) + reg_f(params, *hparams)\n",
    "\n",
    "\n",
    "def val_loss(params, hparams, more=False, verbose=False):\n",
    "    x_mb, y_mb = next(val_iterator)\n",
    "    out = out_f(x_mb, params)\n",
    "    val_loss = F.cross_entropy(out, y_mb)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y_mb.view_as(pred)).sum().item() / len(y_mb)\n",
    "\n",
    "    #val_losses.append(tonp(val_loss)) # get rid of it\n",
    "    #val_accs.append(acc) # get rid of it\n",
    "\n",
    "    if more:\n",
    "        return val_loss, acc\n",
    "    else:\n",
    "        return val_loss\n",
    "\n",
    "\n",
    "def eval(params, x, y):\n",
    "    out = out_f(x, params)\n",
    "    loss = F.cross_entropy(out, y)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y.view_as(pred)).sum().item() / len(y)\n",
    "\n",
    "    return loss, acc\n",
    "\n",
    "inner_opt = torch.optim.SGD(lr=inner_lr, momentum=inner_mu, params=parameters)\n",
    "\n",
    "def update_tensor_grads(params, grads):\n",
    "    for l, g in zip(params, grads):\n",
    "        if l.grad is None:\n",
    "            l.grad = torch.zeros_like(l)\n",
    "        if g is not None:\n",
    "            l.grad += g\n",
    "\n",
    "def inner_solver(hparams, steps, optim, params0=None):\n",
    "\n",
    "    if params0 is not None:\n",
    "        for param, param0 in zip(parameters, params0):\n",
    "            param.data = param0.data\n",
    "\n",
    "    for t in range(steps):\n",
    "        data = next(train_iterator)\n",
    "        loss = train_loss(parameters, hparams, data)\n",
    "        optim.zero_grad()\n",
    "        grads = torch.autograd.grad(loss, parameters)\n",
    "        update_tensor_grads(parameters, grads)\n",
    "        optim.step()\n",
    "        #print('inner grad norm:', torch.norm(grads[0]).item())\n",
    "        #print('current train loss:', loss.item())\n",
    "\n",
    "    return [param.clone() for param in parameters]\n",
    "\n",
    "\n",
    "# final_params = inner_solver(hparams, 1000, initial_parameters)\n",
    "# pdb.set_trace()\n",
    "\n",
    "params_history = []\n",
    "total_time = 0\n",
    "running_time = []\n",
    "hg_norms = []\n",
    "for o_step in range(n_steps):\n",
    "    start_time = time.time()\n",
    "\n",
    "    final_params = inner_solver(hparams, T, optim=inner_opt, params0=initial_parameters)\n",
    "\n",
    "    # pdb.set_trace()\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    _, vloss, vacc = hyperAlgo(final_params, hparams, val_loss, inner_solver, params0=initial_parameters, optim=inner_opt, mu=mu, T=T, p=1, set_grad=True, more=True)\n",
    "    outer_opt.step()\n",
    "    val_losses.append(tonp(vloss))\n",
    "    val_accs.append(vacc)\n",
    "\n",
    "    # for hparam in hparams:\n",
    "    #     hparam.data.clamp_(min=-5.0, max=5.0)\n",
    "    #pdb.set_trace()\n",
    "\n",
    "    for init_p, new_p in zip(initial_parameters, final_params):\n",
    "        if warm_start:\n",
    "            init_p.data = new_p\n",
    "        else:\n",
    "            init_p.data = torch.zeros_like(init_p)\n",
    "\n",
    "    iter_time = time.time() - start_time\n",
    "    total_time += iter_time\n",
    "    running_time.append(total_time)\n",
    "    hg_norms.append(torch.norm(hparams[0]))\n",
    "\n",
    "    if o_step % val_log_interval == 0 or o_step == n_steps - 1:\n",
    "        test_loss, test_acc = eval(final_params, x_test, y_test)\n",
    "        test_losses.append(test_loss)\n",
    "        test_accs.append(test_acc)\n",
    "        #print('step train los is: ', step_train_loss)\n",
    "        print('o_step={} ({:.2e}s) Val loss: {:.4e}, Val Acc: {:.2f}%'.format(o_step, iter_time, val_losses[-1],\n",
    "                                                                              100 * val_accs[-1]))\n",
    "        print('          Test loss: {:.4e}, Test Acc: {:.2f}%'.format(test_loss, 100 * test_acc))\n",
    "        print('          l2_hp norm: {:.4e}'.format(torch.norm(hparams[0])))\n",
    "        print('          l2_hp: {:.4e}'.format(hparams[0].item()))\n",
    "        if len(hparams) == 2:\n",
    "            print('          l1_hp : ', torch.norm(hparams[1]))\n",
    "\n",
    "print('HPO ended in {:.2e} seconds\\n'.format(total_time))\n",
    "\n",
    "plt.title('val_accuracy')\n",
    "plt.plot(running_time, val_accs)\n",
    "#figname = 'logistic_plots/VAL_ACC_' + experiment_name + '_' + prefix + '_' + reg_type + '_' + str(T) + '_' + str(mu) + '_' + str(outer_lr) + '.png'\n",
    "#plt.savefig(figname, bbox_inches='tight')\n",
    "#plt.close()\n",
    "plt.show()\n",
    "\n",
    "\n",
    "plt.title('test_accuracy')\n",
    "plt.plot(test_accs)\n",
    "# figname = 'logistic_plots/TEST_ACC_' + experiment_name + '_' + prefix + '_' + reg_type + '_' + str(T) + '_' + str(mu) + '_' + str(outer_lr) + '.png'\n",
    "# plt.savefig(figname, bbox_inches='tight')\n",
    "# plt.close()\n",
    "plt.show()\n",
    "norm_zoj = hg_norms\n",
    "val_zoj = val_accs\n",
    "run_zoj = running_time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {
    "collapsed": true,
    "jupyter": {
     "outputs_hidden": true
    }
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[0.4864769312356373,\n",
       " 0.6966590065405692,\n",
       " 0.7599434329149726,\n",
       " 0.795474633197808,\n",
       " 0.8136821636910023,\n",
       " 0.8251723528371928,\n",
       " 0.831182605621354,\n",
       " 0.8361322255612516,\n",
       " 0.8386070355312003,\n",
       " 0.8409050733604384,\n",
       " 0.8426727947675446,\n",
       " 0.8433798833303872,\n",
       " 0.8451476047374934,\n",
       " 0.8458546933003359,\n",
       " 0.8453243768782039,\n",
       " 0.8463850097224678,\n",
       " 0.8462082375817571,\n",
       " 0.8465617818631783,\n",
       " 0.8470920982853102,\n",
       " 0.8476224147074422,\n",
       " 0.8486830475517059,\n",
       " 0.8485062754109952,\n",
       " 0.8488598196924165,\n",
       " 0.8492133639738377,\n",
       " 0.8492133639738377,\n",
       " 0.8500972246773908,\n",
       " 0.8499204525366802,\n",
       " 0.8499204525366802,\n",
       " 0.850981085380944,\n",
       " 0.8516881739437865,\n",
       " 0.8516881739437865,\n",
       " 0.8516881739437865,\n",
       " 0.8516881739437865,\n",
       " 0.8516881739437865,\n",
       " 0.851864946084497,\n",
       " 0.851864946084497,\n",
       " 0.8520417182252077,\n",
       " 0.8522184903659183,\n",
       " 0.8520417182252077,\n",
       " 0.8522184903659183,\n",
       " 0.8522184903659183,\n",
       " 0.8522184903659183,\n",
       " 0.852395262506629,\n",
       " 0.8525720346473395,\n",
       " 0.8525720346473395,\n",
       " 0.8527488067880502,\n",
       " 0.8529255789287609,\n",
       " 0.8531023510694714,\n",
       " 0.8532791232101821,\n",
       " 0.8534558953508927,\n",
       " 0.8534558953508927,\n",
       " 0.8536326674916033,\n",
       " 0.8534558953508927,\n",
       " 0.8536326674916033,\n",
       " 0.8538094396323139,\n",
       " 0.8538094396323139,\n",
       " 0.8539862117730246,\n",
       " 0.8541629839137352,\n",
       " 0.8541629839137352,\n",
       " 0.8546933003358671,\n",
       " 0.8548700724765776,\n",
       " 0.8550468446172883,\n",
       " 0.855223616757999,\n",
       " 0.8554003888987096,\n",
       " 0.8554003888987096,\n",
       " 0.8555771610394202,\n",
       " 0.8555771610394202,\n",
       " 0.8555771610394202,\n",
       " 0.8554003888987096,\n",
       " 0.856107477461552,\n",
       " 0.856107477461552,\n",
       " 0.8562842496022627,\n",
       " 0.8566377938836839,\n",
       " 0.8566377938836839,\n",
       " 0.8566377938836839,\n",
       " 0.8564610217429733,\n",
       " 0.8564610217429733,\n",
       " 0.8562842496022627,\n",
       " 0.8564610217429733,\n",
       " 0.8566377938836839,\n",
       " 0.8568145660243945,\n",
       " 0.8568145660243945,\n",
       " 0.8568145660243945,\n",
       " 0.8569913381651052,\n",
       " 0.8569913381651052,\n",
       " 0.8569913381651052,\n",
       " 0.8569913381651052,\n",
       " 0.8573448824465264,\n",
       " 0.8573448824465264,\n",
       " 0.8571681103058159,\n",
       " 0.8571681103058159,\n",
       " 0.8569913381651052,\n",
       " 0.8569913381651052,\n",
       " 0.8569913381651052,\n",
       " 0.8571681103058159,\n",
       " 0.8573448824465264,\n",
       " 0.8573448824465264,\n",
       " 0.8575216545872371,\n",
       " 0.8575216545872371,\n",
       " 0.8575216545872371,\n",
       " 0.8575216545872371,\n",
       " 0.8575216545872371,\n",
       " 0.8575216545872371,\n",
       " 0.8575216545872371,\n",
       " 0.8575216545872371,\n",
       " 0.8575216545872371,\n",
       " 0.8575216545872371,\n",
       " 0.8576984267279477,\n",
       " 0.8576984267279477,\n",
       " 0.8576984267279477,\n",
       " 0.8578751988686583,\n",
       " 0.8578751988686583,\n",
       " 0.8578751988686583,\n",
       " 0.8578751988686583,\n",
       " 0.8578751988686583,\n",
       " 0.8578751988686583,\n",
       " 0.8576984267279477,\n",
       " 0.8578751988686583,\n",
       " 0.8580519710093689,\n",
       " 0.8578751988686583,\n",
       " 0.8576984267279477,\n",
       " 0.8578751988686583,\n",
       " 0.8580519710093689,\n",
       " 0.8578751988686583,\n",
       " 0.8580519710093689,\n",
       " 0.8580519710093689,\n",
       " 0.8580519710093689,\n",
       " 0.8580519710093689,\n",
       " 0.8580519710093689,\n",
       " 0.8580519710093689,\n",
       " 0.8580519710093689,\n",
       " 0.8580519710093689,\n",
       " 0.8580519710093689,\n",
       " 0.8578751988686583,\n",
       " 0.8578751988686583,\n",
       " 0.8578751988686583,\n",
       " 0.8576984267279477,\n",
       " 0.8576984267279477,\n",
       " 0.8578751988686583,\n",
       " 0.8580519710093689,\n",
       " 0.8578751988686583,\n",
       " 0.8578751988686583,\n",
       " 0.8578751988686583,\n",
       " 0.8582287431500796,\n",
       " 0.8584055152907901,\n",
       " 0.8584055152907901,\n",
       " 0.8582287431500796,\n",
       " 0.8584055152907901,\n",
       " 0.8584055152907901,\n",
       " 0.8585822874315008,\n",
       " 0.8585822874315008,\n",
       " 0.8585822874315008,\n",
       " 0.8585822874315008,\n",
       " 0.8585822874315008,\n",
       " 0.8585822874315008,\n",
       " 0.8585822874315008,\n",
       " 0.8587590595722114,\n",
       " 0.8587590595722114,\n",
       " 0.8587590595722114,\n",
       " 0.8589358317129221,\n",
       " 0.8589358317129221,\n",
       " 0.8589358317129221,\n",
       " 0.8591126038536326,\n",
       " 0.8591126038536326,\n",
       " 0.8591126038536326,\n",
       " 0.8591126038536326,\n",
       " 0.8591126038536326,\n",
       " 0.8591126038536326,\n",
       " 0.8591126038536326,\n",
       " 0.8591126038536326,\n",
       " 0.8592893759943433,\n",
       " 0.8592893759943433,\n",
       " 0.8592893759943433,\n",
       " 0.8592893759943433,\n",
       " 0.8592893759943433,\n",
       " 0.8591126038536326,\n",
       " 0.8591126038536326,\n",
       " 0.8589358317129221,\n",
       " 0.8589358317129221,\n",
       " 0.8591126038536326,\n",
       " 0.8591126038536326,\n",
       " 0.8592893759943433,\n",
       " 0.8592893759943433,\n",
       " 0.859466148135054,\n",
       " 0.859466148135054,\n",
       " 0.859466148135054,\n",
       " 0.859466148135054,\n",
       " 0.859466148135054,\n",
       " 0.859466148135054,\n",
       " 0.859466148135054,\n",
       " 0.859466148135054,\n",
       " 0.859466148135054,\n",
       " 0.859466148135054,\n",
       " 0.859466148135054,\n",
       " 0.859466148135054,\n",
       " 0.859466148135054,\n",
       " 0.859466148135054,\n",
       " 0.859466148135054,\n",
       " 0.859466148135054,\n",
       " 0.859466148135054,\n",
       " 0.859466148135054,\n",
       " 0.859466148135054,\n",
       " 0.8596429202757645,\n",
       " 0.8596429202757645,\n",
       " 0.8596429202757645,\n",
       " 0.8598196924164752,\n",
       " 0.8598196924164752,\n",
       " 0.8598196924164752,\n",
       " 0.8598196924164752,\n",
       " 0.8598196924164752,\n",
       " 0.8599964645571858,\n",
       " 0.8599964645571858,\n",
       " 0.8599964645571858,\n",
       " 0.8599964645571858,\n",
       " 0.8599964645571858,\n",
       " 0.8601732366978964,\n",
       " 0.8601732366978964,\n",
       " 0.860350008838607,\n",
       " 0.860350008838607,\n",
       " 0.860350008838607,\n",
       " 0.860350008838607,\n",
       " 0.860350008838607,\n",
       " 0.860350008838607,\n",
       " 0.8601732366978964,\n",
       " 0.8601732366978964,\n",
       " 0.8601732366978964,\n",
       " 0.8601732366978964,\n",
       " 0.8601732366978964,\n",
       " 0.8601732366978964,\n",
       " 0.860350008838607,\n",
       " 0.860350008838607,\n",
       " 0.860350008838607,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8607035531200283,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8605267809793177,\n",
       " 0.8607035531200283,\n",
       " 0.8607035531200283,\n",
       " 0.8607035531200283,\n",
       " 0.8607035531200283,\n",
       " 0.8607035531200283,\n",
       " 0.8608803252607389,\n",
       " 0.8610570974014495,\n",
       " 0.8610570974014495,\n",
       " 0.8610570974014495,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8612338695421602,\n",
       " 0.8614106416828707,\n",
       " 0.8614106416828707,\n",
       " 0.8615874138235814,\n",
       " 0.8615874138235814,\n",
       " 0.8615874138235814,\n",
       " 0.8614106416828707,\n",
       " 0.8614106416828707,\n",
       " 0.8614106416828707,\n",
       " 0.8614106416828707,\n",
       " 0.8614106416828707,\n",
       " 0.8614106416828707,\n",
       " 0.8614106416828707,\n",
       " 0.8614106416828707,\n",
       " 0.8615874138235814,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.861764185964292,\n",
       " 0.8619409581050026,\n",
       " 0.8621177302457133,\n",
       " 0.8621177302457133,\n",
       " 0.8621177302457133,\n",
       " 0.8621177302457133,\n",
       " 0.8621177302457133,\n",
       " 0.8622945023864239,\n",
       " 0.8622945023864239,\n",
       " 0.8622945023864239,\n",
       " 0.8622945023864239,\n",
       " 0.8622945023864239,\n",
       " 0.8622945023864239,\n",
       " 0.8622945023864239,\n",
       " 0.8622945023864239,\n",
       " 0.8622945023864239,\n",
       " 0.8622945023864239,\n",
       " 0.8622945023864239,\n",
       " 0.8622945023864239,\n",
       " 0.8624712745271346,\n",
       " 0.8624712745271346,\n",
       " 0.8624712745271346,\n",
       " 0.8624712745271346,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8626480466678451,\n",
       " 0.8628248188085558,\n",
       " 0.8628248188085558,\n",
       " 0.8628248188085558,\n",
       " 0.8628248188085558,\n",
       " 0.8628248188085558,\n",
       " 0.8628248188085558,\n",
       " 0.8628248188085558,\n",
       " 0.8630015909492664,\n",
       " 0.8630015909492664,\n",
       " 0.8630015909492664,\n",
       " 0.8630015909492664,\n",
       " 0.8630015909492664,\n",
       " 0.8630015909492664,\n",
       " 0.8630015909492664,\n",
       " 0.8630015909492664,\n",
       " 0.8630015909492664,\n",
       " 0.8630015909492664,\n",
       " 0.8630015909492664,\n",
       " 0.8630015909492664,\n",
       " 0.8630015909492664,\n",
       " 0.8630015909492664,\n",
       " 0.8630015909492664,\n",
       " 0.8630015909492664,\n",
       " 0.8630015909492664,\n",
       " 0.8630015909492664,\n",
       " 0.8630015909492664,\n",
       " 0.863178363089977,\n",
       " 0.863178363089977,\n",
       " 0.863178363089977,\n",
       " 0.863178363089977,\n",
       " 0.863178363089977,\n",
       " 0.863178363089977,\n",
       " 0.863178363089977,\n",
       " 0.8633551352306876,\n",
       " 0.8633551352306876,\n",
       " 0.8633551352306876,\n",
       " 0.8633551352306876,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.8635319073713983,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109,\n",
       " 0.863708679512109]"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "val_zoj"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# HOZOG"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dataset 20newsgroup, train_samples=5657, val_samples=5657, test_samples=7532, n_features=130107, n_classes=20\n",
      "logistic_20news with HOZOG, mu = 0.01, beta = 0.02 and T = 5\n",
      "o_step=0 (7.79e-02s) Val loss: 2.2557e+00, Val Acc: 48.40%\n",
      "          Test loss: 2.3279e+00, Test Acc: 42.98%\n",
      "          l2_hp norm: 2.3609e+00\n",
      "          l2_hp: -2.3609e+00\n",
      "o_step=50 (7.40e-02s) Val loss: 5.4175e-01, Val Acc: 85.22%\n",
      "          Test loss: 8.5170e-01, Test Acc: 75.85%\n",
      "          l2_hp norm: 1.1227e+00\n",
      "          l2_hp: -1.1227e+00\n",
      "o_step=100 (7.40e-02s) Val loss: 5.2538e-01, Val Acc: 85.65%\n",
      "          Test loss: 8.4470e-01, Test Acc: 75.93%\n",
      "          l2_hp norm: 1.1276e+00\n",
      "          l2_hp: -1.1276e+00\n",
      "o_step=150 (7.42e-02s) Val loss: 5.2033e-01, Val Acc: 85.75%\n",
      "          Test loss: 8.4444e-01, Test Acc: 76.12%\n",
      "          l2_hp norm: 1.1296e+00\n",
      "          l2_hp: -1.1296e+00\n",
      "o_step=200 (7.42e-02s) Val loss: 5.1799e-01, Val Acc: 85.93%\n",
      "          Test loss: 8.4506e-01, Test Acc: 76.18%\n",
      "          l2_hp norm: 1.1304e+00\n",
      "          l2_hp: -1.1304e+00\n",
      "o_step=250 (7.41e-02s) Val loss: 5.1665e-01, Val Acc: 86.00%\n",
      "          Test loss: 8.4566e-01, Test Acc: 76.27%\n",
      "          l2_hp norm: 1.1304e+00\n",
      "          l2_hp: -1.1304e+00\n",
      "o_step=300 (7.41e-02s) Val loss: 5.1574e-01, Val Acc: 86.04%\n",
      "          Test loss: 8.4608e-01, Test Acc: 76.33%\n",
      "          l2_hp norm: 1.1305e+00\n",
      "          l2_hp: -1.1305e+00\n",
      "o_step=350 (7.42e-02s) Val loss: 5.1505e-01, Val Acc: 86.16%\n",
      "          Test loss: 8.4629e-01, Test Acc: 76.41%\n",
      "          l2_hp norm: 1.1304e+00\n",
      "          l2_hp: -1.1304e+00\n",
      "o_step=400 (7.41e-02s) Val loss: 5.1447e-01, Val Acc: 86.25%\n",
      "          Test loss: 8.4634e-01, Test Acc: 76.42%\n",
      "          l2_hp norm: 1.1303e+00\n",
      "          l2_hp: -1.1303e+00\n",
      "o_step=450 (7.42e-02s) Val loss: 5.1396e-01, Val Acc: 86.30%\n",
      "          Test loss: 8.4624e-01, Test Acc: 76.42%\n",
      "          l2_hp norm: 1.1301e+00\n",
      "          l2_hp: -1.1301e+00\n",
      "o_step=499 (7.42e-02s) Val loss: 5.1350e-01, Val Acc: 86.34%\n",
      "          Test loss: 8.4604e-01, Test Acc: 76.47%\n",
      "          l2_hp norm: 1.1300e+00\n",
      "          l2_hp: -1.1300e+00\n",
      "HPO ended in 3.71e+01 seconds\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from itertools import repeat\n",
    "\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import hypergrad as hg\n",
    "import numpy as np\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.preprocessing import MinMaxScaler\n",
    "from sklearn.datasets import fetch_20newsgroups_vectorized\n",
    "import time\n",
    "import pdb\n",
    "\n",
    "import torch.nn.functional as F\n",
    "\n",
    "experiment_name = 'logistic_20news'\n",
    "algs = {'ZOJ': hg.hgvzoj, 'HOZOG': hg.hozog}\n",
    "\n",
    "# Helper functions to deal with cuda\n",
    "cuda = True and torch.cuda.is_available()\n",
    "\n",
    "default_tensor_str = 'torch.cuda.FloatTensor' if cuda else 'torch.FloatTensor'\n",
    "\n",
    "kwargs = {'num_workers': 4, 'pin_memory': True} if cuda else {}\n",
    "torch.set_default_tensor_type(default_tensor_str)\n",
    "\n",
    "\n",
    "# torch.multiprocessing.set_start_method('forkserver')\n",
    "\n",
    "def frnp(x): return torch.from_numpy(x).cuda().float() if cuda else torch.from_numpy(x).float()\n",
    "\n",
    "\n",
    "def tonp(x, cuda=cuda): return x.detach().cpu().numpy() if cuda else x.detach().numpy()\n",
    "\n",
    "\n",
    "seed = 0\n",
    "torch.manual_seed(seed)\n",
    "np.random.seed(seed)\n",
    "\n",
    "# load twentynews and preprocess\n",
    "val_size = 0.5\n",
    "X, y = fetch_20newsgroups_vectorized(subset='train', return_X_y=True,\n",
    "                                     # remove=('headers', 'footers', 'quotes')\n",
    "                                     )\n",
    "x_test, y_test = fetch_20newsgroups_vectorized(subset='test', return_X_y=True,\n",
    "                                               # remove=('headers', 'footers', 'quotes')\n",
    "                                               )\n",
    "\n",
    "x_train, x_val, y_train, y_val = train_test_split(X, y, stratify=y, test_size=val_size)\n",
    "\n",
    "train_samples, n_features = x_train.shape\n",
    "test_samples, n_features = x_test.shape\n",
    "val_samples, n_features = x_val.shape\n",
    "n_classes = np.unique(y_train).shape[0]\n",
    "\n",
    "print('Dataset 20newsgroup, train_samples=%i, val_samples=%i, test_samples=%i, n_features=%i, n_classes=%i'\n",
    "      % (train_samples, val_samples, test_samples, n_features, n_classes))\n",
    "\n",
    "ys = [frnp(y_train).long(), frnp(y_val).long(), frnp(y_test).long()] # long tensors used for ys\n",
    "xs = [x_train, x_val, x_test]\n",
    "\n",
    "\n",
    "def from_sparse(x):\n",
    "    x = x.tocoo() # tocoo() is a numpy function?\n",
    "    values = x.data\n",
    "    indices = np.vstack((x.row, x.col))\n",
    "\n",
    "    i = torch.LongTensor(indices)\n",
    "    v = torch.FloatTensor(values)\n",
    "    shape = x.shape\n",
    "\n",
    "    return torch.sparse.FloatTensor(i, v, torch.Size(shape)) # float tensors used for xs\n",
    "\n",
    "\n",
    "if cuda:\n",
    "    xs = [from_sparse(x).cuda() for x in xs]\n",
    "else:\n",
    "    xs = [from_sparse(x) for x in xs]\n",
    "\n",
    "x_train, x_val, x_test = xs\n",
    "y_train, y_val, y_test = ys\n",
    "\n",
    "\n",
    "class CustomTensorIterator:\n",
    "    def __init__(self, tensor_list, batch_size, **loader_kwargs):\n",
    "        self.loader = DataLoader(TensorDataset(*tensor_list), batch_size=batch_size, **loader_kwargs)\n",
    "        self.iterator = iter(self.loader)\n",
    "\n",
    "    def __next__(self, *args):\n",
    "        try:\n",
    "            idx = next(self.iterator)\n",
    "        except StopIteration:\n",
    "            self.iterator = iter(self.loader)\n",
    "            idx = next(self.iterator)\n",
    "        return idx\n",
    "\n",
    "\n",
    "# torch.DataLoader has problems with sparse tensor on GPU\n",
    "train_batch_size = len(y_train)\n",
    "val_batch_size = len(y_val)\n",
    "\n",
    "iterators = []\n",
    "for bs, x, y in [(train_batch_size, x_train, y_train), (val_batch_size, x_val, y_val)]:\n",
    "    if bs < len(y):\n",
    "        print('making iterator with batch size ', bs)\n",
    "        iterators.append(CustomTensorIterator([x, y], batch_size=bs, shuffle=True, **kwargs))\n",
    "    else: # only else statement will be executed because of the way they defined train_batch_size and val_batch_size\n",
    "        iterators.append(repeat([x, y]))\n",
    "\n",
    "train_iterator, val_iterator = iterators\n",
    "\n",
    "# HPO set up\n",
    "prefix = 'HOZOG'\n",
    "reg_type = 'ONE' # or EACH\n",
    "mu = 0.01\n",
    "n_steps = 500\n",
    "outer_lr, outer_mu = 0.02, 0.0  # nice with 100.0, 0.0 (torch.SGD) tested with T, K = 5, 10 and CG\n",
    "inner_lr, inner_mu = 80.0, 0.9  # nice with 100., 0.9 (HeavyBall) tested with T, K = 5, 10 and CG\n",
    "T, K = 5, 10\n",
    "tol = 1e-12\n",
    "warm_start = True\n",
    "bias = False  # without bias outer_lr can be bigger (much faster convergence)\n",
    "\n",
    "hyperAlgo = algs[prefix]\n",
    "\n",
    "train_log_interval = 100\n",
    "val_log_interval = 50\n",
    "print(experiment_name + ' with ' + prefix + ', mu = ' + str(mu) + ', beta = ' + str(outer_lr) + ' and T = ' + str(T))\n",
    "\n",
    "if reg_type=='EACH':\n",
    "    l2_reg_params = torch.zeros(n_features).requires_grad_(True)  # one hp per feature\n",
    "    l1_reg_params = (-1. * torch.ones(n_features)).requires_grad_(True)\n",
    "else:\n",
    "    l2_reg_params = (-1.*torch.ones(1)).requires_grad_(True)  # one l2 hp only (best when really low)\n",
    "    l1_reg_params = (0. * torch.ones(1)).requires_grad_(True)  # one l1 hp only (best when really low)\n",
    "\n",
    "\n",
    "hparams = [l2_reg_params]\n",
    "\n",
    "ones_dxc = torch.ones(n_features, n_classes)\n",
    "\n",
    "\n",
    "def reg_f(params, l2_reg_params, l1_reg_params=None):\n",
    "    r = 0.5 * ((params[0] ** 2) * torch.exp(l2_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    if l1_reg_params is not None:\n",
    "        r += (params[0].abs() * torch.exp(l1_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    return r\n",
    "\n",
    "\n",
    "outer_opt = torch.optim.SGD(lr=outer_lr, momentum=outer_mu, params=hparams)\n",
    "# outer_opt = torch.optim.Adam(lr=0.01, params=hparams)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "val_losses, val_accs = [], []\n",
    "test_losses, test_accs = [], []\n",
    "\n",
    "w = torch.zeros(n_features, n_classes).requires_grad_(True)\n",
    "parameters = [w]\n",
    "\n",
    "if bias:\n",
    "    b = torch.zeros(n_classes).requires_grad_(True)\n",
    "    parameters.append(b)\n",
    "\n",
    "initial_parameters = []\n",
    "for param in parameters:\n",
    "    initial_parameters.append(param.clone())\n",
    "\n",
    "def out_f(x, params):\n",
    "    out = x @ params[0]\n",
    "    out += params[1] if len(params) == 2 else 0\n",
    "    return out\n",
    "\n",
    "\n",
    "def train_loss(params, hparams, data):\n",
    "    x_mb, y_mb = data\n",
    "    out = out_f(x_mb, params)\n",
    "    return F.cross_entropy(out, y_mb) + reg_f(params, *hparams)\n",
    "\n",
    "\n",
    "def val_loss(params, hparams, more=False, verbose=False):\n",
    "    x_mb, y_mb = next(val_iterator)\n",
    "    out = out_f(x_mb, params)\n",
    "    val_loss = F.cross_entropy(out, y_mb)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y_mb.view_as(pred)).sum().item() / len(y_mb)\n",
    "\n",
    "    #val_losses.append(tonp(val_loss)) # get rid of it\n",
    "    #val_accs.append(acc) # get rid of it\n",
    "\n",
    "    if more:\n",
    "        return val_loss, acc\n",
    "    else:\n",
    "        return val_loss\n",
    "\n",
    "\n",
    "def eval(params, x, y):\n",
    "    out = out_f(x, params)\n",
    "    loss = F.cross_entropy(out, y)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y.view_as(pred)).sum().item() / len(y)\n",
    "\n",
    "    return loss, acc\n",
    "\n",
    "inner_opt = torch.optim.SGD(lr=inner_lr, momentum=inner_mu, params=parameters)\n",
    "\n",
    "def update_tensor_grads(params, grads):\n",
    "    for l, g in zip(params, grads):\n",
    "        if l.grad is None:\n",
    "            l.grad = torch.zeros_like(l)\n",
    "        if g is not None:\n",
    "            l.grad += g\n",
    "\n",
    "def inner_solver(hparams, steps, optim, params0=None):\n",
    "\n",
    "    if params0 is not None:\n",
    "        for param, param0 in zip(parameters, params0):\n",
    "            param.data = param0.data\n",
    "\n",
    "    for t in range(steps):\n",
    "        data = next(train_iterator)\n",
    "        loss = train_loss(parameters, hparams, data)\n",
    "        optim.zero_grad()\n",
    "        grads = torch.autograd.grad(loss, parameters)\n",
    "        update_tensor_grads(parameters, grads)\n",
    "        optim.step()\n",
    "        #print('inner grad norm:', torch.norm(grads[0]).item())\n",
    "        #print('current train loss:', loss.item())\n",
    "\n",
    "    return [param.clone() for param in parameters]\n",
    "\n",
    "\n",
    "# final_params = inner_solver(hparams, 1000, initial_parameters)\n",
    "# pdb.set_trace()\n",
    "\n",
    "params_history = []\n",
    "total_time = 0\n",
    "running_time = []\n",
    "for o_step in range(n_steps):\n",
    "    start_time = time.time()\n",
    "\n",
    "    final_params = inner_solver(hparams, T, optim=inner_opt, params0=initial_parameters)\n",
    "\n",
    "    # pdb.set_trace()\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    _, vloss, vacc = hyperAlgo(final_params, hparams, val_loss, inner_solver, params0=initial_parameters, optim=inner_opt, mu=mu, T=T, p=1, set_grad=True, more=True)\n",
    "    outer_opt.step()\n",
    "    val_losses.append(tonp(vloss))\n",
    "    val_accs.append(vacc)\n",
    "\n",
    "    # for hparam in hparams:\n",
    "    #     hparam.data.clamp_(min=-5.0, max=5.0)\n",
    "    #pdb.set_trace()\n",
    "\n",
    "    for init_p, new_p in zip(initial_parameters, final_params):\n",
    "        if warm_start:\n",
    "            init_p.data = new_p\n",
    "        else:\n",
    "            init_p.data = torch.zeros_like(init_p)\n",
    "\n",
    "    iter_time = time.time() - start_time\n",
    "    total_time += iter_time\n",
    "    running_time.append(total_time)\n",
    "\n",
    "    if o_step % val_log_interval == 0 or o_step == n_steps - 1:\n",
    "        test_loss, test_acc = eval(final_params, x_test, y_test)\n",
    "        test_losses.append(test_loss)\n",
    "        test_accs.append(test_acc)\n",
    "        #print('step train los is: ', step_train_loss)\n",
    "        print('o_step={} ({:.2e}s) Val loss: {:.4e}, Val Acc: {:.2f}%'.format(o_step, iter_time, val_losses[-1],\n",
    "                                                                              100 * val_accs[-1]))\n",
    "        print('          Test loss: {:.4e}, Test Acc: {:.2f}%'.format(test_loss, 100 * test_acc))\n",
    "        print('          l2_hp norm: {:.4e}'.format(torch.norm(hparams[0])))\n",
    "        print('          l2_hp: {:.4e}'.format(hparams[0].item()))\n",
    "        if len(hparams) == 2:\n",
    "            print('          l1_hp : ', torch.norm(hparams[1]))\n",
    "\n",
    "print('HPO ended in {:.2e} seconds\\n'.format(total_time))\n",
    "\n",
    "plt.title('val_accuracy')\n",
    "plt.plot(running_time, val_accs)\n",
    "#figname = 'logistic_plots/VAL_ACC_' + experiment_name + '_' + prefix + '_' + reg_type + '_' + str(T) + '_' + str(mu) + '_' + str(outer_lr) + '.png'\n",
    "#plt.savefig(figname, bbox_inches='tight')\n",
    "#plt.close()\n",
    "plt.show()\n",
    "\n",
    "\n",
    "plt.title('test_accuracy')\n",
    "plt.plot(test_accs)\n",
    "# figname = 'logistic_plots/TEST_ACC_' + experiment_name + '_' + prefix + '_' + reg_type + '_' + str(T) + '_' + str(mu) + '_' + str(outer_lr) + '.png'\n",
    "# plt.savefig(figname, bbox_inches='tight')\n",
    "# plt.close()\n",
    "plt.show()\n",
    "\n",
    "val_hozog = val_accs\n",
    "run_hozog = running_time"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# FP"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dataset 20newsgroup, train_samples=5657, val_samples=5657, test_samples=7532, n_features=130107, n_classes=20\n",
      "o_step=0 (1.27e-01s) Val loss: 2.1360e+00, Val Acc: 48.65%\n",
      "          Test loss: 2.2243e+00, Test Acc: 43.26%\n",
      "          l2_hp norm: 1.0284e+00\n",
      "o_step=50 (1.22e-01s) Val loss: 5.3498e-01, Val Acc: 85.35%\n",
      "          Test loss: 8.5343e-01, Test Acc: 75.84%\n",
      "          l2_hp norm: 1.4106e+00\n",
      "o_step=100 (1.23e-01s) Val loss: 5.2517e-01, Val Acc: 85.66%\n",
      "          Test loss: 8.5216e-01, Test Acc: 76.05%\n",
      "          l2_hp norm: 1.4769e+00\n",
      "o_step=150 (1.27e-01s) Val loss: 5.2217e-01, Val Acc: 85.82%\n",
      "          Test loss: 8.5366e-01, Test Acc: 76.19%\n",
      "          l2_hp norm: 1.4658e+00\n",
      "o_step=200 (1.31e-01s) Val loss: 5.2078e-01, Val Acc: 85.81%\n",
      "          Test loss: 8.5498e-01, Test Acc: 76.30%\n",
      "          l2_hp norm: 1.4122e+00\n",
      "o_step=250 (1.33e-01s) Val loss: 5.1986e-01, Val Acc: 85.95%\n",
      "          Test loss: 8.5564e-01, Test Acc: 76.41%\n",
      "          l2_hp norm: 1.3283e+00\n",
      "o_step=300 (1.36e-01s) Val loss: 5.1905e-01, Val Acc: 86.02%\n",
      "          Test loss: 8.5554e-01, Test Acc: 76.47%\n",
      "          l2_hp norm: 1.2199e+00\n",
      "o_step=350 (1.39e-01s) Val loss: 5.1818e-01, Val Acc: 86.04%\n",
      "          Test loss: 8.5466e-01, Test Acc: 76.47%\n",
      "          l2_hp norm: 1.0910e+00\n",
      "o_step=400 (1.44e-01s) Val loss: 5.1719e-01, Val Acc: 86.18%\n",
      "          Test loss: 8.5298e-01, Test Acc: 76.53%\n",
      "          l2_hp norm: 9.4645e-01\n",
      "o_step=450 (1.49e-01s) Val loss: 5.1610e-01, Val Acc: 86.21%\n",
      "          Test loss: 8.5056e-01, Test Acc: 76.55%\n",
      "          l2_hp norm: 7.9504e-01\n",
      "o_step=499 (1.48e-01s) Val loss: 5.1501e-01, Val Acc: 86.26%\n",
      "          Test loss: 8.4764e-01, Test Acc: 76.59%\n",
      "          l2_hp norm: 6.5449e-01\n",
      "HPO ended in 6.70e+01 seconds\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from itertools import repeat\n",
    "\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import hypergrad as hg\n",
    "import numpy as np\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.preprocessing import MinMaxScaler\n",
    "from sklearn.datasets import fetch_20newsgroups_vectorized\n",
    "import time\n",
    "\n",
    "import torch.nn.functional as F\n",
    "\n",
    "\n",
    "# Helper functions to deal with cuda\n",
    "cuda = True and torch.cuda.is_available()\n",
    "\n",
    "default_tensor_str = 'torch.cuda.FloatTensor' if cuda else 'torch.FloatTensor'\n",
    "\n",
    "kwargs = {'num_workers': 1, 'pin_memory': True} if cuda else {}\n",
    "torch.set_default_tensor_type(default_tensor_str)\n",
    "#torch.multiprocessing.set_start_method('forkserver')\n",
    "\n",
    "def frnp(x): return torch.from_numpy(x).cuda().float() if cuda else torch.from_numpy(x).float()\n",
    "def tonp(x, cuda=cuda): return x.detach().cpu().numpy() if cuda else x.detach().numpy()\n",
    "\n",
    "\n",
    "seed = 0\n",
    "torch.manual_seed(seed)\n",
    "np.random.seed(seed)\n",
    "\n",
    "\n",
    "# load twentynews and preprocess\n",
    "val_size = 0.5\n",
    "X, y = fetch_20newsgroups_vectorized(subset='train', return_X_y=True,\n",
    "                                     #remove=('headers', 'footers', 'quotes')\n",
    "                                     )\n",
    "x_test, y_test = fetch_20newsgroups_vectorized(subset='test', return_X_y=True,\n",
    "                                               #remove=('headers', 'footers', 'quotes')\n",
    "                                               )\n",
    "\n",
    "\n",
    "x_train, x_val, y_train, y_val = train_test_split(X, y, stratify=y, test_size=val_size)\n",
    "\n",
    "\n",
    "train_samples, n_features = x_train.shape\n",
    "test_samples, n_features = x_test.shape\n",
    "val_samples, n_features = x_val.shape\n",
    "n_classes = np.unique(y_train).shape[0]\n",
    "\n",
    "print('Dataset 20newsgroup, train_samples=%i, val_samples=%i, test_samples=%i, n_features=%i, n_classes=%i'\n",
    "      % (train_samples, val_samples, test_samples, n_features, n_classes))\n",
    "\n",
    "\n",
    "ys = [frnp(y_train).long(), frnp(y_val).long(), frnp(y_test).long()]\n",
    "xs = [x_train, x_val, x_test]\n",
    "\n",
    "\n",
    "def from_sparse(x):\n",
    "    x = x.tocoo()\n",
    "    values = x.data\n",
    "    indices = np.vstack((x.row, x.col))\n",
    "\n",
    "    i = torch.LongTensor(indices)\n",
    "    v = torch.FloatTensor(values)\n",
    "    shape = x.shape\n",
    "\n",
    "    return torch.sparse.FloatTensor(i, v, torch.Size(shape))\n",
    "\n",
    "\n",
    "if cuda:\n",
    "    xs = [from_sparse(x).cuda() for x in xs]\n",
    "else:\n",
    "    xs = [from_sparse(x) for x in xs]\n",
    "\n",
    "x_train, x_val, x_test = xs\n",
    "y_train, y_val, y_test = ys\n",
    "\n",
    "\n",
    "class CustomTensorIterator:\n",
    "    def __init__(self, tensor_list, batch_size, **loader_kwargs):\n",
    "        self.loader = DataLoader(TensorDataset(*tensor_list), batch_size=batch_size, **loader_kwargs)\n",
    "        self.iterator = iter(self.loader)\n",
    "\n",
    "    def __next__(self, *args):\n",
    "        try:\n",
    "            idx = next(self.iterator)\n",
    "        except StopIteration:\n",
    "            self.iterator = iter(self.loader)\n",
    "            idx = next(self.iterator)\n",
    "        return idx\n",
    "\n",
    "\n",
    "# torch.DataLoader has problems with sparse tensor on GPU\n",
    "train_batch_size = len(y_train)\n",
    "val_batch_size = len(y_val)\n",
    "\n",
    "iterators = []\n",
    "for bs, x, y in [(train_batch_size, x_train, y_train), (val_batch_size, x_val, y_val)]:\n",
    "    if bs < len(y):\n",
    "        print('making iterator with batch size ', bs)\n",
    "        iterators.append(CustomTensorIterator([x, y], batch_size=bs, shuffle=True, **kwargs))\n",
    "    else:\n",
    "        iterators.append(repeat([x, y]))\n",
    "\n",
    "train_iterator, val_iterator = iterators\n",
    "\n",
    "# HPO set up\n",
    "n_steps = 500\n",
    "outer_lr, outer_mu = 100.0, 0.0  # nice with 100.0, 0.0 (torch.SGD) tested with T, K = 5, 10 and CG\n",
    "inner_lr, inner_mu = 100., 0.9   # nice with 100., 0.9 (HeavyBall) tested with T, K = 5, 10 and CG\n",
    "T, K = 5, 10\n",
    "tol = 1e-12\n",
    "warm_start = True\n",
    "bias = False  # without bias outer_lr can be bigger (much faster convergence)\n",
    "\n",
    "train_log_interval = 100\n",
    "val_log_interval = 50\n",
    "\n",
    "#l2_reg_params = torch.zeros(n_features).requires_grad_(True)  # one hp per feature\n",
    "l2_reg_params = (-1.*torch.ones(1)).requires_grad_(True)  # one l2 hp only (best when really low)\n",
    "l1_reg_params = (0.*torch.ones(1)).requires_grad_(True)  # one l1 hp only (best when really low)\n",
    "#l1_reg_params = (-1.*torch.ones(n_features)).requires_grad_(True)\n",
    "\n",
    "hparams = [l2_reg_params]\n",
    "\n",
    "ones_dxc = torch.ones(n_features, n_classes)\n",
    "\n",
    "\n",
    "def reg_f(params, l2_reg_params, l1_reg_params=None):\n",
    "    r = 0.5 * ((params[0] ** 2) * torch.exp(l2_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    if l1_reg_params is not None:\n",
    "        r += (params[0].abs() * torch.exp(l1_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    return r\n",
    "\n",
    "\n",
    "outer_opt = torch.optim.SGD(lr=outer_lr, momentum=outer_mu, params=hparams)\n",
    "#outer_opt = torch.optim.Adam(lr=0.01, params=hparams)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "val_losses, val_accs = [], []\n",
    "test_losses, test_accs = [], []\n",
    "\n",
    "w = torch.zeros(n_features, n_classes).requires_grad_(True)\n",
    "parameters = [w]\n",
    "\n",
    "if bias:\n",
    "    b = torch.zeros(n_classes).requires_grad_(True)\n",
    "    parameters.append(b)\n",
    "\n",
    "\n",
    "def out_f(x, params):\n",
    "    out = x @ params[0]\n",
    "    out += params[1] if len(params) == 2 else 0\n",
    "    return out\n",
    "\n",
    "\n",
    "def train_loss(params, hparams, data):\n",
    "    x_mb, y_mb = data\n",
    "    out = out_f(x_mb,  params)\n",
    "    return F.cross_entropy(out, y_mb) + reg_f(params, *hparams)\n",
    "\n",
    "\n",
    "def val_loss(opt_params, hparams):\n",
    "    x_mb, y_mb = next(val_iterator)\n",
    "    out = out_f(x_mb,  opt_params[:len(parameters)])\n",
    "    val_loss = F.cross_entropy(out, y_mb)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y_mb.view_as(pred)).sum().item() / len(y_mb)\n",
    "\n",
    "    val_losses.append(tonp(val_loss))\n",
    "    val_accs.append(acc)\n",
    "    return val_loss\n",
    "\n",
    "\n",
    "def eval(params, x, y):\n",
    "    out = out_f(x,  params)\n",
    "    loss = F.cross_entropy(out, y)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y.view_as(pred)).sum().item() / len(y)\n",
    "\n",
    "    return loss, acc\n",
    "\n",
    "\n",
    "if inner_mu > 0:\n",
    "    #inner_opt = hg.Momentum(train_loss, inner_lr, inner_mu, data_or_iter=train_iterator)\n",
    "    inner_opt = hg.HeavyBall(train_loss, inner_lr, inner_mu, data_or_iter=train_iterator)\n",
    "else:\n",
    "    inner_opt = hg.GradientDescent(train_loss, inner_lr, data_or_iter=train_iterator)\n",
    "\n",
    "inner_opt_cg = hg.GradientDescent(train_loss, 1., data_or_iter=train_iterator)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "total_time = 0\n",
    "running_time = []\n",
    "for o_step in range(n_steps):\n",
    "    start_time = time.time()\n",
    "\n",
    "    inner_losses = []\n",
    "    if params_history:\n",
    "        params_history = [params_history[-1]]\n",
    "    else:\n",
    "        params_history = [inner_opt.get_opt_params(parameters)]\n",
    "    for t in range(T):\n",
    "        params_history.append(inner_opt(params_history[-1], hparams, create_graph=False))\n",
    "        inner_losses.append(inner_opt.curr_loss)\n",
    "\n",
    "#         if t % train_log_interval == 0 or t == T-1:\n",
    "#             print('t={} loss: {}'.format(t, inner_losses[-1]))\n",
    "\n",
    "    final_params = params_history[-1]\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    #hg.reverse(params_history[-K-1:], hparams, [inner_opt]*K, val_loss)\n",
    "    hg.fixed_point(final_params, hparams, K, inner_opt, val_loss, stochastic=False, tol=tol)\n",
    "    #hg.CG(final_params[:len(parameters)], hparams, K, inner_opt_cg, val_loss, stochastic=False, tol=tol)\n",
    "    outer_opt.step()\n",
    "\n",
    "    for p, new_p in zip(parameters, final_params[:len(parameters)]):\n",
    "        if warm_start:\n",
    "            p.data = new_p\n",
    "        else:\n",
    "            p.data = torch.zeros_like(p)\n",
    "\n",
    "    iter_time = time.time() - start_time\n",
    "    total_time += iter_time\n",
    "    running_time.append(total_time)\n",
    "    if o_step % val_log_interval == 0 or o_step == n_steps-1:\n",
    "        test_loss, test_acc = eval(final_params[:len(parameters)], x_test, y_test)\n",
    "        test_losses.append(test_loss)\n",
    "        test_accs.append(test_acc)\n",
    "        print('o_step={} ({:.2e}s) Val loss: {:.4e}, Val Acc: {:.2f}%'.format(o_step, iter_time, val_losses[-1],\n",
    "                                                                              100*val_accs[-1]))\n",
    "        print('          Test loss: {:.4e}, Test Acc: {:.2f}%'.format(test_loss, 100*test_acc))\n",
    "        print('          l2_hp norm: {:.4e}'.format(torch.norm(hparams[0])))\n",
    "        if len(hparams) == 2:\n",
    "            print('          l1_hp : ', torch.norm(hparams[1]))\n",
    "\n",
    "print('HPO ended in {:.2e} seconds\\n'.format(total_time))\n",
    "\n",
    "plt.title('val_accuracy')\n",
    "plt.plot(running_time, val_accs)\n",
    "plt.show()\n",
    "\n",
    "plt.title('test_accuracy')\n",
    "plt.plot(test_accs)\n",
    "plt.show()\n",
    "\n",
    "val_fp = val_accs\n",
    "run_fp = running_time"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# CG"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dataset 20newsgroup, train_samples=5657, val_samples=5657, test_samples=7532, n_features=130107, n_classes=20\n",
      "o_step=0 (1.28e-01s) Val loss: 2.1360e+00, Val Acc: 48.65%\n",
      "          Test loss: 2.2243e+00, Test Acc: 43.26%\n",
      "          l2_hp norm: 1.0747e+00\n",
      "o_step=50 (1.23e-01s) Val loss: 5.3441e-01, Val Acc: 85.33%\n",
      "          Test loss: 8.5428e-01, Test Acc: 75.82%\n",
      "          l2_hp norm: 4.1391e+00\n",
      "o_step=100 (1.24e-01s) Val loss: 5.2557e-01, Val Acc: 85.66%\n",
      "          Test loss: 8.5565e-01, Test Acc: 75.97%\n",
      "          l2_hp norm: 4.4399e+00\n",
      "o_step=150 (1.27e-01s) Val loss: 5.2382e-01, Val Acc: 85.82%\n",
      "          Test loss: 8.6008e-01, Test Acc: 76.19%\n",
      "          l2_hp norm: 3.3054e+00\n",
      "o_step=200 (1.31e-01s) Val loss: 5.3276e-01, Val Acc: 85.73%\n",
      "          Test loss: 8.4270e-01, Test Acc: 76.00%\n",
      "          l2_hp norm: 5.7764e+00\n",
      "o_step=250 (1.34e-01s) Val loss: 5.1796e-01, Val Acc: 85.96%\n",
      "          Test loss: 8.3939e-01, Test Acc: 76.18%\n",
      "          l2_hp norm: 6.0519e+00\n",
      "o_step=300 (1.37e-01s) Val loss: 5.1497e-01, Val Acc: 86.05%\n",
      "          Test loss: 8.4321e-01, Test Acc: 76.29%\n",
      "          l2_hp norm: 6.1297e+00\n",
      "o_step=350 (1.41e-01s) Val loss: 5.1457e-01, Val Acc: 86.26%\n",
      "          Test loss: 8.4775e-01, Test Acc: 76.41%\n",
      "          l2_hp norm: 6.0051e+00\n",
      "o_step=400 (1.44e-01s) Val loss: 5.1506e-01, Val Acc: 86.32%\n",
      "          Test loss: 8.5213e-01, Test Acc: 76.53%\n",
      "          l2_hp norm: 5.5310e+00\n",
      "o_step=450 (1.48e-01s) Val loss: 5.1584e-01, Val Acc: 86.34%\n",
      "          Test loss: 8.5601e-01, Test Acc: 76.59%\n",
      "          l2_hp norm: 8.0534e-01\n",
      "o_step=499 (1.49e-01s) Val loss: 5.3218e-01, Val Acc: 85.50%\n",
      "          Test loss: 8.4491e-01, Test Acc: 75.97%\n",
      "          l2_hp norm: 1.0823e+01\n",
      "HPO ended in 6.74e+01 seconds\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from itertools import repeat\n",
    "\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import hypergrad as hg\n",
    "import numpy as np\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.preprocessing import MinMaxScaler\n",
    "from sklearn.datasets import fetch_20newsgroups_vectorized\n",
    "import time\n",
    "\n",
    "import torch.nn.functional as F\n",
    "\n",
    "\n",
    "# Helper functions to deal with cuda\n",
    "cuda = True and torch.cuda.is_available()\n",
    "\n",
    "default_tensor_str = 'torch.cuda.FloatTensor' if cuda else 'torch.FloatTensor'\n",
    "\n",
    "kwargs = {'num_workers': 1, 'pin_memory': True} if cuda else {}\n",
    "torch.set_default_tensor_type(default_tensor_str)\n",
    "#torch.multiprocessing.set_start_method('forkserver')\n",
    "\n",
    "def frnp(x): return torch.from_numpy(x).cuda().float() if cuda else torch.from_numpy(x).float()\n",
    "def tonp(x, cuda=cuda): return x.detach().cpu().numpy() if cuda else x.detach().numpy()\n",
    "\n",
    "\n",
    "seed = 0\n",
    "torch.manual_seed(seed)\n",
    "np.random.seed(seed)\n",
    "\n",
    "\n",
    "# load twentynews and preprocess\n",
    "val_size = 0.5\n",
    "X, y = fetch_20newsgroups_vectorized(subset='train', return_X_y=True,\n",
    "                                     #remove=('headers', 'footers', 'quotes')\n",
    "                                     )\n",
    "x_test, y_test = fetch_20newsgroups_vectorized(subset='test', return_X_y=True,\n",
    "                                               #remove=('headers', 'footers', 'quotes')\n",
    "                                               )\n",
    "\n",
    "\n",
    "x_train, x_val, y_train, y_val = train_test_split(X, y, stratify=y, test_size=val_size)\n",
    "\n",
    "\n",
    "train_samples, n_features = x_train.shape\n",
    "test_samples, n_features = x_test.shape\n",
    "val_samples, n_features = x_val.shape\n",
    "n_classes = np.unique(y_train).shape[0]\n",
    "\n",
    "print('Dataset 20newsgroup, train_samples=%i, val_samples=%i, test_samples=%i, n_features=%i, n_classes=%i'\n",
    "      % (train_samples, val_samples, test_samples, n_features, n_classes))\n",
    "\n",
    "\n",
    "ys = [frnp(y_train).long(), frnp(y_val).long(), frnp(y_test).long()]\n",
    "xs = [x_train, x_val, x_test]\n",
    "\n",
    "\n",
    "def from_sparse(x):\n",
    "    x = x.tocoo()\n",
    "    values = x.data\n",
    "    indices = np.vstack((x.row, x.col))\n",
    "\n",
    "    i = torch.LongTensor(indices)\n",
    "    v = torch.FloatTensor(values)\n",
    "    shape = x.shape\n",
    "\n",
    "    return torch.sparse.FloatTensor(i, v, torch.Size(shape))\n",
    "\n",
    "\n",
    "if cuda:\n",
    "    xs = [from_sparse(x).cuda() for x in xs]\n",
    "else:\n",
    "    xs = [from_sparse(x) for x in xs]\n",
    "\n",
    "x_train, x_val, x_test = xs\n",
    "y_train, y_val, y_test = ys\n",
    "\n",
    "\n",
    "class CustomTensorIterator:\n",
    "    def __init__(self, tensor_list, batch_size, **loader_kwargs):\n",
    "        self.loader = DataLoader(TensorDataset(*tensor_list), batch_size=batch_size, **loader_kwargs)\n",
    "        self.iterator = iter(self.loader)\n",
    "\n",
    "    def __next__(self, *args):\n",
    "        try:\n",
    "            idx = next(self.iterator)\n",
    "        except StopIteration:\n",
    "            self.iterator = iter(self.loader)\n",
    "            idx = next(self.iterator)\n",
    "        return idx\n",
    "\n",
    "\n",
    "# torch.DataLoader has problems with sparse tensor on GPU\n",
    "train_batch_size = len(y_train)\n",
    "val_batch_size = len(y_val)\n",
    "\n",
    "iterators = []\n",
    "for bs, x, y in [(train_batch_size, x_train, y_train), (val_batch_size, x_val, y_val)]:\n",
    "    if bs < len(y):\n",
    "        print('making iterator with batch size ', bs)\n",
    "        iterators.append(CustomTensorIterator([x, y], batch_size=bs, shuffle=True, **kwargs))\n",
    "    else:\n",
    "        iterators.append(repeat([x, y]))\n",
    "\n",
    "train_iterator, val_iterator = iterators\n",
    "\n",
    "# HPO set up\n",
    "n_steps = 500\n",
    "outer_lr, outer_mu = 100., 0.0  # nice with 100.0, 0.0 (torch.SGD) tested with T, K = 5, 10 and CG\n",
    "inner_lr, inner_mu = 100., 0.9   # nice with 100., 0.9 (HeavyBall) tested with T, K = 5, 10 and CG\n",
    "T, K = 5, 10\n",
    "tol = 1e-12\n",
    "warm_start = True\n",
    "bias = False  # without bias outer_lr can be bigger (much faster convergence)\n",
    "\n",
    "train_log_interval = 100\n",
    "val_log_interval = 50\n",
    "\n",
    "#l2_reg_params = torch.zeros(n_features).requires_grad_(True)  # one hp per feature\n",
    "l2_reg_params = (-1.*torch.ones(1)).requires_grad_(True)  # one l2 hp only (best when really low)\n",
    "l1_reg_params = (0.*torch.ones(1)).requires_grad_(True)  # one l1 hp only (best when really low)\n",
    "#l1_reg_params = (-1.*torch.ones(n_features)).requires_grad_(True)\n",
    "\n",
    "hparams = [l2_reg_params]\n",
    "\n",
    "ones_dxc = torch.ones(n_features, n_classes)\n",
    "\n",
    "\n",
    "def reg_f(params, l2_reg_params, l1_reg_params=None):\n",
    "    r = 0.5 * ((params[0] ** 2) * torch.exp(l2_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    if l1_reg_params is not None:\n",
    "        r += (params[0].abs() * torch.exp(l1_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    return r\n",
    "\n",
    "\n",
    "outer_opt = torch.optim.SGD(lr=outer_lr, momentum=outer_mu, params=hparams)\n",
    "#outer_opt = torch.optim.Adam(lr=0.01, params=hparams)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "val_losses, val_accs = [], []\n",
    "test_losses, test_accs = [], []\n",
    "\n",
    "w = torch.zeros(n_features, n_classes).requires_grad_(True)\n",
    "parameters = [w]\n",
    "\n",
    "if bias:\n",
    "    b = torch.zeros(n_classes).requires_grad_(True)\n",
    "    parameters.append(b)\n",
    "\n",
    "\n",
    "def out_f(x, params):\n",
    "    out = x @ params[0]\n",
    "    out += params[1] if len(params) == 2 else 0\n",
    "    return out\n",
    "\n",
    "\n",
    "def train_loss(params, hparams, data):\n",
    "    x_mb, y_mb = data\n",
    "    out = out_f(x_mb,  params)\n",
    "    return F.cross_entropy(out, y_mb) + reg_f(params, *hparams)\n",
    "\n",
    "\n",
    "def val_loss(opt_params, hparams):\n",
    "    x_mb, y_mb = next(val_iterator)\n",
    "    out = out_f(x_mb,  opt_params[:len(parameters)])\n",
    "    val_loss = F.cross_entropy(out, y_mb)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y_mb.view_as(pred)).sum().item() / len(y_mb)\n",
    "\n",
    "    val_losses.append(tonp(val_loss))\n",
    "    val_accs.append(acc)\n",
    "    return val_loss\n",
    "\n",
    "\n",
    "def eval(params, x, y):\n",
    "    out = out_f(x,  params)\n",
    "    loss = F.cross_entropy(out, y)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y.view_as(pred)).sum().item() / len(y)\n",
    "\n",
    "    return loss, acc\n",
    "\n",
    "\n",
    "if inner_mu > 0:\n",
    "    #inner_opt = hg.Momentum(train_loss, inner_lr, inner_mu, data_or_iter=train_iterator)\n",
    "    inner_opt = hg.HeavyBall(train_loss, inner_lr, inner_mu, data_or_iter=train_iterator)\n",
    "else:\n",
    "    inner_opt = hg.GradientDescent(train_loss, inner_lr, data_or_iter=train_iterator)\n",
    "\n",
    "inner_opt_cg = hg.GradientDescent(train_loss, 1., data_or_iter=train_iterator)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "total_time = 0\n",
    "running_time = []\n",
    "for o_step in range(n_steps):\n",
    "    start_time = time.time()\n",
    "\n",
    "    inner_losses = []\n",
    "    if params_history:\n",
    "        params_history = [params_history[-1]]\n",
    "    else:\n",
    "        params_history = [inner_opt.get_opt_params(parameters)]\n",
    "    for t in range(T):\n",
    "        params_history.append(inner_opt(params_history[-1], hparams, create_graph=False))\n",
    "        inner_losses.append(inner_opt.curr_loss)\n",
    "\n",
    "#         if t % train_log_interval == 0 or t == T-1:\n",
    "#             print('t={} loss: {}'.format(t, inner_losses[-1]))\n",
    "\n",
    "    final_params = params_history[-1]\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    #hg.reverse(params_history[-K-1:], hparams, [inner_opt]*K, val_loss)\n",
    "    #hg.fixed_point(final_params, hparams, K, inner_opt, val_loss, stochastic=False, tol=tol)\n",
    "    hg.CG(final_params[:len(parameters)], hparams, K, inner_opt_cg, val_loss, stochastic=False, tol=tol)\n",
    "    outer_opt.step()\n",
    "\n",
    "    for p, new_p in zip(parameters, final_params[:len(parameters)]):\n",
    "        if warm_start:\n",
    "            p.data = new_p\n",
    "        else:\n",
    "            p.data = torch.zeros_like(p)\n",
    "\n",
    "    iter_time = time.time() - start_time\n",
    "    total_time += iter_time\n",
    "    running_time.append(total_time)\n",
    "    if o_step % val_log_interval == 0 or o_step == n_steps-1:\n",
    "        test_loss, test_acc = eval(final_params[:len(parameters)], x_test, y_test)\n",
    "        test_losses.append(test_loss)\n",
    "        test_accs.append(test_acc)\n",
    "        print('o_step={} ({:.2e}s) Val loss: {:.4e}, Val Acc: {:.2f}%'.format(o_step, iter_time, val_losses[-1],\n",
    "                                                                              100*val_accs[-1]))\n",
    "        print('          Test loss: {:.4e}, Test Acc: {:.2f}%'.format(test_loss, 100*test_acc))\n",
    "        print('          l2_hp norm: {:.4e}'.format(torch.norm(hparams[0])))\n",
    "        if len(hparams) == 2:\n",
    "            print('          l1_hp : ', torch.norm(hparams[1]))\n",
    "\n",
    "print('HPO ended in {:.2e} seconds\\n'.format(total_time))\n",
    "\n",
    "plt.title('val_accuracy')\n",
    "plt.plot(running_time, val_accs)\n",
    "plt.show()\n",
    "\n",
    "plt.title('test_accuracy')\n",
    "plt.plot(test_accs)\n",
    "plt.show()\n",
    "\n",
    "val_cg = val_accs\n",
    "run_cg = running_time"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# reverse"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dataset 20newsgroup, train_samples=5657, val_samples=5657, test_samples=7532, n_features=130107, n_classes=20\n",
      "o_step=0 (1.15e-01s) Val loss: 2.1360e+00, Val Acc: 48.65%\n",
      "          Test loss: 2.2243e+00, Test Acc: 43.26%\n",
      "          l2_hp norm: 1.0020e+00\n",
      "o_step=50 (1.11e-01s) Val loss: 5.3517e-01, Val Acc: 85.35%\n",
      "          Test loss: 8.5323e-01, Test Acc: 75.84%\n",
      "          l2_hp norm: 1.1325e+00\n",
      "o_step=100 (1.12e-01s) Val loss: 5.2515e-01, Val Acc: 85.68%\n",
      "          Test loss: 8.5114e-01, Test Acc: 76.06%\n",
      "          l2_hp norm: 1.1638e+00\n",
      "o_step=150 (1.15e-01s) Val loss: 5.2182e-01, Val Acc: 85.79%\n",
      "          Test loss: 8.5172e-01, Test Acc: 76.22%\n",
      "          l2_hp norm: 1.1646e+00\n",
      "o_step=200 (1.19e-01s) Val loss: 5.2009e-01, Val Acc: 85.82%\n",
      "          Test loss: 8.5221e-01, Test Acc: 76.34%\n",
      "          l2_hp norm: 1.1499e+00\n",
      "o_step=250 (1.22e-01s) Val loss: 5.1892e-01, Val Acc: 85.95%\n",
      "          Test loss: 8.5229e-01, Test Acc: 76.39%\n",
      "          l2_hp norm: 1.1257e+00\n",
      "o_step=300 (1.25e-01s) Val loss: 5.1795e-01, Val Acc: 85.96%\n",
      "          Test loss: 8.5195e-01, Test Acc: 76.47%\n",
      "          l2_hp norm: 1.0957e+00\n",
      "o_step=350 (1.28e-01s) Val loss: 5.1708e-01, Val Acc: 86.09%\n",
      "          Test loss: 8.5127e-01, Test Acc: 76.47%\n",
      "          l2_hp norm: 1.0619e+00\n",
      "o_step=400 (1.32e-01s) Val loss: 5.1625e-01, Val Acc: 86.18%\n",
      "          Test loss: 8.5031e-01, Test Acc: 76.54%\n",
      "          l2_hp norm: 1.0261e+00\n",
      "o_step=450 (1.35e-01s) Val loss: 5.1545e-01, Val Acc: 86.21%\n",
      "          Test loss: 8.4914e-01, Test Acc: 76.59%\n",
      "          l2_hp norm: 9.8963e-01\n",
      "o_step=499 (1.37e-01s) Val loss: 5.1468e-01, Val Acc: 86.34%\n",
      "          Test loss: 8.4785e-01, Test Acc: 76.61%\n",
      "          l2_hp norm: 9.5427e-01\n",
      "HPO ended in 6.13e+01 seconds\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from itertools import repeat\n",
    "\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import hypergrad as hg\n",
    "import numpy as np\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.preprocessing import MinMaxScaler\n",
    "from sklearn.datasets import fetch_20newsgroups_vectorized\n",
    "import time\n",
    "\n",
    "import torch.nn.functional as F\n",
    "\n",
    "\n",
    "# Helper functions to deal with cuda\n",
    "cuda = True and torch.cuda.is_available()\n",
    "\n",
    "default_tensor_str = 'torch.cuda.FloatTensor' if cuda else 'torch.FloatTensor'\n",
    "\n",
    "kwargs = {'num_workers': 1, 'pin_memory': True} if cuda else {}\n",
    "torch.set_default_tensor_type(default_tensor_str)\n",
    "#torch.multiprocessing.set_start_method('forkserver')\n",
    "\n",
    "def frnp(x): return torch.from_numpy(x).cuda().float() if cuda else torch.from_numpy(x).float()\n",
    "def tonp(x, cuda=cuda): return x.detach().cpu().numpy() if cuda else x.detach().numpy()\n",
    "\n",
    "\n",
    "seed = 0\n",
    "torch.manual_seed(seed)\n",
    "np.random.seed(seed)\n",
    "\n",
    "\n",
    "# load twentynews and preprocess\n",
    "val_size = 0.5\n",
    "X, y = fetch_20newsgroups_vectorized(subset='train', return_X_y=True,\n",
    "                                     #remove=('headers', 'footers', 'quotes')\n",
    "                                     )\n",
    "x_test, y_test = fetch_20newsgroups_vectorized(subset='test', return_X_y=True,\n",
    "                                               #remove=('headers', 'footers', 'quotes')\n",
    "                                               )\n",
    "\n",
    "\n",
    "x_train, x_val, y_train, y_val = train_test_split(X, y, stratify=y, test_size=val_size)\n",
    "\n",
    "\n",
    "train_samples, n_features = x_train.shape\n",
    "test_samples, n_features = x_test.shape\n",
    "val_samples, n_features = x_val.shape\n",
    "n_classes = np.unique(y_train).shape[0]\n",
    "\n",
    "print('Dataset 20newsgroup, train_samples=%i, val_samples=%i, test_samples=%i, n_features=%i, n_classes=%i'\n",
    "      % (train_samples, val_samples, test_samples, n_features, n_classes))\n",
    "\n",
    "\n",
    "ys = [frnp(y_train).long(), frnp(y_val).long(), frnp(y_test).long()]\n",
    "xs = [x_train, x_val, x_test]\n",
    "\n",
    "\n",
    "def from_sparse(x):\n",
    "    x = x.tocoo()\n",
    "    values = x.data\n",
    "    indices = np.vstack((x.row, x.col))\n",
    "\n",
    "    i = torch.LongTensor(indices)\n",
    "    v = torch.FloatTensor(values)\n",
    "    shape = x.shape\n",
    "\n",
    "    return torch.sparse.FloatTensor(i, v, torch.Size(shape))\n",
    "\n",
    "\n",
    "if cuda:\n",
    "    xs = [from_sparse(x).cuda() for x in xs]\n",
    "else:\n",
    "    xs = [from_sparse(x) for x in xs]\n",
    "\n",
    "x_train, x_val, x_test = xs\n",
    "y_train, y_val, y_test = ys\n",
    "\n",
    "\n",
    "class CustomTensorIterator:\n",
    "    def __init__(self, tensor_list, batch_size, **loader_kwargs):\n",
    "        self.loader = DataLoader(TensorDataset(*tensor_list), batch_size=batch_size, **loader_kwargs)\n",
    "        self.iterator = iter(self.loader)\n",
    "\n",
    "    def __next__(self, *args):\n",
    "        try:\n",
    "            idx = next(self.iterator)\n",
    "        except StopIteration:\n",
    "            self.iterator = iter(self.loader)\n",
    "            idx = next(self.iterator)\n",
    "        return idx\n",
    "\n",
    "\n",
    "# torch.DataLoader has problems with sparse tensor on GPU\n",
    "train_batch_size = len(y_train)\n",
    "val_batch_size = len(y_val)\n",
    "\n",
    "iterators = []\n",
    "for bs, x, y in [(train_batch_size, x_train, y_train), (val_batch_size, x_val, y_val)]:\n",
    "    if bs < len(y):\n",
    "        print('making iterator with batch size ', bs)\n",
    "        iterators.append(CustomTensorIterator([x, y], batch_size=bs, shuffle=True, **kwargs))\n",
    "    else:\n",
    "        iterators.append(repeat([x, y]))\n",
    "\n",
    "train_iterator, val_iterator = iterators\n",
    "\n",
    "# HPO set up\n",
    "n_steps = 500\n",
    "outer_lr, outer_mu = 100.0, 0.0  # nice with 100.0, 0.0 (torch.SGD) tested with T, K = 5, 10 and CG\n",
    "inner_lr, inner_mu = 100.0, 0.9   # nice with 100., 0.9 (HeavyBall) tested with T, K = 5, 10 and CG\n",
    "T, K = 5, 10\n",
    "tol = 1e-12\n",
    "warm_start = True\n",
    "bias = False  # without bias outer_lr can be bigger (much faster convergence)\n",
    "\n",
    "train_log_interval = 100\n",
    "val_log_interval = 50\n",
    "\n",
    "#l2_reg_params = torch.zeros(n_features).requires_grad_(True)  # one hp per feature\n",
    "l2_reg_params = (-1.*torch.ones(1)).requires_grad_(True)  # one l2 hp only (best when really low)\n",
    "l1_reg_params = (0.*torch.ones(1)).requires_grad_(True)  # one l1 hp only (best when really low)\n",
    "#l1_reg_params = (-1.*torch.ones(n_features)).requires_grad_(True)\n",
    "\n",
    "hparams = [l2_reg_params]\n",
    "\n",
    "ones_dxc = torch.ones(n_features, n_classes)\n",
    "\n",
    "\n",
    "def reg_f(params, l2_reg_params, l1_reg_params=None):\n",
    "    r = 0.5 * ((params[0] ** 2) * torch.exp(l2_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    if l1_reg_params is not None:\n",
    "        r += (params[0].abs() * torch.exp(l1_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    return r\n",
    "\n",
    "\n",
    "outer_opt = torch.optim.SGD(lr=outer_lr, momentum=outer_mu, params=hparams)\n",
    "#outer_opt = torch.optim.Adam(lr=0.01, params=hparams)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "val_losses, val_accs = [], []\n",
    "test_losses, test_accs = [], []\n",
    "\n",
    "w = torch.zeros(n_features, n_classes).requires_grad_(True)\n",
    "parameters = [w]\n",
    "\n",
    "if bias:\n",
    "    b = torch.zeros(n_classes).requires_grad_(True)\n",
    "    parameters.append(b)\n",
    "\n",
    "\n",
    "def out_f(x, params):\n",
    "    out = x @ params[0]\n",
    "    out += params[1] if len(params) == 2 else 0\n",
    "    return out\n",
    "\n",
    "\n",
    "def train_loss(params, hparams, data):\n",
    "    x_mb, y_mb = data\n",
    "    out = out_f(x_mb,  params)\n",
    "    return F.cross_entropy(out, y_mb) + reg_f(params, *hparams)\n",
    "\n",
    "\n",
    "def val_loss(opt_params, hparams):\n",
    "    x_mb, y_mb = next(val_iterator)\n",
    "    out = out_f(x_mb,  opt_params[:len(parameters)])\n",
    "    val_loss = F.cross_entropy(out, y_mb)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y_mb.view_as(pred)).sum().item() / len(y_mb)\n",
    "\n",
    "    val_losses.append(tonp(val_loss))\n",
    "    val_accs.append(acc)\n",
    "    return val_loss\n",
    "\n",
    "\n",
    "def eval(params, x, y):\n",
    "    out = out_f(x,  params)\n",
    "    loss = F.cross_entropy(out, y)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y.view_as(pred)).sum().item() / len(y)\n",
    "\n",
    "    return loss, acc\n",
    "\n",
    "\n",
    "if inner_mu > 0:\n",
    "    #inner_opt = hg.Momentum(train_loss, inner_lr, inner_mu, data_or_iter=train_iterator)\n",
    "    inner_opt = hg.HeavyBall(train_loss, inner_lr, inner_mu, data_or_iter=train_iterator)\n",
    "else:\n",
    "    inner_opt = hg.GradientDescent(train_loss, inner_lr, data_or_iter=train_iterator)\n",
    "\n",
    "inner_opt_cg = hg.GradientDescent(train_loss, 1., data_or_iter=train_iterator)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "total_time = 0\n",
    "running_time = []\n",
    "for o_step in range(n_steps):\n",
    "    start_time = time.time()\n",
    "\n",
    "    inner_losses = []\n",
    "    if params_history:\n",
    "        params_history = [params_history[-1]]\n",
    "    else:\n",
    "        params_history = [inner_opt.get_opt_params(parameters)]\n",
    "    for t in range(T):\n",
    "        params_history.append(inner_opt(params_history[-1], hparams, create_graph=False))\n",
    "        inner_losses.append(inner_opt.curr_loss)\n",
    "\n",
    "#         if t % train_log_interval == 0 or t == T-1:\n",
    "#             print('t={} loss: {}'.format(t, inner_losses[-1]))\n",
    "\n",
    "    final_params = params_history[-1]\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    hg.reverse(params_history[-K-1:], hparams, [inner_opt]*K, val_loss)\n",
    "    #hg.fixed_point(final_params, hparams, K, inner_opt, val_loss, stochastic=False, tol=tol)\n",
    "    #hg.CG(final_params[:len(parameters)], hparams, K, inner_opt_cg, val_loss, stochastic=False, tol=tol)\n",
    "    outer_opt.step()\n",
    "\n",
    "    for p, new_p in zip(parameters, final_params[:len(parameters)]):\n",
    "        if warm_start:\n",
    "            p.data = new_p\n",
    "        else:\n",
    "            p.data = torch.zeros_like(p)\n",
    "\n",
    "    iter_time = time.time() - start_time\n",
    "    total_time += iter_time\n",
    "    running_time.append(total_time)\n",
    "    if o_step % val_log_interval == 0 or o_step == n_steps-1:\n",
    "        test_loss, test_acc = eval(final_params[:len(parameters)], x_test, y_test)\n",
    "        test_losses.append(test_loss)\n",
    "        test_accs.append(test_acc)\n",
    "        print('o_step={} ({:.2e}s) Val loss: {:.4e}, Val Acc: {:.2f}%'.format(o_step, iter_time, val_losses[-1],\n",
    "                                                                              100*val_accs[-1]))\n",
    "        print('          Test loss: {:.4e}, Test Acc: {:.2f}%'.format(test_loss, 100*test_acc))\n",
    "        print('          l2_hp norm: {:.4e}'.format(torch.norm(hparams[0])))\n",
    "        if len(hparams) == 2:\n",
    "            print('          l1_hp : ', torch.norm(hparams[1]))\n",
    "\n",
    "print('HPO ended in {:.2e} seconds\\n'.format(total_time))\n",
    "\n",
    "plt.title('val_accuracy')\n",
    "plt.plot(running_time, val_accs)\n",
    "plt.show()\n",
    "\n",
    "plt.title('test_accuracy')\n",
    "plt.plot(test_accs)\n",
    "plt.show()\n",
    "\n",
    "val_rv = val_accs\n",
    "run_rv = running_time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA34AAAHgCAYAAAD62r8OAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4xLjEsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy8QZhcZAAAgAElEQVR4nOzdd3xV9f3H8de5K3sRNgQSliJDBGQICmpduBH3arUqWq171WodtY6qxarUOnAriiCi4q/WgeJGyhKRvcImIXvc9f39cW6SmxBGxs0NyfvZnseZ99xPQoj3zfd7vl/LGIOIiIiIiIi0XI5oFyAiIiIiIiKRpeAnIiIiIiLSwin4iYiIiIiItHAKfiIiIiIiIi2cgp+IiIiIiEgLp+AnIiIiIiLSwrmiXUBjadu2rcnMzIx2GSIiIiIiIlExf/78ncaYdrWdazHBLzMzk59++inaZYiIiIiIiESFZVnr93ROXT1FRERERERaOAU/ERERERGRFk7BT0REREREpIVrMc/41cbn85GdnU1ZWVm0S4ma2NhYunbtitvtjnYpIiIiIiISJS06+GVnZ5OUlERmZiaWZUW7nCZnjCEnJ4fs7GyysrKiXY6IiIiIiERJi+7qWVZWRnp6eqsMfQCWZZGent6qWzxFRERERKSFBz+g1Ya+Cq396xcRERERkVYQ/A4k48aNIy8vL9pliIiIiIhIC9Oin/E70MyePTvaJYiIiIiISAukFr8Ie/bZZxk0aBCDBg0iKyuLo48+mrfeeosBAwbQv39/br/99sprMzMz2blzZxSrFRERERGRlqjVtPjd98FSftlc0Kj3PKRzMn85td9er5k4cSITJ07E5/NxzDHH8Nvf/pbbb7+d+fPnk5aWxvHHH8/MmTM544wzGrU2ERERERGRChFt8bMs60TLspZblrXKsqw7ajnfzbKsLyzLWmBZ1mLLssaFjmdallVqWdbC0PJsJOtsCtdffz3HHHMMqampjB07lnbt2uFyubjwwgv56quvol2eiIiIiIi0YBFr8bMsywk8AxwHZAPzLMuaZYz5JeyyPwPvGGP+ZVnWIcBsIDN0brUxZlBj1bOvlrlIevnll1m/fj1PP/00s2bNilodIiIiIiLSOkWyxW8YsMoYs8YY4wWmAqfXuMYAyaHtFGBzBOuJivnz5/PYY4/x+uuv43A4GD58OF9++SU7d+4kEAjw1ltvMWbMmGiXKSIiIiIiLVgkn/HrAmwM288Ghte45l7gE8uyrgMSgN+EncuyLGsBUAD82RgzN4K1RszTTz9Nbm4uRx99NABDhw7loYce4uijj8YYw7hx4zj99Ko8rHn3RERERESksUUy+NWWYEyN/fOBl40xj1uWNRJ4zbKs/sAWoJsxJseyrCHATMuy+hljqo3OYlnWlcCVAN26dWv8r6ARvPTSS7Uev+CCC6rtBwIBCgsLSU5OrvV6ERERERGR+opkV89sICNsvyu7d+W8HHgHwBjzHRALtDXGlBtjckLH5wOrgT4138AY85wxZqgxZmi7du0i8CU0nX79+vH73/8et9sd7VJERESaPWMM/kCQUm+A/FIfvkAw2iWJiDRrkWzxmwf0tiwrC9gEnAdcUOOaDcCxwMuWZfXFDn47LMtqB+QaYwKWZfUAegNrIlhr1P3666/RLkFERFo5Ywy+gMEXCOILBPEGgva+P4g/GMTrr/1ctf1AEH8giLfiPpXnq17rC1S/l6/mubD7Vpz3Vmz7q/bDTb96JEO6t4nSd05EpPmLWPAzxvgty7oW+A/gBKYYY5ZalnU/8JMxZhZwM/C8ZVk3YncD/a0xxliWdRRwv2VZfiAATDTG5EaqVhERiYxg0P6AXu4LUu4PUO4PYgy4XRZupwO304HH6cDttHA6rCZ9ztkfCFLiC1BSHqDY669ae/2UeMOOewMUl4etfQFKyv0EDTgscFh23RXbDgehfSvsPDX2w663alzvqLo+aAyBgMEfNASCFetg9f3AHo6HXx+w96ufCxIImhqhq+YTGfvL4MFPAqUkWOXEU0YipcRb5SRQSjbtWeXoYf9Zu+w/b5fDgSe0Hf6z4HE5SIip/rPhrnidI2zb6cAT9touqfGN+vMhItLSRHQCd2PMbOwpGsKP3RO2/QswqpbXTQemR7I2EZHWzusPklfiJbfES26xl13FPorKfZT7qwe1cn8Qrz+07wuGjgX267qarTJ7Y1ns/mG/5rar+od9t9OBx2WHiIptjwMcviIozYOyfFzefJzeApy+Inw+P/5AAJ/fTzAYxMLgIIgDE1qC1fYrznssQ7LLIsYJHid4HBYOy2ARxDL2EsTCb5wEcODHgR8nfmOvfTgJGCc+HJXH/MaB1zjx48BnHPiNfcxnHPiwzwVxgsOFcThxOhw4HBZOyw7JzrBtj2Xhclph58FpOXA6LZxuO1C6nBYOyxH2Wjtkuh0OPI4A8ZQRZ0qJo4y4YAkxppTYYCkxwVI8wRI8gRLcocXlr1iKcfiLcfqKsYL+Pf/hDrsKxv2hEX5qRcKUl0NeHuzaVbXsaT8vDwIBcDqrFoej+n5THrP/hSgi3xZ/0E9ZoIwyfxmlvlLKAmWU+ksp9ZdR5q/Yt7dL/WX4gj4WjsigNCnW/kcoHGCBhf07w7J3qu1b2P+Atb9rB47Kf9irvE/N6/bnXqHXVtRTuQ+V1+yx5tD71+Vr2OO9Q8eO7XYsSZ6kiPw5RkJEg5+IiDSNihC3q8Rnh7hQmMsr8ZJb7Ku+X1IR8vbyQT3EsiDG5SDG5bTX7rBtl906kxrvCZ2rOh7jcoaurf5aj9P+j70/UKP7X43ugAG/1w5s5YV4fPl4fAV4/IXE+AuJLSsgNlBEfLCQ+EAh8cFiEkwhiabYXijGudtYYjU4Q0tdBR1gHGDVshgDQX9o8dX93ha1D4tW+d6hpam54sCTADGJ4AktiW2r79c870mw1zGh7cSOUShcmj1joLh498C2v2GutHTv94+Ph7Q0e0lJAZcLfD4oK4Ng0A6C4UtDjpn6tpY3PheQGFr2192TT2djUgzBYJAAAYImiKn4nwkt4fth66CxfzEZYwgSxIS+FxXnal5f2z3D71H5Ogz2/8P2K94nrD4Mle9bcU1TGdhuoIKfiIjUzhhT2RJWrbWsRstZuW/3VrTicj+7SnzsKq4Ib97K/cK9hLgEj5O0BA9tEjykxXvIaptg78d7SAsdS0tw0ybBQ2KMi9jKAOfE7axn90tjwFdS2epGWV7dtn3Fe7+/KxZiUyExBeJSIbYjxFZsp9jnam57EsHhqj20WVb1fYez9mvqIhi0A2BlGAyEbfsh4Nv9WDBQ+2sC9QiSdeFwVYU0T1JVkHMngLOVfVQwxv4g7/XaIaG2dc1jFR/8g0F7Hb5d27F9nW/Ia/b7y7Q/pAdNkEAwSNDYH/YrlkBoP2DscwETxITOVd6j8sN81T7hH74DAVwFRbjyC0PrItz5hbjzi3AVFOMuKMLhD+y1Tm9SPN7kBHzJCXiT4ynvkoC3bzu8yQmUJ8Xbx0JL1bWJ+JMTMTH2gHl7bfXZjxYmy9p9P7xVzMLCgYVlDFYgiCNgcBiDIwgEAziCBr+vnKKyfIrLCykuLaCkvIhCXyHFvmKKfcUUeosoCu2X+EuqfZ9rsrCId8eT6E4k0Z1EoieBRHciCW57neixtxPcCSR7ku3jnqrzMc6Y3X6vP9ClC3g8+/3z05ztMWiG1pUhs2I/PKyG7Vf8HIeH18r90Gs7xHeIzhdZT63st3nTW7duHaeccgo///xztEsRkXryB4LklfoqW88qWtR2hcJXbrGPgjLfboGttq6RXn/DmmziPU7S4kMhLsFDZtuEavtt4j2kxbsrg15qvJsY1x6atoyxg0XAC4FyCJTZ214vlHhDx32hc6Ftb7Ed0MryQyFtT9v5+271ikm2Q1lFYGvTIyywhR2vbdsd26DvY5NwOMARA8REu5LoqAhQ+xueonFuT9c3o9abSLGoaviO5HjifgcUJDgpiHeyI8FJYbyTgvZOCrKcFCSk2fvxTgoqziVU7RfFOQg6av6DS2loydn9zXyhw7Wcam7iXHEkuZNITkgmyZNEsqcdnT1Joe3kynXldoy9TvIkkehOrOyCKLurCOl77UHRSin4iUirEgiayi6Ruyqfbattv6rbZH7pngNMnNtJmwQPyXFuYkNdG5Pj3Hvs8uipPL6HrpFOBzEui1i8xFU8Y2XsZ6xiAiV4gqVQvsMOYN5Ce11eBAXFsLMQvCWhkOYFf3kouFUEO19VgPOHwlxDusVYzt2DWWr3vQe2iu2Y5NbXktTclJVBfr7dbS4/v2oJ3w/fLi6ue9iKZIByOMDjwbjdGI8b43IRdLsIuJ0EXA4CLid+p4XPZeF3gtdp4Y0Bb1yQcqfB57DwuWLwOWPxOsHnAq8DvE6DzwnlTkO5E7yOIOUOQ7nDfp3fZeGrvK+FzwlBh2X3Ag592AxaYCwIWva+oeoYlhV2HsCq3DYWGMvedzicOJ1unA6XvTjdOJ0uHM6K/dC204XT4cZVca3ThcthL06HE5flsheHC7fDbR8Lna84XnHMbbmrv85R/V5uh7uypWyvzz5VnHc4MPFxWA4HDhwkWBaJWHSyqHzma2+tanV59qrivfe3e2JtrUJBE6zsWri37oYVrUO7dTes2T0ybO12uKsFOrdT03dJ09N/dZtAIBDgiiuu4Ntvv6VLly68//77LF++nIkTJ1JSUkLPnj2ZMmUKpaWljBs3rvJ1S5YsYc0aexaLyy67jB07dtCuXTteeuklunXrxurVq7nwwgsJBAKcdNJJPPHEExQVFUXryxSJGmMMeSU+theWs62gjG0FZWwvLGd7QRnbCsrZUVRe2T0yv9S3x8+iMS4H6QkeUkMtaF3S4mkTaj1Liw9rUQt1i0yL9xDr3suDYgEfFG2Dgi1QuLlqXZgH3qKq0OYt2n1/fwOZ01P1XFVMIrjj7W6Q7ng7YDnd9jWumKrt3ZaKa2oei6l+3ukBd1xVePMkRGxwAtkHv3/fQW1v5/Ly7GC2N5ZlPxeVkgKpqZCQYHcFS0y01x4PuN37t65xzLjdeB2GUmeAUitAmRWgxOGn1PJTbPkoorxyXYSXQlNGgSmjwJRSQBn5wRIKgyUU+4opD5Tv89tV0TWuoqtbvCsej9NTLdjUDDpuh7ty22O5iHfUfj78upr3qHa+DueclrNJR7gVkdah9QS/j++ArUsa954dB8BJD+/zspUrV/LWW2/x/PPPc8455zB9+nQeffRRnnrqKcaMGcM999zDfffdx6RJk1i4cCEAzzzzDF9++SXdu3fn1FNP5ZJLLuHSSy9lypQp/PGPf2TmzJlcf/31XH/99Zx//vk8++yzjfu1iTQDxhjyS8MDnb3eUTPgFZbX2oUyKdZFh+RY2iXG0LdzcuUzbW3CukKGB7o4Tx1G+ygrgF3rqwe6gi1QuAUKNtvrou3sFuCcHohLCxsQIwkS24OnR9V+xcAYNfdrnvMk2mFNoisYtENUeXnjLRX3KyjYc+vbviQk2IGtIry1awe9ekFKCiYlhWByEoHkRHxJCfiTE/Alhp6XSoilPCmW8jg3PhPAG/TiD/rxBXx4g158QR++gA9f0Ic3ENqvWAJV2+WBcop9uyj2FVPkK6LYG1qHnmkyAWN3zdsLj8NDoscOahXPKKW50+jqrnqmqebzS1XPPiVWnotzxalrnIi0eq0n+EVRVlYWgwYNAmDIkCGsXr2avLw8xowZA8Cll17K2WefXXn9N998wwsvvMDcuXMB+O6775gxYwYAF198Mbfddlvl8ZkzZwJwwQUXcMsttzTZ1yTSUEXlfrbml7I1PxTiCsvYXlDO9kI74FWs9xTo2ifF0CE5lsMz29A+KYb2ybF0SI6hfVLVuk5BrkLAD8Xbd2+lqxnuvLW0rsemQnIXSO5k/8NQcmdI6lR9HZ+uVrJICQbtUf5KSqrWFUv4/r7OlZXVHsL2tPgab+AV43QS9LgJetwEPC58CbF4E+MoS4ylrEssZX2SKU7oQUm8m6J4J0XxLorjXRTEOyiMs5f8eIuCGCi3ApVhzA5sBfgCOZVBrprC0LKl/rVbWHicHtwOd+U6ISygdYjvsFs4q20wivDXeJz6hw0RkcbSeoLffrTMRUpMTNWD/U6nk7y8vD1eu2XLFi6//HJmzZpFYmLtA/Gq+4c0Z8YYdpX42JJfytb8MrYWlLE1v4wt+XYL3ZZ8e7+2qQSSYly0D4W2Id3S7Na6UMDrkBwbCngxxHvq+KvLXw7FO+wWuMr1dijaEVqHHS/N3f31Dpcd3JI6QYdDoNdv7HCX1Dm0Di0eTSBdJ7m5sHw5bNmy70C2P+GtrKx+dcTGQlycPfR7XBzExGBiPJS7LPtZsFgHvqQY/O44fG4nPpeF1+3A64Jyl4XXBWUu+5mwUqehzBmk1BWk1BmkxBGg1BGk2OmnxPJT7PBT6gzidVt4XfbiczvsbbdVy0AW4Xy4LEOMC/tZUKeLGKcLt9ON2+GuFrjahvbDz4WfrzjucXiqX1PLvWrdd7p3u4/TUZ/5MUREpKm0nuDXjKSkpJCWlsbcuXM58sgjee211xgzZgw+n49zzjmHRx55hD59+lRef8QRRzB16lQuvvhi3njjDUaPHg3AiBEjmD59Oueeey5Tp06N1pcjrYg/EGRnkbfWUFexvbWgbLdWOocF7ZNi6ZgSS692iYzu1ZZOKfZ+h+RYOibH1j3QeUuqh7fiHbUHueLt9giTtfEkQkI7u6tlei/oNtLeTmwfFuo629c41E2sXrxeWL3aDng1l5x9DL0XH1+1VASz+HhISoL27fd8fj/3i92Gjf6dbCzexMbCjZVLdmE2W4q37HU4dbC7Ica4Yohx2kusMxaP00OsK3a3/TbOGDqFrotx2edi9rHvcXrs42H7Lof+sy0iIvVjmRYyZPHQoUPNTz/9VO3YsmXL6Nu3b5QqstWczuGxxx6jqKiIM844o3Jwlx49evDSSy+xePFiTjjhBA4++ODK18+ePRuv18tll13Gzp07qw3usnLlSi666CKMMZx88sk899xzbNq0abcamsP3QQ4cxhg25JawYEMeSzbls2lXaWWo215YRrDGrwyP00HHUIjrmBxbGeg6JtvrTilxtE304HLWIziV7oKcNZCzCnJX2+uc1ZC7Fsr3EOZiUyAhFN4qQl1Cu7Dt9pDYzl6rha5xGAPbttUe7tautYf0r9ChAxx0EPTpY68POggyMqqHsvh4iIlpcJdYYww5ZTlkF2ZXC3YVS25Z9dbd1JhUMpIy6JrUlYykDHs7sSvJMcmVwSzWZQewGGeMnhkTEZFmx7Ks+caYobWeU/A7cJWUlBAXF4dlWUydOpW33nqL999/f7frWvr3QRomv9THoo15LNyYx4INu1iUnU9usT3aX6zbQde0eDqFWuZqC3Vp8e6GdT/2FofCXEWwCwt6JWEtQpYDUjLslrn0nnbXyvBgV7F2tdI505pCaSmsXFl7wCsoqLouNhZ6964KdhVLnz72YCONyB/0s6V4S2VLXc1wV+ovrbzWwqJjQseqUBcW8DKSMkjyJDVqbSIiIk1tb8FPfUYOYPPnz+faa6/FGENqaipTpkyJdknSzPkDQX7dWsiCjXks3JDHwo27WL3DHh3QsqBXu0SOPbg9g7qlclhGGn06JNavpW63Ny6HXeuqWuxyVkFuKOAV1hhNIqmzHez6ngpteoaCXi9I665Q1xSCQdi0qfZwt2FD9XnZuna1A91FF1UPeN26NWrX2BJfCdlF2bWGuy1FW/CbqudFPQ4PXZK6kJGUweEdD68W8romdtVgISIi0mop+B3AjjzySBYtWhTtMqSZMsawJb+MhWGteUs25VPms59bSk/wMCgjlTMP68KgjDQGZqSQHNuACWUDfsjfUHvXzPyNEP68VHy6HeZ6HG2HvPRQwGsTmtJAIq+wEFas2D3crVhhD5ZSITHRDnOjRsFll1W13PXpY08X0AiMMewq37Xbc3YV2ztLd1a7PsmTREZSBoekH8IJmSdUa7VrH99eXTBFRERqoeAn0kIUl/tZsimfBaGWvIUb89hWYE9s7HE66NclmfOHdWNQRiqDu6XRNS2u7l00g0G7ha4y2K2uasHbtQ6CYcPaxyTbQa7r4XDoeaFg1xPSe9jz2ElklJfD1q32SJl7WjZvtp/Jq+BwQGamHerGjq3eetepU72etTPGUOIvIac0h9yyXHJKc8gpCy1hx3LLctlRuoNiX/V56drHtycjKYPRXUZXC3YZSRmkxKQ07HskIiLSCin4iRyAjDGs3lHE/9bnsSDUmrdiW2HlwCvd0+MZ0SOdwzJSGdQtjb6dkohx7edQ68bYz9ZV65YZCni5a8AX1hrkirXDXPuDoe8pYV0ze9rP22nqkcZTXLz3IFexnVvbdBQOe1CVTp2gc2cYMgR69KgKd7162YOp7EPQBMkvz68MceHhrbZAVxaofXqFZE8y6XHppMem0yetD0d0PqLa83ZdErsQ64pt6HdMREREwij4iRwgNuWV8s2qnXy7aiffrs5he6HdmpcU62JQRirHH9KBQd1SObRrKumJ+/EsXFlB9WftKoPe6uojZjpckJZpB7qsMdW7ZiZ11jQHDWEM5OfvHt5qWwoLd3+9222HuU6d7MFUjjqqaj98ad8enLUHf1/AZ4e4otzaA13YsV1luwiYwG73cFpO2sS2IT0unTaxbchMzqy2XxHy2sS2oU1sG9zOBnQpFhERkXpR8BNppnYVe/luTQ7frNrJN6t2si7Hbmlrm+hhZM+2jOqZztDMNvRom4Bjr5M+hxTnwPpvYN1cWPc1bP8l7KQVGjGzJww8p/ozd6ndwalfFfUSCNjTGSxZAr/+Wnu4q23S8fj4qtB26KFw4onVg1znzva6TZu9tqoW+4pZl7+ONet+Irsom9zS3GrBLqcsh0JvLYESiHPF2aEtNp3OiZ0Z0HZAtRBXGepi00mOSdZzdSIiIs2cPs2JNBMlXj/z1u3i21U7+XrVTn7ZUoAxkOBxMrxHOhePzGRUr3QO6pC0f8/mleTC+m+rgt42ey5J3AnQbQT0Hw/t+tohLy0L3OpaV2/G2M/VLVkCP/9ctV661J4CoUJKSlV4GzmyeogLX5KS9rubrDGGHaU7WJO/hrX5ayuXNflr2F6yvdq1NbtY1tYiV7Ed79YchyIiIi2Jgp9IlPgCQRZtzOObVTl8s3onCzbswhcwuJ0Wg7ulceNv+jCqVzoDu6bi3p8pFUrzQkHva1j3FWz9GTDgioNuw+GYuyHzSOgyGNTVrv7y8+1AVzPk5YTNOdihAwwYAFddZa/794dDDrFHyKwnX8DHxsKNdrArWMuavDWV2+EDoyS4E+iR0oMRnUaQlZJFVnIWWSlZdE3SVAYiIiKtmYJfE3j11Vd57LHHsCyLgQMHcu+993LhhRcSCAQ46aSTeOKJJygqKop2mRJhwaBh+bZC+zm91Tn8sCaHYm8Ay4J+nZO5bFQWo3q15fDMNsR59mMglrJ82PA9rP3KbtXbshg76MVCxjA4+k+hoDcEXPrAX2fl5Xb3zPBwt2SJPZddhcREO9SNH2+vK0Jeu3b1ftsCb0G1lruKZWPhxmrP13WI70BWShan9TyNHik97JCXkkW7uHZ1H61VREREWrxWE/we+fERfs39tVHveXCbg7l92O17vWbp0qU8+OCDfPPNN7Rt25bc3FwuueQSrr/+es4//3yeffbZRq1JmpeNuSX2M3qrc/h21U5yir0AZLVN4MzBXRjVsy0jeqSTlrAfway8MCzofQ1bFtpz4zk90HUYjL2jKuip2+b+CwarnsMLD3nLl9vP6IE9iMrBB8Po0dUDXvfu9Rq5NGiCbCveVtkls6Llbm3+2mpz1rkcLjKTM+md1pvjuh9HVkoWPVJ7kJmcSYJb8x2KiIjI/ms1wS9aPv/8cyZMmEDbtm0BaNOmDd999x0zZ84E4IILLuCWW26JZonSyLJ3lfDC3LV89us2Nubaz3e1T4rhqD7tOKJnOqN6taVzaty+b+Qthg3f2SFv7VzYvABMABxue268o26FzNH2tns/7tfaGQPbt9vBLjzkLV1afcLyHj3sUHfmmVUhr08fO/zVUXmgnPUF63drvVtXsI5Sf9Wzf0meJHqk9GB0l9HVWu+6JHbB5dCvaREREWm4VvOJYl8tc5FijFG3q1ZiU14pz3yximk/bcTCYsxB7bh8VBaje7elZ7vE/fs5CPhg1Wew+G1YPhv8ZfZ0Cl2GwugbIetIu3XPo4E39mntWvj00+ohb2dVaxrt29uh7oorqlrw+vWr93N4Gwo2MG/rvGqtd5uKNhE0wcprOid0Jis1iyEdhlSGux4pPWgT20a/J0RERCSiWk3wi5Zjjz2WM888kxtvvJH09HRyc3MZMWIE06dP59xzz2Xq1KnRLlEaaFNeKZO/WMU7ocB33uHduHpsz/1r1QO7JSp7nh32fp4BpbkQ1wYOuwgOGmePwOlRt779Ygx8+y088QTMnGl340xIsEPdGWdU76bZvn2D384X8PHZxs94d/m7/LD1BwA8Dg+ZKZkckn4Ip/Q4pTLgdU/uTpxLLbMiIiISHQp+EdavXz/uuusuxowZg9Pp5LDDDmPSpElcdNFFPP7445x88smkpKREu0yph815pUyes4q3520E4NzDM7hmbK/9D3w7V8Lid2DJO7BrnT0oy0HjYOC50OtYjbxZF34/zJhhB74ffoC0NLjjDvjtb6Fnz0afZH5j4UbeXfEuM1fNJLcsl84JnbnusOs4MfNEuiR2wenYj8F5RERERJqQgl8TuPTSS7n00ksr90tKSvj++++xLIupU6cydOjQKFYndVUz8J0zNINrju5Fl/0JfEXb4efpduve5gWABT3GwJjb4eBTIDY5ssW3NPn58OKL8OST9mibvXrBM8/ApZfaLX2NyBf0MWfjHKYtn8Z3W77DaTk5qutRnN3nbI7ofITCnoiIiDRrCn5RMH/+fK699lqMMaSmpjJlypRolyT7YUt+KZO/WM3b8zZiMPsf+MqL4NeP7LC3Zo49QEvHgXD8g9D/LEju1CT1tyjr19th74UXoLAQxsZ44qQAACAASURBVIyBp56CU05p9Na9TUWbmL5iOu+teo+dpTvpEN+BawZdw/he4+mQ0KFR30tEREQkUhT8ouDII49k0aJF0S5D9tOW/FL+NWc1U3+0A9/ZQzO4ZmxPuqbtZYCVgB/WfGF35fz1Q/CVQEo3GH0DDDgH2h/cdF9AS/LDD3Z3znfftQPeOefATTfBkCGN+jb+oJ+vsr9i2oppfLPpGyzLYnSX0ZzT5xxGdxmt1j0RERE54Cj4iezB1vwyJs9ZxdQfNxI0duD7w9F7CXzGwKb/2c/s/TwdindAbKr9zN7AcyBjRKO3RrUKgYA9UMsTT9gDt6SkwC23wLXXQkZGo77V1uKtTF85nRkrZ7C9ZDvt49pz5cArOav3WXRKVMusiIiIHLgU/ERq2Jpfxr/mrOKtsMB3zdieZLTZQ+DLXQOLp9ldOXNXgzMG+pxgB77ex4Erpmm/gJaisBBeegkmTbKnZsjKsrt3XnZZvadcqE0gGODrTV8zbcU05m6aizGGI7ocwZ+G/4kxXcdoHj0RERFpEfSJRiRkW0EZ/5qzmjd/3EAwaDh7aFeuGdur9sBnjN2F85sn7akYsOzJ1EffAH1Pg7jUJq+/xcjOhn/+E557zh68ZdQoeOwxOP10cDZeF8vtJduZsXIG01dOZ2vxVtJj07m8/+WM7z2erkldG+19RERERJoDBT9p9WoGvglDuvKHo/cQ+ADyNsLHt9kTrKf3ht/cBwMmQIrCQoPMn29353znHXv+vQkT7Of3hg9vtLcImiDfbv6Wacun8WX2lwRMgBGdRnDb4bcxNmMsboem0BAREZGWScFPWq06B76AH378N3z+IGDguAdgxNWab68hgkH44AM78H31FSQlwR//CNddB5mZjfY2O0t38t7K95i+cjqbijbRJrYNl/S7hAm9J9AtuVujvY+IiIhIc6Xg14SMMRhjcGiAj6jaXlDGv75czZs/bMAfNEwYbAe+bul7GaVz8wL44HrYsgh6Hw/jHoO07k1XdEtTXAyvvAL/+AesWgXdu9vh7/LLIblx5jIMmiDfb/med1e8yxcbvsBv/AzrOIwbBt/AMd2OweP0NMr7iIiIiBwIFPwibN26dZx00kkcffTRvPXWW5xxxhm89NJLALz88svMnz+fp556KspVth5fLN/O9W8toNgb4KzBXbj26N57D3zlhXYL34//hoR2cPbLcMgZYFlNVnOLsnkzPP00PPss7NoFw4bB22/D+PHgapxfRzmlOcxcNZPpK6ezsXAjqTGpXNj3Qib0mUBmSmajvIeIiIjIgab1BL8bboCFCxv3noMG2SMO7sPy5ct56aWXuO+++xg5cmTl8bfffpu77rqrcWuSWgWDhslzVvH4f1dwcMdknrngMHq028fIkL9+BLNvhYLNcPjlcOw9EJvSNAW3NAsX2i16U6eC3w9nngk33wwjRzZKiDbGMG/rPKatmManGz7FH/QzuP1grhl0Dcd1P44Yp0ZWFRERkdat9QS/KOrevTsjRowAoEePHnz//ff07t2b5cuXM2rUqChX1/IVlvm4+Z1FfPLLNs4Y1JmHxg8kzrOX0SHzN9mDt/z6IbTvZ7fyZQxrsnpbjGAQPv7YDnyffw4JCXD11fYzfD17Nspb7CrbxazVs3h3xbusK1hHkieJ8w46jwl9JtAztXHeQ0RERKQlaD3Bbz9a5iIlISGhcvvcc8/lnXfe4eCDD+bMM8/EUpfBiFq1vYirXvuJdTkl3H3KIVw2KnPP3/NgAH58Hj5/wN7+zb0w8loN3lJXJSXw2mv283vLl0PXrvDoo3DFFZDa8GkujDHM3zafaSum8d/1/8UX9DGo3SAeHP0gx3c/nlhXbCN8ESIiIiItS+sJfs3E+PHjefDBB+nevTuPPPJItMtp0T5ZupWb3llEjMvB65cPZ2TP9D1fvHkhfHiDPYhLz2Ph5MehTVbTFdsSFBfb8+099RTk5MDgwfDGG3D22eBueHj2BXzMWDmDN399kzX5a0h0JzKhzwQm9JlAn7Q+jfAFiIiIiLRcCn5NLC0tjUMOOYRffvmFYcPUfTASgkHDpE9X8M/PVzGwawrPXjSEzqlxtV9cXgRzHoLvJ0N8W5gwBfqN1+AtdWEMTJtmP7OXnQ2nnWZvH3lko3wfgybI/639P55a8BTZRdn0T+/P/UfczwmZJxDv3svAPCIiIiJSScEvwjIzM/n555+rHfvwww+jVE3Ll1/q44apC/hi+Q7OHtKVB87oT6x7D8/zLf8YProFCrJhyO/srp1xDe+K2KosXWrPuffFF/ZgR1OnQiM9t2qM4dvN3/Lk/55kWe4y+qT1YfKxkxndZbS6SIuIiIjUkYKftBjLtxZy5Ws/sTmvlAfO6M9Fw7vVHhAKNsPHt8OyWdCuL1z2CXQb3vQFH8jy8+Hee+1uncnJMHkyXHklOPcyaE4dLNmxhEn/m8SPW3+kS2IX/jb6b5zc42QclubAFBEREakPBT9pET5cvJlbpy0mMdbF1CtHMKR7m90vCgZg3ovw2f0Q9MGxf7EHb3FpIu/9FgzaA7fcdhvs2GEP2PLgg9C2baPcfm3+Wp5a8BT/Xf9f2sS24Y5hd3B2n7M12bqIiIhIAyn4yQHNHwjy90+W8+8v1zCkexr/unAw7ZNrGdVxy2J78JZN86HnMaHBW3o0fcEHsv/9D669Fr77DkaMgNmzYciQRrn1tuJt/GvRv5i5aiYxzhiuPvRqLu13KQnuhH2/WERERET2qcUHP2NMq34eyBgT7RIiJrfYyx/fWsDXq3Zy0Yhu3HNKPzyuGl0BvcX24C3fTYb4NnDWi9D/LA3eUhc5OXDXXfDcc9CuHbz0ElxyCTga3u0yvzyfKT9P4Y1lbxAwAc496FyuHHgl6XF7GYFVREREROqsRQe/2NhYcnJySE9Pb5XhzxhDTk4OsbEtb16znzflc9Vr89lRVM6jEwZyztCM3S/augSmXgB5G2DwpXDcfRCX1vTFHqgCAXj+eTv05efbE6/fe2+jzMVX5i/jzV/f5MUlL1LoLWRcj3H8YdAfyEiq5c9RRERERBososHPsqwTgScBJ/CCMebhGue7Aa8AqaFr7jDGzA6duxO4HAgAfzTG/Keu79+1a1eys7PZsWNHw76QA1hsbCxdu3aNdhmNasb/srlzxhLaJHiYdtVIDs2oJYisnWuHvpgk+N3/QfeRTV/ogezbb+1unQsWwNix9iAu/fs3+Lb+oJ/3V73P5EWT2V6yndFdRnPD4Bs4qM1BDa9ZRERERPYoYsHPsiwn8AxwHJANzLMsa5Yx5pewy/4MvGOM+ZdlWYcAs4HM0PZ5QD+gM/CpZVl9jDGButTgdrvJytIk3C2FLxDkwY+W8fK36xie1YZnLhxM28SY3S9cOhNmXGE/w3fRdEhpWcE3orZuhdtvh1dfhS5d7OkZzjmnwV1jjTF8tuEz/rngn6zNX8vAtgN5+MiHObzj4Y1UuIiIiIjsTSRb/IYBq4wxawAsy5oKnA6EBz8DJIe2U4DNoe3TganGmHJgrWVZq0L3+y6C9UoztqOwnD+8+T9+XJvLZaOyuHPcwbidtTxj9uPzMPtWyBgG50+1n+uTffP54Omn4S9/gbIyuOMOu4tnYmKDbz1v6zwmzZ/E4p2LyUrJYtLYSRzT7ZhW2f1aREREJFoiGfy6ABvD9rOBmpOl3Qt8YlnWdUAC8Juw135f47Vdar6BZVlXAlcCdOvWrVGKluZn4cY8Jr42n7xSL0+eN4jTB+32owDGwOd/hbmPwUHjYMIUcMc1fbEHos8/tydh/+UXOPFEePJJ6NOnwbf9NfdXJv1vEt9s+ob28e2574j7OK3nabgcLfrRYhEREZFmKZKfwGr75/yaQ0yeD7xsjHncsqyRwGuWZfXfz9dijHkOeA5g6NChLXf4ylbs7XkbuHvmUtonxzD96iPo1zll94sCfnuqhgWvweBL4OR/gFPhYp82boSbb4Zp0yArC95/H049tcHdOjcWbuTpBU8ze+1skj3J3DTkJs4/+HxiXS1vkCERERGRA0UkPx1nA+FD9HWlqitnhcuBEwGMMd9ZlhULtN3P10oLVu4PcN8Hv/DmDxs4sndb/nneYaQl1DKJt7cEpl8Oy2fDUbfC0XdpqoZ9KS+Hxx+3J14PBuG+++DWWyGuYS2kOaU5/Hvxv5m2Yhouy8Xl/S/nd/1/R0pMLWFdRERERJpUJIPfPKC3ZVlZwCbswVouqHHNBuBY4GXLsvoCscAOYBbwpmVZT2AP7tIb+DGCtUozsq2gjImvz2fBhjyuHtuTW44/CKejljBXkgtvnQcbf4Rxj8GwK5q+2APN7Nlw/fWwahWMH28HwMzMBt2y2FfMK0tf4eWlL+MNeDmz95lcfejVtI9v3zg1i4iIiEiDRSz4GWP8lmVdC/wHe6qGKcaYpZZl3Q/8ZIyZBdwMPG9Z1o3YXTl/a+wZx5dalvUO9kAwfuAPdR3RUw5M89fvYuLr8yku9zP5wsGMG9Cp9gvzs+H1syB3DZz9MvQ7o0nrPOCsXg033ggffAAHHQT/+Q8cf3yDbukNeJm2YhrPLX6O3LJcjut+HNcddh1ZKRpJV0RERKS5seycdeAbOnSo+emnn6JdhjTAzqJyfvPEl6TGuXnukqH06ZBU+4Xbl9mhr7wQznsTso5s2kIPJCUl8NBD8Pe/g9sN99xjt/h5auk2u58CwQCz187mmYXPsKloE8M7DueGITfQv23D5/kTERERkfqzLGu+MWZobec0AoY0G/fOWkpJeYB3J46kV/s9hL4N38Ob54ArDn43GzoOaNoiDxTGwIwZcNNNsGEDXHghPPoodO7cgFsa5m6ay5P/e5IVu1bQt01f7vnNPYzsPFJTM4iIiIg0cwp+0iz895dtfLh4Czcf12fPoe/X2fDu7+wJ2S+aAWndm7bIA8WyZfDHP8Knn8LAgfDaa3DUUQ265aIdi/jH/H8wf9t8MpIyePSoRzkh8wQcVi1zKYqIiIhIs6PgJ1FXUObjzzOXcHDHJK4a07P2i+a/Yk/Z0PkwuOAdSGjbtEUeCAoK4P777Xn4EhPhqadg4kRw1f+v+Zq8NTz5vyf5fOPntIltw5+G/4kJvSfgdrobsXARERERiTQFP4m6hz/+lR2F5Tx38VA8rhotSMbAV3+HLx6EXsfBOa+AJyE6hTZXxsAbb9hTMmzbBpddBn/7G7Sv/6iaW4u3MnnhZN5f/T5xrjiuHXQtFx9yMfHu+EYsXERERESaioKfRNX3a3J484cNXHFkFodmpFY/GQzAx7fBvBfg0PPhtKdALU3VrV0Ll1wCX38Nhx9uT8I+bFi9b+cL+Ji8aDKvLn0Vg+HCvhdyxYArSItNa8SiRURERKSpKfhJ1JT5AtwxfTHd2sRz03EHVT/pK4MZV8CyWTDqevjNfZqYvaYffoBTTwWfD154AX73O3DU/5m77SXbueXLW1iwfQGn9DiF6w67js6J9R8MRkRERESaDwU/iZpJn65kXU4Jb/5+OHEeZ9WJ0jyYeiGs/xpOeAhGXhO9Ipur6dPhoovsUTpnz7bn5muAn7b+xC1f3kKJv4S/H/V3Tsw6sZEKFREREZHmQEPySVT8vCmf5+eu4dyhGRzRK2ygloIt8PLJsPEHOOtFhb6ajIHHH4ezz4ZBg+D77xsU+owxvLL0FX7/ye9J8iTx5rg3FfpEREREWiC1+EmT8wWC3PbuYtITPPzp5L5VJ3auhNfGQ2kuXPgO9DwmekU2R36/PU3Dv/4FEybAq69CXFy9b1fsK+aeb+7hk/WfcGy3Y/nrqL+S6ElsxIJFREREpLlQ8JMm9/zcNfyypYBnLxpCSlxosJbsn+CNs8HhhN9+aE/bIFUKC+G88+xunbfdBg891KDn+dbkreGGOTewvmA9Nw25id/2+60mYRcRERFpwRT8pEmt3lHEpE9XclL/jpzYv6N9cOV/4Z1LILE9XPwetOkR3SKbm02b4JRTYMkSePZZuOqqBt3uk3WfcPc3dxPriuW5455jeKfhjVSoiIiIiDRXCn7SZIJBw53TlxDrcnDf6f3sg4unwXtXQcf+cOG7dviTKosWwcknQ34+fPghnFj/5+/8QT+T5k/ilV9eYWDbgTw+9nE6JnRsxGJFREREpLlS8JMm88aPG/hxXS6PThhI+6RYKNgMs66DbiPggrchJinaJTYvH38M55wDKSn2PH2HHlrvW+0s3cmtX97KT9t+4tyDzuW2w2/D4/Q0YrEiIiIi0pwp+EmT2JJfyiMf/8roXm05e0hX++AXD4IJwBmTFfpq+ve/4Q9/gAED7Ja+Ll3qfauF2xdy85ybKfAW8LfRf+PUnqc2YqEiIiIiciDQdA4SccYY7nrvZwJBw0PjB9iDiGz9GRa8AcOuhLTMaJfYfASD9uAtEyfCCSfAV1/VO/QZY3hz2Zv87v9+R4wrhtfHva7QJyIiItJKqcVPIm7Wos18/ut2/nxyXzLaxNsH/3s3xKbAUbdEt7jmpLQULrkE3n0Xrr4a/vlPcNXvr2iJr4T7v7+fj9Z8xNiuY3nwyAdJ9iQ3csEiIiIicqBQ8JOIyi32ct8Hv3BoRiq/G5VlH1z1Kaz+HE74G8SlRbfA5mLHDjjtNPjhB3jsMbjpJqjn9ArrC9Zzwxc3sDpvNdcddh2/H/B7HJYa90VERERaMwU/iaj7P1hKYZmPR88aiNNhQTAAn9xjd+88/PfRLq95WL4cxo2DzZth2jQ466x63+rzDZ9z19d34XQ4efY3z3JElyMasVAREREROVAp+EnEfLF8OzMXbub6Y3tzUMfQ4C2L3oLtS2HCS+CKiW6BzcGXX8KZZ9pdOufMgeH1m1MvEAzwzMJneH7J8/RL78cTY5+gc2Lnxq1VRERERA5YCn4SEUXlfu6asYTe7RO55uie9kFvMXz+V+gyFPqdGd0Cm4PXX4fLLoOePeGjj6BH/Sau31W2i9u+uo3vt3zPWb3P4s7hdxLjVKgWERERkSoKfhIRj/7fr2wpKOPdiUcQ43LaB7+bDIVb4OyX6/38WotgDPz1r3DPPTB2LMyYAWn1e9ZxyY4l3PTlTeSW5nLfEfcxvvf4xq1VRERERFoEBT9pdPPW5fLa9+v57RGZDOkeCjRF2+GbSdD3VHvC9tbK64Urr4RXXoGLL4YXXgBP3SdSN8bw7sp3eeiHh2gX145Xx71Kv/R+EShYRERERFoCBT9pVGW+ALdPX0znlDhuOf6gqhNzHgJ/GfzmvugVF215eTB+PHzxBdx7r93iV4+WzzJ/GQ/+8CAzV81kVOdRPHzkw6TGpjZ+vSIiIiLSYij4SaN6+vNVrNlRzKuXDSMhJvTjtf1XmP+KPYpnes/oFhgt69bZI3euWmW39l1ySb1uk12YzU1zbmJZ7jImHjqRiQMn4nQ4G7dWEREREWlxFPyk0fyyuYBnv1zNWYO7clSfdlUnPv0LeBJgzO3RKy6afvwRTj3V7ub5ySf2c331MDd7LnfMvQOD4eljnmZMxpjGrVNEREREWizN6iyNwh8Icvv0xaTGu7n7lL5VJ9Z+BSv+D468CRLSo1dgtLz3nh30EhLg22/rFfqCJsjkhZP5w2d/oFNCJ94++W2FPhERERGpE7X4SaOY8s1almzK55kLBpMaHxqsJBiET/4MKRkwfGJ0C2xqxsCkSXDzzTBsGMyaBe3b1/k2+eX53DH3Dr7e9DWn9TyNP4/4M3GuuAgULCIiIiItmYKfNNi6ncU8/skKjjukA+MGdKw68fO7sGURnPkcuFtRWPH74YYb4Jln4Kyz4NVXIT6+zrdZlrOMG+fcyLaSbdw94m7O7nM2VmueBkNERERE6k3BTxrEGMOdM5bgcTn46xn9q4KJrww+ux86HQoDzo5ukU2pqAjOO8+ekP2WW+CRR8BR9x7V7618jwd/eJDUmFReOfEVBrYbGIFiRURERKS1UPCTBnl73ka+W5PDQ+MH0CE5turED89C/kY4Y3K9gs8BafNmOOUUWLQIJk+Gq6+u8y28AS8P/fgQ7654l+Edh/PIUY+QHtcKn40UERERkUal4Cf1tq2gjAdnL2Nkj3TOOzyj6kRxDsx9HPqcCFlHRa/AprR4MZx8sj1X3wcf2FM31NGWoi3cNOcmfs75mcv7X861h12Ly6G/oiIiIiLScPpUKfVijOHPM3/G6w/y0PgB1Z89++pR8BbDcfdHr8Cm9J//wNlnQ1ISzJ0LgwbV+Rbfbf6O2766DV/Qx6Sxkzi2+7ERKFREREREWqtW0gdPGtvsJVv57y/buOm4PmS2Tag6kbMa5r0Agy+BdgdFr8Cm8txzdktfVhb88EOdQ1/QBHl+8fNM/HQibePaMvXkqQp9IiIiItLo1OIndZZX4uUvs35mQJcULh+dVf3kp38BVyyMvTM6xTWVYBD+9Cd78JYTT4S334bk5DrdosBbwF1f38WcjXM4KfMk7j3iXuLddR/9U0RERERkXxT8pM7++tEy8kp8vHrZcFzOsEbjDd/Dsg/g6LsgqUP0Coy00lK49FKYNg2uugqefhpcdfurtGLXCm784kY2F23mjmF3cMHBF2iqBhERERGJGAU/qZOvVuzg3fnZXHt0Lw7pHNbCZYw9WXtSJxj5h+gVGGnG2NM1zJoFjz5qT9lQx8C2YtcKLpp9EQnuBF484UUGdxgcoWJFRERERGwKfrLfisv9/Om9JfRol8C1x/SqfvKXmZA9D057GjwJtd+gJXj1VTv0PfYY3HxznV9e4ivh1i9vJd4Vz9STp9IhoQW3jIqIiIhIs6HgJ/tt0qcryN5VyrSJI4l1O6tO+Mvh03uhfT8YdEHU6ou4TZvg+uth9Gi48cZ63eLhHx9mbf5anjv+OYU+EREREWkyCn6yX0q9Ad76cSNnDOrM4Zltqp+c9yLsWgcXzQCHs9bXH/CMsZ/n83phypR6TUr/4ZoPeW/Ve1wx4ApGdBoRgSJFRERERGqn4Cf75dNl2ygq93PO0IzqJ0p3wZePQM9joFcLnobgtdfgo4/gH/+A3r3r/PL1Bet54LsHGNx+MNcMuiYCBYqIiIiI7Jnm8ZP98t6CTXRKiWVEj/TqJ+Y+DmX5cNwD0SmsKWzeXNXF849/rPPLvQEvt355K26nm0eOegSXQ//eIiIiIiJNS8FP9mlnUTlfrtjB6YO64HCEjWC5ax388G8YdCF07B+1+iLKGLjySigvr3cXz8d/epxluct44IgH6JjQMQJFioiIiIjsnZoeZJ8+WLSZQNBw5mFdqp/47H6wnHDMXdEprCk0sIvnZxs+481f3+SivhdxdLejI1CgiIiIiMi+qcVP9um9BZs4pFMyB3VMqjqYPR9+ng5HXAvJnaNXXCQ1sIvnlqIt3PPNPfRt05cbh9RvFFARERERkcag4Cd7tWp7EYuz8xk/OKy1r2Ky9oR2MOr66BUXSQ3s4ukL+rjtq9sImACPjXkMj9MToUJFRERERPYtosHPsqwTLctablnWKsuy7qjl/D8sy1oYWlZYlpUXdi4Qdm5WJOuUPXtvQTYOC047NKxVb/ls2PAtjL0TYpL2/OIDWUUXz7/9rV5dPCcvnMzCHQu5Z8Q9dEvuFoECRURERET2X8Se8bMsywk8AxwHZAPzLMuaZYz5peIaY8yNYddfBxwWdotSY8ygSNUn+xYMGmYu2Mzo3u1onxxrHwz44L/3QNs+MPjS6BYYKRVdPEeNguuuq/PLv938LS8ueZHxvcczrse4CBQoIiIiIlI3kWzxGwasMsasMcZ4ganA6Xu5/nzgrQjWI3U0b10um/JKGR8+qMv8lyFnlT19g7MFjg0U3sXzpZfAWbcJ6XeW7uTOuXfSI6UHdwzbrZFbRERERCQqIhn8ugAbw/azQ8d2Y1lWdyAL+DzscKxlWT9ZlvW9ZVlnRK5M2ZP3Fmwi3uPk+H4d7ANlBTDnYcg8EvqcEN3iIqUBXTyDJsidc++kxFfCY2MeI84VF6EiRURERETqJpJNNlYtx8werj0PeNcYEwg71s0Ys9myrB7A55ZlLTHGrK72BpZ1JXAlQLdueo6qMZX5Any0ZAsn9u9IvCf0Y/LNJCjZCcc/AFZtf7wHuAZ28XxxyYt8v+V77h15L73SekWgQBERERGR+olki182kBG23xXYvIdrz6NGN09jzObQeg0wh+rP/1Vc85wxZqgxZmi7du0ao2YJ+WzZdgrL/Iw/rKt9ID8bvnsGBpwDnXf7ozjwGQNXXQVlZfXq4rlg+wKeWfgMJ2WexPje4yNUpIiIiIhI/UQy+M0DeluWlWVZlgc73O02OqdlWQcBacB3YcfSLMuKCW23BUYBv9R8rUTOewuy6ZAcw8ie6faBz/9qh6Nj745uYZHy+uvw4Yf16uKZV5bHbV/dRufEztwz8h6sltgaKiIiIiIHtIh19TTG+C3Luhb4D+AEphhjllqWdT/wkzGmIgSeD0w1xoR3A+0L/NuyrCB2OH04fDRQiayconLmLN/BZaOzcDos2LIIFk2FUX+E1BbYpXbzZnuC9lGj6jxRuzGGu7+9m52lO3n9pNdJ9CRGqEgRERERkfqL6LCMxpjZwOwax+6psX9vLa/7FhgQydpkzz5cvAV/0HDmYV1Ck7XfDXFpMPqmaJfW+MK7eE6ZUucunm8se4M5G+dw2+G30a9tvwgVKSIiIiLSMC1wPH5pqBkLNnFwxyT6dkqGlZ/C2i/hxEcgLjXapTW+ii6eTzwBffrU6aVLc5by+PzHGdt1LBf1vShCBYqIiIiINFwkn/GTA9DqHUUs2pjH+MGhmTcWvAqJHWDoZdEtLBIa0MWzyFvErV/eSnpsOg+MekDP9YmIiIhIs6YWP6nm/QWbcFhw+qAuEPDB6i+g3xng8kS7tMbVgC6exhju//5+NhVtYsoJU0iNbYEtoSIiIiLSoij4SSVjDO8t3MSoXm3pkBwLa7+C8gLofXy0S2t8Ci0ceAAAIABJREFUDeji+d6q9/h47cdcO+hahnQYEqECRUREREQaj7p6SqWf1u9iY26pPagLwIr/gMMNPcZGs6zG14Aunqt2reKhHx5ieKfh/H7A7yNUoIiIiIhI41Lwk0oz/reJOLeTE/p1tA+s/C9kjoKYpOgW1pga0MWz1F/KrV/dSrw7noePfBino24jgIqIiIiIRIu6egoAZb7A/7N379Fxlfe9/z9f3SVbV0u2bPmKsQEDNsaXXEhIaIAQmkAgQCCBQm6c/laTNG2anOScriQlJ6dt0l/bX9usNiaBEAMhECAYQmJDCoQQgSR8A19ljC+SZ3S/WdJIGs3z+2NGIHQdybNnRqP3ay0tzez97NnfhlkqH/bzPF/9eu8pffj8BZqTnSG1HZOaD0kb7kh0abF1BlM8/7HqH3Wk/Yh+dPmPVJpb6lGBAAAAQOzxxA+SpOcONqozENR1Fy8OHzi8I/x79YcTV1Ss+XzTnuL52zd/q0drH9XnLvic3lvxXo8KBAAAALxB8IOkcO++svxsXbJyXvhA7XapZKU0b2ViC4uVM5jiebLzpL5T+R2tK1unv1j/Fx4WCQAAAHiD4Ae1dffr+UONunbdImWkp0n93dKbL6bWbp4PPCA9+aT0ve9NaYrnwOCAvvb7rynN0vT9S7+vzLRMD4sEAAAAvMEaP+ipvac0MOh03VDT9jdflAb7pNUpEvyGpni+973SX/7llC79l53/on0t+/Svl/2rFs1d5FGBAAAAgLd44gc9vqte5yzI15qFBeEDtdulzDnSsksSW1gsDE3x7O2V7r13SlM8nz/5vLbu36pbzr1FH1r6IQ+LBAAAALxF8JvljjV3a+eJdl13cYXMLByUDu+QVl4mZWQnurwzN80pnv5uv/72pb/VuSXn6qsbv+phgQAAAID3CH6z3OO76mUmXXtRZBpj436psy411vdNc4pnMBTU//z9/1T/YL9+cOkPlJ2eAgEYAAAAsxpr/GYx55x+tbte7105TwsLc8MHD28P/57pwe8Mpnj+557/1M7Gnfr79/+9lhcu965GAAAAIE544jeL7TzRpuMtPfr4RRVvH6zdIZVfKBUsTFxhsTDNKZ4v+17W3Xvv1sfP/rg+etZHPSwQAAAAiB+C3yz22M565WSm6SMXRkJeb5t08hVp1Qxv2j7NKZ7Nvc365ovf1IrCFfrm5m96WCAAAAAQX0z1nKX6goN6aq9PV64p19zsyNfgyO8kF5JWz+DgN80pniEX0v/+w/9WV3+XfnTFj5SXmedxoQAAAED88MRvlnruYJM6egfe7t0nhad55s2TKjYkrrAzNc0pnve+fq/+eOqP+vqmr2t1cfTXAQAAADMBwW+WenxXnUrnZuv9Z5eGD4QGpdpnpLMvl9Ki3wglqUxziufuxt36913/riuXXakbV9/oYYEAAABAYhD8ZqH2nn49d7BJ16xbpIz0yFeg/lWpt3Xm7ubpnPTnfx6e4nnPPVFP8ezo69DXf/91lc8p13fe+51wL0MAAAAgxbDGbxb69Ws+9Q+GdP3waZ6Ht0uWJq38k8QVdiYeeEDatk36p3+Szjknqkucc/r2H7+tpp4m/ewjP1N+Vr7HRQIAAACJwRO/WejxnfVaNX+uzl9U8PbB2h3SkndJeSWJK2y6hqZ4vuc90le+EvVlDx16SL878Tt9ZcNXdGHZhR4WCAAAACQWwW+WOdHSo5rjbbru4oq3pzV2+iT/3pk5zXP4FM8p7OJ5sPWgflD9A72/4v26bc1tHhcJAAAAJBZTPWeZx3fVS5KuHdm0XZqZbRwefHDKUzx7Bnr0tRe+puLsYn3vfd9TmvHfPwAAAJDaCH6ziHNOj++q07vPKlFFUe7bJ2p3SAWLpflrElfcdPh80pe+NKUpns45fffl7+pE1wn9+Mofqzin2OMiAQAAgMTjUccssutku4619Oj69YvfPhjsk954Tlp9pTSTdrSc5hTPJ954Qk8dfUp/vu7Ptal8k8dFAgAAAMmBJ36zyOM765WdkaaPXFj+9sHjL0kD3TNvfd80pngebT+q//vK/9Xm8s2688I7PS4QAAAASB488Zsl+oMhPbn3lK5Ys0D5OZlvn6h9RkrPllZcmrjipmoaUzwDwYD+5vd/o9yMXP39+/9e6TO1ST0AAAAwDTzxmyVeONyk9p6Bd/buk8L9+1a8X8qak5jCpmqaUzx/UP0D1bbV6j8v/0/Nz5vvcZEAAABAcuGJ3yzx+K46zZuTpfevKnv7YMsbUusb0qoZtJvn0BTP//N/op7iuf3Ydj18+GF95vzP6H0V7/O4QAAAACD5EPxmgY7eAT17oFEfW7dImenD/pEf3h7+veqKxBQ2VX7/lKd41nXV6Tt//I7Wlq7Vly7+kscFAgAAAMmJ4DcLPP2aT/3B0OhpnrXbpdLVUsmKxBQ2FUNTPHt6op7iOTA4oK///usymb7/ge8rMy1z0msAAACAVETwmwUe31mvs8rm6MKKwrcP9p2Wjr00c3bzfPBB6YknpjTF8992/Ztea35Nf3fJ36libsXkFwAAAAApiuCX4k629qjqWKuuX18hG96n7+jzUmhAWj0D1vedPi19+cvhKZ5/9VdRXbK7cbd+uu+n+uQ5n9QVy2bIVFYAAADAIwS/FPerXfWSpGsvGmOaZ3aBtPQ9Cahqih56SGptDffsi3IXz3tfv1dF2UX66savelwcAAAAkPwIfinMOafHd9Vr84oSLSnJG34i3L9v5WVS+gxY97Zli3T++eEnflE42XlSz518TjeuvlG5GbkeFwcAAAAkP4JfCttb16Gjzd26fv2Ip33+vVKXb2a0cdi9W6qulu68Uxo+VXUCDx58UOlp6brl3Fs8Lg4AAACYGQh+KezxXfXKykjTRy5c+M4Th3eEf599efyLmqq775ZycqRbb41qeFd/lx6rfUwfWf4RleWVTX4BAAAAMAsQ/FLUwGBIT+45pSvOW6DC3BHTOWt3SIvWS/kLElNctLq7pfvvl268USopieqSx2ofU0+wR7euiS4oAgAAALMBwS9F/f5wk1q6+3XdyGme3S1SXfXMmOb58MNSZ2d4mmcUgqGgHjzwoDYs2KA189Z4XBwAAAAwcxD8UtRju+pVnJepD5wzYrrjkWclOWn1DOjft2WLdN550iWXRDX8uZPP6VT3Kd225jaPCwMAAABmFoJfCuoMDOiZ/Q362LpFykwf8Y+4drs0Z760cH1iiovWa69JL78sfeELUW/qsnX/Vi2eu1gfXPxBb2sDAAAAZhiCXwr6zWs+9QdDo6d5DgbDT/xWXSGlJfk/+rvvlrKypD/7s6iGv978unY17tKnz/u00tOi6/UHAAAAzBZJ/m//mI7HdtZrRekcXbSk6J0n6qqkQEc4+CWznh5p61bphhukefOiumTr/q2amzlX1626zuPiAAAAgJnH0+BnZleZ2SEzO2Jm3xjj/L+Y2e7Iz2Ezax927nYzq4383O5lnamkrq1Hr7zZquvWV8hGTpGs3SGlZUgr/yQxxUXrl7+U2tvD0zyj0NDdoB3Hduj6VddrTuYcj4sDAAAAZp4Mrz7YzNIl/VDSFZLqJFWb2Tbn3P6hMc65vxo2/kuS1kdel0j6tqSNkpykVyPXtnlVb6p4YvcpSRo9zVMK9+9b+h4ppzDOVU3Rli3SqlXSBz4Q1fCfH/y5QgrpU+d9yuPCAAAAgJnJyyd+myUdcc4ddc71S3pI0rUTjL9F0s8jrz8s6RnnXGsk7D0j6SoPa00Jzjk9vqtem5YXa0lJ3jtPtp+UGvdJq5J8N8/9+6WXXgq3cIhiU5eegR49cvgRfWjph1Qxd4ywCwAAAMDT4Fch6eSw93WRY6OY2TJJKyT991Svxdter+/UkcbTum794tEna3eEf69O8v59d98tZWZKt0c3u/epo0+ps7+TFg4AAADABLwMfmM9rnHjjL1Z0i+dc4NTudbM7jSzGjOraWpqmmaZqeOxXXXKSk/Tn164cPTJ2h1S0TKpdHX8C4tWICDdd590/fVSWdmkw0MupK37t+r8eefrorKL4lAgAAAAMDN5GfzqJC0Z9n6xpFPjjL1Zb0/zjPpa59wW59xG59zGsiiCQioLDob05J5T+tB581WYl/nOkwMB6egL4WmeUfbES4hHH5Xa2qLe1OUP9X/Qsc5jum3NbaM3sgEAAADwFi+DX7WkVWa2wsyyFA5320YOMrNzJBVLqhx2eLukK82s2MyKJV0ZOYZxvFjbrObT/fr4WJu6HPuDFOxN/mmeW7ZIK1dKl10W1fD799+v+XnzdeXyJF+3CAAAACSYZ8HPOReU9EWFA9sBSQ875/aZ2V1mds2wobdIesg554Zd2yrpuwqHx2pJd0WOYRzb9pxSUV6mLjtn/uiTtduljFxp+fviX1i0Dh2Sfv/78NO+KJrL17bVqtJXqVvOvUWZaZmTjgcAAABmM8/aOUiSc+5pSU+POPatEe+/M86190i6x7PiUkxtY5cuWlKkrIwRock56fB26awPSJm5iSkuGnffLWVkSHfcEdXw+w/cr5z0HN24+kZv6wIAAABSgKcN3BE//o6AFhbmjD7RfFhqP57cbRz6+qSf/lT6+MelBQsmHd7S26Kn3nhK16y8RoXZSd6TEAAAAEgCBL8U0BccVPPpfpUXjPFE73BkaWQyB7/HH5daWqLe1OWRw4+oP9SvT6/5tMeFAQAAAKmB4JcCGjr6JGnsJ361O6T5a6SiJaPPJYstW6Tly6XLL590aP9gvx46+JDeV/E+nVV4lve1AQAAACmA4JcCfB29kqSFRSOCX6BDOlGZ3E/7amul556LelOX37z5G7UEWmjYDgAAAEwBwS8F+DsDksZ44vfGc1IomNxtHH78Yyk9XfrMZyYd6pzT/Qfu19lFZ+s9C98Th+IAAACA1EDwSwGn2sPBr7xwxBq/2h1STpG0eHMCqopCf790773SNddICxdOOrymoUYHWw/SsB0AAACYIoJfCvB39Co/O0Nzs4d15wiFwsHv7A9J6Z527Zi+J56Qmpqi3tTlZ/t/puLsYl294mqPCwMAAABSC8EvBfg6AqPX9/l2Sd1Nyb2+b8sWaelS6crJazzReUIvnHxBN51zk3IyxtjEBgAAAMC4CH4pwN8ZGGOa5zOSTDp78p0yE+LoUenZZ6XPfz68xm8SDxx4QOlp6br53JvjUBwAAACQWiYNfma20syyI68/aGZfNrMi70tDtHwdAS0sGPEU7PB2afFGaU5pYoqazI9/HN7F87OfnXRoZ3+nHj/yuK5ecbVKc5P0/x4AAAAgiUXzxO9RSYNmdrakn0haIelBT6tC1PqDITWf7lP58B09TzdKp3ZKq5J0N8+BAemee6Q//VOpomLS4Y8dfky9wV5aOAAAAADTFE3wCznngpKuk/Svzrm/kjT5FoyIi4bOgJyTFg1f41f7TPj36iRd3/fkk1JDg3TnnZMODYaCevDgg9pUvknnlpwbh+IAAACA1BNN8Bsws1sk3S7pqcixTO9KwlQM9fB7xxq/2u3S3HKpfG2CqprEli3S4sXSVVdNOvR3J34nX7dPt53H0z4AAABguqIJfp+R9B5J33POvWlmKyTd721ZiJavY0Tz9sGBcOP2VVdIydjr7tgxaccO6XOfkzImbzOxdf9WLclfoksXX+p9bQAAAECKmvTfvJ1z+yV9WZLMrFhSvnPuH7wuDNHxtfdK0ttr/E68LPV1SquTdH3fT34SDqRRbOqyt2mv9jTt0Tc2f0PpaZPv/AkAAABgbNHs6vm8mRWYWYmkPZLuNbN/9r40RMPXEdCcrHTlDzVvr90upWVKZ30wkWWNLRgMB7+rrgr375vE/fvvV35mvq47+7o4FAcAAACkrmimehY65zolXS/pXufcBklJ2hxu9vF3BLSwKFc2NK3z8A5p+SVSdn5iCxvLr38t+XxRberi7/Zrx/Edun7V9crLzItDcQAAAEDqiib4ZZjZQkk36e3NXZAkfJ2Bt9f3tR2Tmg8lbxuHLVukhQvDbRwm8eDBB+Xk9KnzPhWHwgAAAIDUFk3wu0vSdklvOOeqzewsSbXeloVo+dp7VT7UvP3wjvDvZFzfd+KE9NvfRrWpS89Aj355+Je6fOnlWjR3UZwKBAAAAFJXNJu7PCLpkWHvj0r6hJdFIToDgyE1ne57+4lf7Q6p5Cxp3srEFjaWe+6RnAsHv0lse2Obuvq7aNgOAAAAxEg0m7ssNrPHzazRzBrM7FEzWxyP4jCxxq4+OSctLMqV+nukYy8m5zTPoU1drrxSWr58wqEhF9L9B+7XhaUXal3ZuvjUBwAAAKS4aKZ63itpm6RFkiokPRk5hgTzdwxr5fDm76VgQFp9ZYKrGsNvfyvV1UW1qcuLdS/qeOdx3bbmtrc3rAEAAABwRqIJfmXOuXudc8HIz08llXlcF6LwjubttdulzDnSsksSXNUYtmyRFiyQPvaxSYduPbBVC/IW6PJlbBwLAAAAxEo0wa/ZzG41s/TIz62SWrwuDJPztUeCX35OeGOXlZdJGdkJrmqE+vpwG4fPflbKzJxw6KHWQ3rF94puOfcWZaZNPBYAAABA9KIJfp9VuJWDX5JP0g2RY0gwX0dAeVnpKug6LHXWSauScJrnPfdIoZD0+c9POvT+A/crNyNXN6y+IQ6FAQAAALNHNLt6npB0TRxqwRT5O3tVXpgjO/JM+MCqKxJb0EiDg9KPfyxdfrl01lkTDm3ubdavj/5a16+6XoXZhXEqEAAAAJgdxg1+Zvbvktx4551zX/akIkTN1xFp3n54h1R+oVSQZD3vduwI9+/7p3+adOgjhx7RQGhAt553axwKAwAAAGaXiZ741cStCkyLrz2gy1dkS4dfkd73V4kuZ7QtW6SyMunaaycc1jfYp4cOPaRLF1+q5YXL41MbAAAAMIuMG/ycc/fFsxBMTXAwpMaugN7lDkpuUFqdZP37fD7pySelr35VysqacOjTR59Wa6CVhu0AAACAR6LZ3AVJqOl0n0JOWhqqCx8oX5vYgka6997wGr9JNnVxzmnrga1aVbxK7yp/V5yKAwAAAGYXgt8MNdTDr2ywUZpbLmXmJLiiYUIh6e67pcsuk1atmnBolb9KtW21uu08GrYDAAAAXpkw+EX69iXh4jEM9fAr7PNJRUsTXM0Izz4rHTsm3XnnpEO37t+qkpwSXX3W1d7XBQAAAMxSEwY/59ygpIl35kBC+Dp6JUk5PfVS0ZIEVzPCli3SvHnSdddNOOxYxzG9UPeCPnnOJ5WdnmSN5wEAAIAUEs1Uz5fM7D/M7P1mdvHQj+eVYUL+joDyMqW0zvrkeuLX0CA98YR0xx1S9sRh7oEDDygzLVM3nXNTfGoDAAAAZqlJG7hLem/k913DjjlJfxL7chAtX2dA5+f3ynoGkiv4/fSnUjA46aYuHX0deuKNJ3T1iqtVmlsan9oAAACAWWrS4OecuywehWBqfO29ujC3Q+qRVJgkwW9oU5dLL5XOPXfCoY/WPqreYC8tHAAAAIA4mHSqp5ktMLOfmNlvIu/XmNnnvC8NE/F3BLQquzX8Jlme+D33nPTGG5Nu6jIQGtCDBx7Uu8rfpXNKzolTcQAAAMDsFc0av59K2i5pUeT9YUlf8aogTG4w5NTQ1adl6c3hA4WLE1vQkC1bpOJi6ROfmHDY747/Tg09Dbp1za1xKgwAAACY3aIJfqXOuYclhSTJOReUNOhpVZhQ8+k+DYacyl2TNKdMyspLdElSU5P0+OPS7bdLORP3FNy6f6uWFSzTpYsvjVNxAAAAwOwWTfDrNrN5Cm/oIjN7t6QOT6vChIaat5cM+JNnmud990kDA9IXvjDhsN2Nu7W3ea8+fd6nlWbRfP0AAAAAnKlodvX8a0nbJK00s5cklUm60dOqMCFfe7iH39yAT6pYm+BqJDkXnuZ5ySXSmjUTDt26f6vys/J17UraQwIAAADxEk3w2yfpA5LOkWSSDim6J4XwiK8jIFNIWafrpaKPJroc6YUXpNpa6W//dsJhp06f0rMnntXta25XXmYSTE8FAAAAZoloAlylcy7onNvnnHvdOTcgqdLrwjA+f2dAizK6ZIN9yTHVc8sWqahIunHiB8E/P/hzmUyfOu9TcSoMAAAAgDTBEz8zK5dUISnXzNYr/LRPkgok8bgmgXwdAa2d2ykFlPjg19IiPfqo9D/+h5SbO+6wnoEePXr4UV2x7AqVzymPY4EAAAAAJprq+WFJd0haLOmfhx3vkvS/PKwJk/C19+r9OW3JEfx+9jOpv3/STV1+deRX6hroooUDAAAAkADjBj/n3H2S7jOzTzjnHo1jTZiEryOglfmR5u2FSxJXyNCmLu9+t3ThheMOC7mQHjjwgNaWrdW6snVxLBAAAACAFN3mLheY2fkjDzrn7prsQjO7StL/Jyld0o+dc/8wxpibJH1H4XYRe5xzn4ocH5T0WmTYCefcNVHUmvJCIaeGzoAqCpql3BIpe27iivnDH6SDB6V77plw2AsnX9CJrhP60sVfilNhAAAAAIaLJvidHvY6R9JHJR2Y7CIzS5f0Q0lXSKqTVG1m25xz+4eNWSXpm5Iucc61mdn8YR/R65y7KIr6ZpXm7j4FQ07zQw2Jn+a5ZYtUUCDddNOEw7Ye2KryOeW6fOnlcSoMAAAAwHCTBj/n3P87/L2Z/ZPCff0ms1nSEefc0ch1D0m6VtL+YWO+IOmHzrm2yL0ao6x71vK1h5u3F/b5pbKJe+Z5qrVVeuQR6bOflebMGXfYwdaDqvZX6683/LUy0qL57wwAAAAAYm06/fjyJJ0VxbgKSSeHva+LHBtutaTVZvaSmb0cmRo6JMfMaiLHPz6NOlOSryMgySmvp14qWpa4Qu6/X+rrk+68c8JhW/dvVW5Grj6x+hNxKgwAAADASJM+gjGz1xRefyeF1+qVSZp0fZ/ebv8wnBvxPkPSKkkfVHj30BfN7ALnXLukpc65U2Z2lqT/NrPXnHNvjKjtTkl3StLSpUnQzy4O/B29mqdOpQ0GEjfVc2hTl02bpIvGn43b3Nus37z5G92w+gYVZBXEsUAAAAAAw0Uz9+6jw14HJTU454JRXFcnafiWk4slnRpjzMuRpvBvmtkhhYNgtXPulCQ5546a2fOS1kt6R/Bzzm2RtEWSNm7cODJUpiRfZ0DLMxK8o2dlpbRvn3T33RMO+8WhXygYCurT5306ToUBAAAAGMukUz2dc8clFUn6mKTrJEW7sKxa0iozW2FmWZJu1ui1gb+SdJkkmVmpwlM/j5pZsZllDzt+id65NnDW8rUHtCavLfwmUU/8tmyR5s6Vbr553CF9g316+NDD+sDiD2hZQQKnpAIAAACYPPiZ2V9KekDS/MjPA2Y26b78kaeCX5S0XeFdQB92zu0zs7vMbKg1w3ZJLWa2X9Jzkr7mnGuRdJ6kGjPbEzn+D8N3A53N/B0Brc4eCn4JeOLX3i49/LD0qU+Fw984fn3012oNtOq2NbfFsTgAAAAAY4lmqufnJL3LOdctSWb2j5IqJf37ZBc6556W9PSIY98a9tpJ+uvIz/Axf5Q0fkfwWczX2atlWS1STmH4J94eeEDq7Z1wUxfnnLbu36rVxau1qXxTHIsDAAAAMJZodvU0SYPD3g9q7I1b4LFQyKmho0+L1JSYaZ7OST/6kXTxxdKGDeMOe9n3so60H9Fta26TGV8VAAAAINGieeJ3r6RXzOzxyPuPS/qJdyVhPK09/eofDGlesEEqPCf+BVRVSa+9Jv3Xf004bOv+rZqXM09Xr7g6ToUBAAAAmEg0m7v8s6TPSGqV1CbpM865f/W6MIwWbt7ulB84lZgnflu2hJu133LLuEOOdhzVi/Uv6pPnflJZ6VlxLA4AAADAeKJ54ifn3E5JOz2uBZPwdfSqSKeVEeyJf/Dr7JQeeigc+grG78n3wP4HlJWWpZtW3xTH4gAAAABMJJo1fkgS/s6AKqw5/CbeO3o++KDU0zPhpi4dfR3a9sY2/elZf6p5ufPiWBwAAACAiRD8ZhBfR0DL0oaCXxyf+A1t6rJunbRp/F06Hzn8iAKDAd265tb41QYAAABgUgS/GcTX3qtzcyM9/Arj+MTv1Vel3bvDT/vG2aVzIDSgnx/8ud618F1aXbw6frUBAAAAmBTBbwbxdQR0dlablJUv5RbH78Zbtki5udKnPz3ukGeOPaPGnkb92Zo/i19dAAAAAKJC8JtB/J0BLU5rDk/zjFd/vK4u6ec/lz75Salw7IbxQw3blxcs1/sq3hefugAAAABEjeA3Qzjn5OsIaEGoMb4buzz0kHT69ISbuuxu2q3XW17XrefdqjTjKwUAAAAkG/4tfYZo7e5XfzCkon5ffDd22bJFuuAC6d3vHnfI1v1bVZBVoI+t/Fj86gIAAAAQNYLfDOHrCKhA3coOno5f8Nu5U6qpmXBTl/rT9frdid/phtU3KC8zLz51AQAAAJgSgt8M4e8Y1sMvXjt63n23lJMj3Tp+e4YHDzwok+mWc2+JT00AAAAApozgN0P4OgNabE3hN/F44tfdLT3wgHTjjVLx2DuIdg9067Hax3TlsitVPqfc+5oAAAAATAvBb4bwd/RqaTybtz/2WHhHzy98YdwhT73xlE4PnNZta27zvh4AAAAA00bwmyF87QGtym6TMvOkvHne3/D556WSEul947dneOnUS6qYW6ELyy70vh4AAAAA00bwmyF8HQEtz2iJXw+/ysrwTp7j3CvkQnq14VVtLt/sfS0AAAAAzgjBb4bwdwa0SE3x2dilrU06cGDCFg6HWg+ps79Tm8o3eV8PAAAAgDNC8JsBws3be1UabIjP+r5XXgn/fs97xh1S7a+WJIIfAAAAMAMQ/GaA9p4BpQ90K2+wMz7Br7IyPMVz8/jTOKv91Vqav5TdPAEAAIAZgOA3A/iG9/ArisNUz8pK6YILpIJKdavcAAAgAElEQVSCMU8Phgb1asOrPO0DAAAAZgiC3wzg7+wd1sNvmbc3C4XCUz0nmOZ5sO2guga6CH4AAADADEHwmwF8HcOat3u9ucuBA1Jn58Tr+3ys7wMAAABmEoLfDOBrD2ixNctl5Ehz53t7s8rK8O8Jgl+Vv0rLC5Zrfp7HtQAAAACICYLfDODrCOjsrFZZ4RLve/hVVoYbt69ePebpYCionY07edoHAAAAzCAEvxnA39mrJWkt8dvYZYLG7QdaDqh7oJvgBwAAAMwgBL8ZwNcRULlr9L6Vw1Dj9onW9zWwvg8AAACYaQh+Sc45p/b2duUPtnu/sUsUjdur/FU6q/AsleaWelsLAAAAgJgh+CW5zt6gioON4Tdet3KorJTS0sZt3D4QGtDOBtb3AQAAADMNwS/J+Tp7teStHn4eT/Ucatyenz/m6f0t+9Ub7CX4AQAAADMMwS/JvaOHn5ebu0TRuL3aH17ft3HBRu/qAAAAABBzBL8k5+8IqMKa5dIypbnl3t1o//5JG7dX+ap0dtHZmpc7z7s6AAAAAMQcwS/J+dp7w0/8ipaE1995ZZLG7QODA9rdtJtpngAAAMAMRPBLcr6OgJZnRJq3e6myUpo3T1q1aszTr7e8zvo+AAAAYIYi+CU5f2dAFdYUn41dJmjczvo+AAAAYOYi+CW55vZOlYTavA1+ra3SwYOT9u9bXbxaxTnF3tUBAAAAwBMEvyTmnJN1nAy/8TL4TdK4vX+wX7sbWd8HAAAAzFQEvyTW1RfUvGBD+I2XwW+Sxu2vNb+mvsE+gh8AAAAwQxH8kph/eA8/Lzd3qayULrxQmjt3zNNV/iqZjPV9AAAAwAxF8EtivkgPv5BlSPkLvbnJ4OCkjdtr/DU6p+QcFWYXelMDAAAAAE8R/JLYUA+/UP4iKT3Dm5vs3y91dY0b/PoG+1jfBwAAAMxwBL8kNvTEL63Y4/V90rjBb2/TXvWH+rVpAcEPAAAAmKkIfknM3xHQ0rRmpRUv8+4mlZVSaal09tljnq72VyvN0rShfIN3NQAAAADwFMEviTV2dKlMbd5v7DJB4/Yqf5XOLTlXBVkF3tUAAAAAwFMEvyQWajupNDnvWjm0tkqHDo07zTMQDGhv016meQIAAAAznKfBz8yuMrNDZnbEzL4xzpibzGy/me0zsweHHb/dzGojP7d7WWeyyuyqC7/wKvi9/HL49zjBb0/THg2EBrR54dj9/QAAAADMDB5tFSmZWbqkH0q6QlKdpGoz2+ac2z9szCpJ35R0iXOuzczmR46XSPq2pI2SnKRXI9e2eVVvsukKDKgk6JcyJRV5NNVzqHH7prGf6FX5q5RmaVo/f7039wcAAAAQF14+8dss6Yhz7qhzrl/SQ5KuHTHmC5J+OBTonHONkeMflvSMc641cu4ZSVd5WGvSaegM7+jplCYVVHhzk8pKae3acRu31/hrtKZkjfKz8r25PwAAAIC48DL4VUg6Oex9XeTYcKslrTazl8zsZTO7agrXpjRfR0CLrUn9c8ql9MzY32CSxu29wV7tbd6rTQtZ3wcAAADMdJ5N9ZQ01jaRboz7r5L0QUmLJb1oZhdEea3M7E5Jd0rS0qUe9rpLAF97QMusWc6rHT337ZNOnx43+O1q3KVgKMjGLgAAAEAK8PKJX52k4allsaRTY4x5wjk34Jx7U9IhhYNgNNfKObfFObfRObexrKwspsUn2lDz9sx5y725wSSN22v8NUq3dF284GJv7g8AAAAgbrwMftWSVpnZCjPLknSzpG0jxvxK0mWSZGalCk/9PCppu6QrzazYzIolXRk5Nms0dnRpobUqvdijJ5lDjdtXrhzzdJW/SueXnq85mXO8uT8AAACAuPEs+DnngpK+qHBgOyDpYefcPjO7y8yuiQzbLqnFzPZLek7S15xzLc65VknfVTg8Vku6K3Js1uhrrVO6Qt61cqisDD/tG6Nxe89Aj/Y172OaJwAAAJAivFzjJ+fc05KeHnHsW8NeO0l/HfkZee09ku7xsr5kZu2RvW28CH4tLdLhw9Idd4x5elfjLgVdUJvL6d8HAAAApAJPG7hj+rK7I83bvdjcZZLG7VX+KmVYhi6af1Hs7w0AAAAg7gh+Sai7L6h5Aw1yMqlwcexvUFkppaeP27i9xl+jC0ovUF5mXuzvDQAAACDuCH5JyN8Z7uEXyJkvZWTH/gZDjdvnjN64pXugW/ta9mlTOev7AAAAgFRB8EtCvvZwK4eBfA+e9g0OSlVV407zfLXhVQ26QYIfAAAAkEIIfknI19GrxdbkTSuH11+fsHF7jb9GGWms7wMAAABSiae7emJ6Gtq7tdBaZaXLY//hkzRur/JXaW3pWuVm5Mb+3gAAAAASgid+Sai7pU6ZNqiMkmWx//DKSqmsTDrrrFGnuvq7dKD1ANM8AQAAgBRD8EtCobYT4Rde9PCboHH7zoadCrkQ/fsAAACAFEPwS0LpnR41b29ulmprJ5zmmZmWqbVla2N7XwAAAAAJRfBLQnm9p8IvYt3Db5LG7dX+aq0rW6ecjJzY3hcAAABAQhH8kkxv/6BKB/zqzponZcZ4g5Whxu0bN4461dHXoYOtB5nmCQAAAKQggl+S8XeGe/gF8ipi/+GVldK6dWM2bn+14VU5OW0sHx0KAQAAAMxsBL8k42sP9/BzhUti+8HB4ISN26v91cpOz2Z9HwAAAJCCCH5Jxtfeo0XWoox5MW7l8PrrUnf3pOv7stOzY3tfAAAAAAlH8Esync31yrag8spG99k7IxM0bu/o69DhtsP07wMAAABSFMEvyfS3vClJyipdHtsPrqyU5s+XVqwYdarGXyMnx8YuAAAAQIoi+CWbtkgPv1iv8ZugcXuVv0o56Tm6oPSC2N4TAAAAQFIg+CWZ7NN14RdFMQx+TU3SkSPjr+9rqNZF8y9SVnpW7O4JAAAAIGkQ/JJMfp9PpzOKpKzRLRembYLG7a2BVtW21TLNEwAAAEhhBL8kEhgYVGmwQd05i2L7wZWVUkbGmI3ba/w1ksTGLgAAAEAKI/glkYbOgBZbk/rzF8f2g4cat+fljTpV7a9Wbkauzi89P7b3BAAAAJA0CH5J5FRbryqsWWmxXN8XReP29fPXKzMtM3b3BAAAAJBUCH5JpK2pTjk2oKzS0S0Xpu2116SenjGDX0tvi97oeINpngAAAECKI/glkZ6mcA+//AUxbN4+QeP26oZqSWJjFwAAACDFEfySyGDrCUlSTtny2H1oZaW0YIG0fPRnVvuqlZeRp/PmnRe7+wEAAABIOgS/JJLe4UHz9gkat1c3VOviBRezvg8AAABIcQS/JJLbU6/TaflSTkFsPrCxUXrjjTGneTb1NOnNjjeZ5gkAAADMAgS/JFLY71dHVnnsPnCCxu01DfTvAwAAAGYLgl+S6AsOav5gg3rnxLCH3wSN26v8VZqbOVfnlpwbu/sBAAAASEoEvyTR0B7QYmtWqCDGwe+ii6Tc3FGnqv3h9X0ZaRmxux8AAACApETwSxJNjaeUZ31KL1kWmw8MBqXq6jGneTZ0N+h453HW9wEAAACzBMEvSZxuOCpJypsfo+bte/eO27h9qH8f6/sAAACA2YHglyQCzcclSYULV8bmAydq3O6vVn5Wvs4pPic29wIAAACQ1Ah+yaItHPzyymL0xK+yUiovl5aNnjpa7a/WhgUblJ6WHpt7AQAAAEhqBL8kkXm6Tqc1R8otis0HjtO43d/t18muk9q0gGmeAAAAwGxB8EsScwKn1Jq5IDYf1tgoHT067jRPSdq8kI1dAAAAgNmC4JckSvr96spZFJsPm2B9X5W/SgVZBVpdvDo29wIAAACQ9Ah+SaB/YFDlrkn9cyti84FDjds3bBh1qtpfrY0LNirN+EcPAAAAzBb8238SaGxqUL71SkVLYvOBlZXS+vWjGrfXn65X/el6pnkCAAAAswzBLwl0+N6QJGWVxmBHz4GBcRu3D63vo38fAAAAMLsQ/JJAd+ObkqS5C2IQ/PbulXp7xw1+RdlFOrvo7DO/DwAAAIAZg+CXBIKtxyRJJRUxCGTjbOzinFO1v1qbyjexvg8AAACYZUgASSCt/aR6XLbyi+af+YdVVkoLF0pLl77jcN3pOvm6fdq4YOOZ3wMAAADAjELwSwI5PfVqSF8wqtn6tIzTuL3GXyNJ2lzOxi4AAADAbEPwSwIFfT61Zy088w9qaJDefHPc/n0lOSVaWbTyzO8DAAAAYEYh+CWB0mCDenNjEPwmWN9X5a/SxgUbZbF4qggAAABgRvE0+JnZVWZ2yMyOmNk3xjh/h5k1mdnuyM/nh50bHHZ8m5d1JtJAd5sK1K1gwdLJB0+mslLKzBzVuP1k10k19jQyzRMAAACYpTK8+mAzS5f0Q0lXSKqTVG1m25xz+0cM/YVz7otjfESvc+4ir+pLFq2njmqBpLSSGAW/9eulnJx3HK7yV0mSNi2kfx8AAAAwG3n5xG+zpCPOuaPOuX5JD0m61sP7zUhd/nDz9pyy5Wf2QQMDUk3NuP37SnNLtaIgBn0CAQAAAMw4Xga/Ckknh72vixwb6RNmttfMfmlmS4YdzzGzGjN72cw+PtYNzOzOyJiapqamGJYeP4HmY5KkwoVnuOnKnj1jNm5/q3/fgk2s7wMAAABmKS+D31gpw414/6Sk5c65tZKelXTfsHNLnXMbJX1K0r+a2ahk5Jzb4pzb6JzbWFZWFqu64yrUely9Lkvz5y8+sw8aZ2OXY53H1NTbpI3l9O8DAAAAZisvg1+dpOFP8BZLOjV8gHOuxTnXF3l7t6QNw86divw+Kul5Ses9rDVhMrrqdUqlKsjLPLMPqqyUFi2Slix5x+Fqf7Uk+vcBAAAAs5mXwa9a0iozW2FmWZJulvSO3TnNbHgPg2skHYgcLzaz7MjrUkmXSBq5KUxKmNNbr+aM8jOfhjlO4/Zqf7Xm587XsoJlZ/b5AAAAAGYsz4Kfcy4o6YuStisc6B52zu0zs7vM7JrIsC+b2T4z2yPpy5LuiBw/T1JN5Phzkv5hjN1AU0Jxv1+nc8rP7EP8funYsXHX920sp38fAAAAMJt51s5BkpxzT0t6esSxbw17/U1J3xzjuj9KutDL2pJC32kVuE4F5nizvu/NjjfVEmhhmicAAAAwy3nawB0TC7YelyS5ojPs4TfUuP3ii99x+K3+feX07wMAAABmM4JfAnU2HJUkZZac4fq7yspw6BvRuL3aX60FeQu0JH/JOBcCAAAAmA0Ifgl0uuFNSdKcBWfQWL2/f8zG7c451TTUaHP5Ztb3AQAAALMcwS+BBlqOqc9lqGT+GTyR27NHCgRGBb8j7UfUGmhlmicAAAAAgl8iWftJ1btSLSrOm/6HjLOxy1D/PoIfAAAAAIJfAmV318lnZSrMPYPm7ZWVUkXFmI3bF81ZpMX5Z7hjKAAAAIAZj+CXQPkBn9oyz7B5+1Dj9mFCLqTqhnD/PgAAAAAg+CXKQK8KBtvUnbto+p/h80nHj48KfrVttero66B/HwAAAABJBL/EaT8pSRooOIONXVjfBwAAACAKBL8EGWw7IUlKKzrD4JeVNapxe7W/WhVzK7Ro7hk8TQQAAACQMgh+CdLdGG7enl12Bj38hhq3Z2e/dSjkQm/17wMAAAAAieCXML2Nb2rApauwdJpP/MZp3H6o9ZA6+zuZ5gkAAADgLQS/BBlsO65Tbp7Ki+dM7wN275b6+ljfBwAAAGBSBL8ESe88qTpXpkVFudP7gAk2dlmav1Tlc8rPsEIAAAAAqYLglyB5PT75rEzFedNs3l5ZKS1eHP6JGAwN6tWGV3naBwAAAOAdCH6JEOxT/kCTOrMXTr95+xiN2w+2HVTXQBfBDwAAAMA7EPwSoaNOktSbVzG960+dkk6cGD3N08f6PgAAAACjEfwSoT3cw88VTnNHz/HW9zVUa3nBcs3Pm38m1QEAAABIMQS/BAhFmrenz1s2vQ8Yaty+fv1bh4KhIOv7AAAAAIyJ4JcAvU1HFXRpmjvdHn6VldKGDe9o3H6g5YC6B7oJfgAAAABGIfglQH/LcflVogVFc6dxcb/06qtjTvOUWN8HAAAAYDSCXwK4thPT7+G3a9eYjdur/FVaUbhCpbmlMaoSAAAAQKog+CVAVne96l2pygtzpn7xGBu7DIQGtKthlzaXb45RhQAAAABSCcEv3oL9ygs0yqf5KsnLmvr1lZXSkiVSxdutIPa37FdPsIdpngAAAADGRPCLt856pSmkrtyFSkubRvP2MRq3V/vD6/s2LtgYiwoBAAAApBiCX7x1nJQk9c9dPPVr6+ulkyfHDH5nF52tebnzYlEhAAAAgBRD8Iu3SPN2K1o69WvHWt83OKBdjbuY5gkAAABgXBmJLmC2cW3H5Zwpe940evhVVoZ79w1r3P56y+vqDfYS/AAAAACMi+AXZ30tx9WqYi0oyp/6xUON27Pe3hSG9X0AAAAAJsNUzzgbbD2uOlem8sIp9vDr6xuzcXuVv0qrilepOKc4hlUCAAAASCUEvzhL6zipeleqRUVT7OG3a5fU3/+O4Nc/2K89jXvo3wcAAABgQgS/eBoMKrvXH3niN8XgN8bGLq81v6bAYID1fQAAAAAmRPCLp65TSnOD8qlMpXOyp3ZtZaW0dKm0aNFbh6r8VTIZ6/sAAAAATIjgF0/t4R5+PbmLpt68fYzG7TX+Gp1Tco4KswtjVSEAAACAFETwi6dID7/Bwim2cqirC/8MC359g33a3bibaZ4AAAAAJkXwi6dI8MsonmLwG2N9396mveoP9WvTAoIfAAAAgInRxy+OXPsJNbpilRVPcWpmZaWUkyNddNFbh6r91TKZNpRviHGVAAAAAFINT/ziKNh6XHWuVOUF09jRc0Tj9ip/lc4tOVcFWQUxrhIAAABAqiH4xVGo/cTUe/j19Uk7d75jmmcgGNDepr307wMAAAAQFYJfvIQGlXn6VKSHX2701+3cOapx+56mPRoIDWjzQoIfAAAAgMkR/OKly6+00IDqXJkWTqV5+xgbu1T5q5RmaVo/f32MiwQAAACQigh+8dIR7uHnV6lK506heXtlpbRsmbRw4VuHavw1WlOyRvlZ+bGuEgAAAEAKIvjFS6SVQ++cxUqfSvP2EY3be4O92tu8l/59AAAAAKJG8IuX9uPh30WLo7/m5Empvv4dwW93424FQ0GCHwAAAICoEfzipf2kWq1QJUVF0V8zxvq+an+10i1dFy+4OMYFAgAAAEhVngY/M7vKzA6Z2REz+8YY5+8wsyYz2x35+fywc7ebWW3k53Yv64wH135CJ0OlWjiVHn5DjdvXrXvrUJW/SufPO19zMud4UCUAAACAVJTh1QebWbqkH0q6QlKdpGoz2+ac2z9i6C+cc18ccW2JpG9L2ijJSXo1cm2bV/V6LdQWCX5FU2jlUFkpbdz4VuP2noEe7Wvep9vPn/E5GAAAAEAcefnEb7OkI865o865fkkPSbo2yms/LOkZ51xrJOw9I+kqj+r0Xigk66ybWiuHQGBU4/ZdjbsUdEEatwMAAACYEi+DX4Wkk8Pe10WOjfQJM9trZr80syVTvHZm6G5U2mCf6l2pyqMNfjt3SgMDo/r3ZViGLpp/kUeFAgAAAEhFXga/sXoWuBHvn5S03Dm3VtKzku6bwrUyszvNrMbMapqams6oWE+1hzPslJ74jbGxS42/RheUXqC8zLxYVwgAAAAghXkZ/OokLRn2frGkU8MHOOdanHN9kbd3S9oQ7bWR67c45zY65zaWlZXFrPCYi7RyOKUylUXbvL2yUlq+XCovlyR1D3RrX8s+2jgAAAAAmDIvg1+1pFVmtsLMsiTdLGnb8AFmtnDY22skHYi83i7pSjMrNrNiSVdGjs1MkebtA3MrlJEexf/kzo1q3L6zYacG3SDBDwAAAMCUebarp3MuaGZfVDiwpUu6xzm3z8zuklTjnNsm6ctmdo2koKRWSXdErm01s+8qHB4l6S7nXKtXtXqu46S60gpUWFQc3fiTJ6VTp0b178tIY30fAAAAgKnzLPhJknPuaUlPjzj2rWGvvynpm+Nce4+ke7ysL27aT+iUzmx9X5W/SmtL1yo3YwrtIAAAAABAHjdwR5hrP6njg/NUXhBlaKuslHJz32rc3tXfpQOtB5jmCQAAAGBaCH5ec05qP6Hjg6VaVDSFJ34bN0qZmZLC6/tCLkT/PgAAAADTQvDzWnezLNgbfQ+/QEDatWvU+r7MtEytLVvrYaEAAAAAUhXBz2sd4R09o+7h9+qrYzZuX1e2TjkZUT4xBAAAAIBhCH5ea387+JUXRrHGb8TGLh19HTrYepD1fQAAAACmjeDntUjw86lU8/OjaN5eWSmtWCEtWCApvL7PyRH8AAAAAEwbwc9r7SfVkzZXOfnFypysefsYjdur/FXKTs9mfR8AAACAaSP4ea39hBrTF0Q3zfPECcnnG7Wxy7qydcpOj+JpIQAAAACMgeDntY6TqnelWlgQxcYsY6zvO9x2mGmeAAAAAM4Iwc9LkR5+RwfmaWE0PfxqasKN29eGp3XuadrD+j4AAAAAZywj0QWktN42qf+0bvzQe9S7YdXk47//fekv/uKtxu2XLr5UOz6xQ6V5pR4XCgAAACCVEfy8FNnRM6dshXLmZE0+Pi0tvKPnMAvnLvSiMgAAAACzCFM9vRQalCo2SCVnJboSAAAAALMYT/y8tHiD9IX/TnQVAAAAAGY5nvgBAAAAQIoj+AEAAABAiiP4AQAAAECKI/gBAAAAQIoj+AEAAABAiiP4AQAAAECKI/gBAAAAQIoj+AEAAABAiiP4AQAAAECKI/gBAAAAQIoj+AEAAABAiiP4AQAAAECKI/gBAAAAQIoj+AEAAABAiiP4AQAAAECKI/gBAAAAQIoj+AEAAABAiiP4AQAAAECKM+dcomuICTNrknR8GpeWSmqOcTlANPjuIZH4/iFR+O4hkfj+IVHi9d1b5pwrG+tEygS/6TKzGufcxkTXgdmH7x4Sie8fEoXvHhKJ7x8SJRm+e0z1BAAAAIAUR/ADAAAAgBRH8JO2JLoAzFp895BIfP+QKHz3kEh8/5AoCf/uzfo1fgAAAACQ6njiBwAAAAApblYEPzO7yswOmdkRM/vGGOezzewXkfOvmNny+FeJVBXF9+8OM2sys92Rn88nok6kHjO7x8wazez1cc6bmf1b5Lu518wujneNSE1RfPc+aGYdw/7ufSveNSI1mdkSM3vOzA6Y2T4z+8sxxvC3D56I8vuXsL9/GfG6UaKYWbqkH0q6QlKdpGoz2+ac2z9s2OcktTnnzjazmyX9o6RPxr9apJoov3+S9Avn3BfjXiBS3U8l/Yekn41z/iOSVkV+3iXpPyO/gTP1U0383ZOkF51zH41POZhFgpK+6pzbaWb5kl41s2dG/P9d/vbBK9F8/6QE/f2bDU/8Nks64pw76pzrl/SQpGtHjLlW0n2R17+U9CEzszjWiNQVzfcP8IRz7veSWicYcq2kn7mwlyUVmdnC+FSHVBbFdw/whHPO55zbGXndJemApIoRw/jbB09E+f1LmNkQ/CoknRz2vk6j/wG8NcY5F5TUIWleXKpDqovm+ydJn4hMN/mlmS2JT2lA1N9PwAvvMbM9ZvYbMzs/0cUg9USW7qyX9MqIU/ztg+cm+P5JCfr7NxuC31hP7kZuZRrNGGA6ovluPSlpuXNuraRn9fbTZ8Br/O1DouyUtMw5t07Sv0v6VYLrQYoxs7mSHpX0Fedc58jTY1zC3z7EzCTfv4T9/ZsNwa9O0vAnKIslnRpvjJllSCoUU1QQG5N+/5xzLc65vsjbuyVtiFNtQDR/H4GYc851OudOR14/LSnTzEoTXBZShJllKvwv3Q845x4bYwh/++CZyb5/ifz7NxuCX7WkVWa2wsyyJN0saduIMdsk3R55fYOk/3Y0OERsTPr9G7Gu4BqF54MD8bBN0p9Fdrh7t6QO55wv0UUh9ZlZ+dBaejPbrPC/j7Qktiqkgsj36ieSDjjn/nmcYfztgyei+f4l8u9fyu/q6ZwLmtkXJW2XlC7pHufcPjO7S1KNc26bwv+AtprZEYWf9N2cuIqRSqL8/n3ZzK5ReCeoVkl3JKxgpBQz+7mkD0oqNbM6Sd+WlClJzrn/kvS0pKslHZHUI+kziakUqSaK794Nkv4fMwtK6pV0M//BFTFyiaTbJL1mZrsjx/6XpKUSf/vguWi+fwn7+2f8nQUAAACA1DYbpnoCAAAAwKxG8AMAAACAFEfwAwAAAIAUR/ADAAAAgBRH8AMAAACAFEfwAwDMeGY2aGa7zex1M3vSzIo8us8fY/AZH47UutvMTpvZocjrn5nZRjP7t1jUCgDAcLRzAADMeGZ22jk3N/L6PkmHnXPfS3BZkzKz5yX9jXOuJtG1AABSG0/8AACpplJShSSZ2QfN7KmhE2b2H2Z2R+T1MTP7OzPbaWavmdm5kePfMbN7zOx5MztqZl8edv3pYZ/7vJn90swOmtkDZmaRc1dHjv3BzP5t+P0nM7zeSB33mdmOSK3Xm9n3I7X+1swyI+M2mNkLZvaqmW03s4Vn+j8gACD1EPwAACnj/2/v/l2qjOI4jr+/2A+DQAikJUhsSCzBCoJ+bEERDi1C/QMtBdHgGDQ3NURES1MRNDQEkeDQ0hSEmghFUEQthQ1CIGb5bbgnNJPS8mae3i+4PNzzPIfvOdPDh3PuuRHRAhwB7i2xy0Rm7gWuAQPz2ruAY8B+4OK3kLXAHuA80A10AociohW4DhzPzMNA+29NZM4OoA84AdwEHmZmDzAF9JVxXQH6M3MfcAP451c6JUl/n8FPklSDTRExAnwAtgBDS+x3t1yfAB3z2u9n5nRmTgDvga2L9H2cmW8zcxYYKf27gJeZ+ao8c3tZs/jRg8ycAcaAFmCwtI+VejuB3cBQmf8FYNsf1pQkVcjgJyrHK4oAAAEcSURBVEmqwVRm9gLbgQ3A2dL+me/fda0L+k2X6xdg3SLti9372TOxvGH/0jRACZczOffD/Nl59cYzs7d8ejLz6AqPQZJUAYOfJKkamTkJnAMGyjbI10B3RGyMiDYa20Cb6RnQGREd5fvJJtd7DrRHxAGAiFgfEbuaXFOStAYZ/CRJVcnMYWAUOJWZb4A7wFPgFjDc5NpTwBlgMCIeAe+AySbW+wT0A5ciYpTGltODzaonSVq7/DsHSZJWUERszsyP5ZTPq8CLzLy82uOSJP3fXPGTJGllnS4HrYwDbTRO+ZQkaVW54idJkiRJlXPFT5IkSZIqZ/CTJEmSpMoZ/CRJkiSpcgY/SZIkSaqcwU+SJEmSKmfwkyRJkqTKfQWbv0z6alDtmAAAAABJRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 1080x576 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "s = 20\n",
    "plt.plot(run_zoj[:s], val_zoj[:s], '-', label='zoj')\n",
    "plt.plot(run_hozog[:s], val_hozog[:s], '-', label='hozog')\n",
    "plt.plot(run_cg[:s], val_cg[:s], '-', label='cg')\n",
    "#plt.plot(run_fp[:s], val_fp[:s], '-', label='fp')\n",
    "plt.plot(run_rv[:s], val_rv[:s], 'r-', label='rv')\n",
    "plt.xlabel('Running Time')\n",
    "plt.ylabel('outer loss')\n",
    "plt.legend(loc='lower right')\n",
    "plt.gcf().set_size_inches(15, 8)\n",
    "plt.legend()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Saved plots for ONE with T=20, lr=0.01 for ZOJ"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dataset 20newsgroup, train_samples=5657, val_samples=5657, test_samples=7532, n_features=130107, n_classes=20\n",
      "logistic_20news with ZOJ, mu = 0.01, beta = 0.02 and T = 20\n",
      "o_step=0 (2.92e-01s) Val loss: 8.9652e-01, Val Acc: 76.21%\n",
      "          Test loss: 1.1634e+00, Test Acc: 66.81%\n",
      "          l2_hp norm: 1.7930e+00\n",
      "          l2_hp: -1.7930e+00\n",
      "o_step=50 (2.82e-01s) Val loss: 5.1869e-01, Val Acc: 85.88%\n",
      "          Test loss: 8.5076e-01, Test Acc: 76.33%\n",
      "          l2_hp norm: 1.5102e+00\n",
      "          l2_hp: -1.5102e+00\n",
      "o_step=100 (2.83e-01s) Val loss: 5.1659e-01, Val Acc: 86.28%\n",
      "          Test loss: 8.5404e-01, Test Acc: 76.57%\n",
      "          l2_hp norm: 1.5107e+00\n",
      "          l2_hp: -1.5107e+00\n",
      "o_step=150 (2.83e-01s) Val loss: 5.1519e-01, Val Acc: 86.44%\n",
      "          Test loss: 8.5389e-01, Test Acc: 76.67%\n",
      "          l2_hp norm: 1.5112e+00\n",
      "          l2_hp: -1.5112e+00\n",
      "o_step=200 (2.83e-01s) Val loss: 5.1380e-01, Val Acc: 86.53%\n",
      "          Test loss: 8.5245e-01, Test Acc: 76.70%\n",
      "          l2_hp norm: 1.5116e+00\n",
      "          l2_hp: -1.5116e+00\n",
      "o_step=250 (2.83e-01s) Val loss: 5.1247e-01, Val Acc: 86.55%\n",
      "          Test loss: 8.5062e-01, Test Acc: 76.69%\n",
      "          l2_hp norm: 1.5115e+00\n",
      "          l2_hp: -1.5115e+00\n",
      "o_step=300 (2.82e-01s) Val loss: 5.1126e-01, Val Acc: 86.58%\n",
      "          Test loss: 8.4878e-01, Test Acc: 76.75%\n",
      "          l2_hp norm: 1.5116e+00\n",
      "          l2_hp: -1.5116e+00\n",
      "o_step=350 (2.82e-01s) Val loss: 5.1017e-01, Val Acc: 86.60%\n",
      "          Test loss: 8.4707e-01, Test Acc: 76.75%\n",
      "          l2_hp norm: 1.5113e+00\n",
      "          l2_hp: -1.5113e+00\n",
      "o_step=400 (2.83e-01s) Val loss: 5.0922e-01, Val Acc: 86.65%\n",
      "          Test loss: 8.4553e-01, Test Acc: 76.81%\n",
      "          l2_hp norm: 1.5111e+00\n",
      "          l2_hp: -1.5111e+00\n",
      "o_step=450 (2.83e-01s) Val loss: 5.0837e-01, Val Acc: 86.65%\n",
      "          Test loss: 8.4417e-01, Test Acc: 76.82%\n",
      "          l2_hp norm: 1.5107e+00\n",
      "          l2_hp: -1.5107e+00\n",
      "o_step=499 (2.82e-01s) Val loss: 5.0764e-01, Val Acc: 86.71%\n",
      "          Test loss: 8.4300e-01, Test Acc: 76.83%\n",
      "          l2_hp norm: 1.5105e+00\n",
      "          l2_hp: -1.5105e+00\n",
      "HPO ended in 1.41e+02 seconds\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAXoAAAEICAYAAABRSj9aAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4xLjEsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy8QZhcZAAAdFklEQVR4nO3de3Bc533e8e+zCywJgBJJkaAskRQpXqzYSR3JZuT7JY6VqE0sudOxLdpJrbRTt3Hk2q7tjOLxKBpl2qYXx21iTVrFVX2XqqquzThsZcWy23Eiu6RsOQ6lSFjSogldsBDFCxYgrvvrH3tArJYAsQAB7O45z2dmZ8/l3XN+ByQenH333XMUEZiZWXrlml2AmZktLwe9mVnKOejNzFLOQW9mlnIOejOzlHPQm5mlnIPezCzlHPTWMiQ9JeltF7iNmyV9d6lqMksDB71Zk0jqaHYNlg0OemsJkr4IXAH8maSypN+R9BpJfyXppKQfSXpLTfubJR2RNCTpJ5LeK+llwH8CXpts4+Q8+/xVST+UdFrSMUm3161/Q83+j0m6OVneJelTko5KOiXpu8myt0jqr9vG2Xcpkm6XdL+kL0k6Ddws6VpJDyf7eFbSZyQVal7/s5IelPSCpAFJn5D0EkkjkjbUtHuVpEFJnYv7F7BUiwg//GiJB/AU8LZkejNwHPh7VE9Irkvme4Ee4DRwVdL2MuBnk+mbge82uL+3AH8n2f4rgAHgHcm6K4AhYC/QCWwArk7W3Ql8J6kxD7wOWJVsr/88x3Q7MAG8I9lnF/Aq4DVAB7AdeBz4cNL+IuBZ4KPA6mT+1cm6/cBv1ezn08AfN/vf0I/WfPiM3lrVrwP7I2J/RFQi4kHgINXgB6gAPyepKyKejYhDC91BRHwnIn6cbP+vgXuANyer3wv8RUTcExETEXE8Ih6VlAP+EfChiHg6IqYi4q8iYqzB3T4cEV9L9nkmIh6JiO9FxGREPAX855oafg14LiI+FRGjETEUEd9P1n0++RkhKU/1D9IXF/ozsGxw0Fur2ga8M+nSOJl0w7wBuCwihoF3A/8MeFbSn0v6mYXuQNKrJX076fI4lWxvY7J6K3B4lpdtpHp2Pdu6Rhyrq+Glkr4h6bmkO+dfNVADwNeBl0vaQfXdzqmI+H+LrMlSzkFvraT2UqrHgC9GxLqaR09E/AFARDwQEddR7bb5W+BPZ9nGfL4C7AO2RsRaqv37qtn/zlle8zwwOse6YaB7eiY50+49zzEC/ElS/+6IuBj4RAM1EBGjwH1U33n8Bj6bt/Nw0FsrGQB2JNNfAt4u6Vck5SWtTj7s3CLpUkk3SOoBxoAyMFWzjS21H2iex0XACxExKula4D01674MvE3SuyR1SNog6eqIqAB3A38o6fKkttdKWgU8CaxOPuTtBD5Jte9+vhpOA+XkXclv1az7BvASSR+WtErSRZJeXbP+C1Q/k7gh+XmZzcpBb63kXwOfTLpp3g3cSPUMd5Dq2e3Hqf6fzVH9gPIZ4AWqfdofSLbxEHAIeE7S8/Ps7wPAHZKGgNuoniEDEBE/pfp5wEeTfTwK/Hyy+mPAj4EDybp/A+Qi4lSyzc8CT1M9w3/RKJxZfIzqH5ghqu9K/ltNDUNUu2XeDjwH9AG/WLP+L6l+VvGDpH/fbFaK8I1HzNqVpIeAr0TEZ5tdi7UuB71Zm5L0C8CDVD9jGGp2Pda63HVjqSbpUPLlqfrHe5td24WQ9HngL6iOuXfI23n5jN7MLOV8Rm9mlnItd1GljRs3xvbt25tdhplZW3nkkUeej4j6720ALRj027dv5+DBg80uw8ysrUg6Otc6d92YmaWcg97MLOUc9GZmKeegNzNLOQe9mVnKOejNzFLOQW9mlnItN47ezGxa9Z6nUImgcvZ5Zjoq1eepZPmL2lbq2kYwVZnZxnTbqUq1TdTuo/Li/Z237fSjwovbRk3byhz1J3VOt33Jxat5z6uvWPKfo4PezOYVEYxNVhibqDA6OcXoxBSjE5XkeYqxyWQ6eR5L1o9N1rSrnU7Wzba96W1NTFWoZOxSXNdcsc5Bb9bqKpVgdHKKkfEpzoxXn0fGJ2emJ6Y4Mz6ZLJ9pc2ZiZtlIsn7m9VOMT07Nv/NlEMD4ZIWxycqit5ETrO7MVx8dOVZ35ikkz6s7c1zSU2B1R3V6VfK8ujNPZz5HLifyEjlBLickyKm6bHp6el11OplP1ueT5TNtRT4HqmtbfX3yOqrbyyfLzmkrkcvN7FuarjHZT66ubc20cry4bV0Ny8VBb/OqVILy+CSnRiY4OTLBqTMTnDwzzpnxKSYrwWQlmJqqnJ2eTKanKsHEVDBVqSTPM+unKsFEpbpucipmfe30dHV95ez2KhHn/qLk6n/Jq7/Q09O52QIid+4v43TbcwOiuo/xycqsoT0T2AsL5Jygu9BBVyFPdyFPV2f1uafQwcY1q+hOlq/qyC/Tv+78VnXkWNVZF8Qd+bNBvbozz6qa4F5Vs25VR57OfPXnas3joM+QsckpTp0N6prQHhnn1Jnp6Zn1p2qWL+YtdE7QkcvRka+eHXXkREc+lzyLjlyuZnl1viM5k+rqzJNf1UHn2dfObCcvEcz0cU5V6vtnq32xUdcfWtuPevaPUG0/as36+n7U6X0U8rmzoby2q5PLLl5dDejpoC50nA3namh3vGj92Tad1WWrOnIOQVt2Dvo2VakEz5w6w9HjI7wwPM7JMxOcTkK7NqxPJ+F98sw4oxNzv/2WYG1XJ2u7OlnX1cnFXZ1ccUk366aXdXfOrO8usLark+5C/mz4duZy5PPJcxLey/lW1Mwa56BvcSPjkxwZHObwYPns8+HBYX7yfHnW4O7qzJ8N5umwfsWWmYC+OAny6eBe11UN7YtWdziYzVLKQd8CIoKB02NJiNcEeqnMM6dGz7aTYOv6bnb09vC6nRvY2buG7Ru76V2zirXJWfjqzub15ZpZa3LQr6DRiSl+8vxwzZl5NdSPDJYZHp/5EK+nkGfnpjW8escGdmzsYeemNezsXcO2Dd0OcjNbMAf9EosIBstjHC4Nc+T5ModLSbfL82X6T5yh9ha9m9d1saO3h3fu2crO3h529q5h56Y1bLpolT+gM7Ml46BfAo8cPcGXv3+Uw8nZ+dDo5Nl1XZ15dvT2cPXW9fyDV25hZ+8advT2sGPjGroKPjs3s+XnoF8Cn3moj4ePHOdV29bzjqs3V8/ON61hR+8aLrt4tT/kNLOmctAvgb5Smete/hL+eO81zS7FzOwcvnrlBRoZn+Tpk2fYvWlNs0sxM5tVQ0Ev6XpJT0gqSrp1lvWflvRo8nhS0smadVdI+qakxyU9Jmn70pXffEcGh4mAXQ56M2tR83bdSMoDdwLXAf3AAUn7IuKx6TYR8ZGa9h8EavswvgD8y4h4UNIaYPFXR2pBxVIZwGf0ZtayGjmjvxYoRsSRiBgH7gVuPE/7vcA9AJJeDnRExIMAEVGOiJELrLml9JWGyOfEtg09zS7FzGxWjQT9ZuBYzXx/suwckrYBVwIPJYteCpyU9FVJP5T075J3CKlRLJXZvqGbQoc/7jCz1tRIOs02NnCuaxneBNwfEdNf8+wA3gh8DPgFYAdw8zk7kN4v6aCkg4ODgw2U1Dr6SmX3z5tZS2sk6PuBrTXzW4Bn5mh7E0m3Tc1rf5h0+0wCXwNeWf+iiLgrIvZExJ7e3t7GKm8B45MVjh4fYfemi5pdipnZnBoJ+gPAbklXSipQDfN99Y0kXQWsBx6ue+16SdPp/VbgsfrXtqunjg8zVQmf0ZtZS5s36JMz8VuAB4DHgfsi4pCkOyTdUNN0L3BvxMzVXJIunI8B35L0Y6rdQH+6lAfQTNMjbhz0ZtbKGvpmbETsB/bXLbutbv72OV77IPCKRdbX0voGykiws9dBb2aty0NFLkBxsMyW9V2+OJmZtTQH/QXoGxhil8/mzazFOegXaaoSHHl+mN2XesSNmbU2B/0iHXthhPHJis/ozazlOegX6eyIm0sd9GbW2hz0i9TnoZVm1iYc9IvUVxri0otXcfHqzmaXYmZ2Xg76RTrsa9yYWZtw0C9CRFAslX2NGzNrCw76RXj21CjD41Ps9Bm9mbUBB/0i9PmuUmbWRhz0i+CLmZlZO3HQL0KxNMT67k429BSaXYqZ2bwc9ItQTEbcSLPdfMvMrLU46BcoIpLbB3rEjZm1Bwf9Ah0fHufkyIT7582sbTjoF6hvwCNuzKy9OOgXqDjoETdm1l4c9AtUHBiip5DnsrWrm12KmVlDHPQLVBz0iBszay8O+gXqG/CIGzNrLw76BTh1ZoLS0Jj7582srTjoF6Doa9yYWRty0C/AYV/jxszakIN+AfpKQxQ6cmy9pLvZpZiZNcxBvwDFUpkdG3vI5zzixszah4N+AfpKZXZf6hE3ZtZeHPQNGhmf5OmTZ9jV6/55M2svDvoGHRkcJgJ2X+qgN7P24qBvkO8qZWbtykHfoL7SEPmc2L6hp9mlmJktSENBL+l6SU9IKkq6dZb1n5b0aPJ4UtLJuvUXS3pa0meWqvCVViyV2bahm0KH/zaaWXvpmK+BpDxwJ3Ad0A8ckLQvIh6bbhMRH6lp/0HgmrrN/D7wf5ak4ibpK5X9jVgza0uNnJ5eCxQj4khEjAP3Ajeep/1e4J7pGUmvAi4FvnkhhTbT+GSFo8dH3D9vZm2pkaDfDByrme9Plp1D0jbgSuChZD4HfAr4+Pl2IOn9kg5KOjg4ONhI3SvqqePDTFWC3b5qpZm1oUaCfravgcYcbW8C7o+IqWT+A8D+iDg2R/vqxiLuiog9EbGnt7e3gZJWlkfcmFk7m7ePnuoZ/Naa+S3AM3O0vQn47Zr51wJvlPQBYA1QkFSOiHM+0G1lfQNlJNjpL0uZWRtqJOgPALslXQk8TTXM31PfSNJVwHrg4ellEfHemvU3A3vaLeShelepzeu66Crkm12KmdmCzdt1ExGTwC3AA8DjwH0RcUjSHZJuqGm6F7g3Iubq1mlbfQNDHnFjZm2rkTN6ImI/sL9u2W1187fPs43PAZ9bUHUtYKoSHHl+mDfu3tjsUszMFsXf/pnHsRdGGJ+seMSNmbUtB/08pkfc7HTXjZm1KQf9PPo8tNLM2pyDfh7FUplNF61ibVdns0sxM1sUB/08iqUhX4PezNqag/48IoJiqey7SplZW3PQn8ezp0YZHp9il+8Ta2ZtzEF/HtMjbvxlKTNrZw768/CIGzNLAwf9eRRLZdZ3d7Khp9DsUszMFs1Bfx7F0hC7Nq1Bmu1KzWZm7cFBP4eIoK9UZpcvfWBmbc5BP4fjw+OcHJlw/7yZtT0H/Rw84sbM0sJBPwePuDGztHDQz+FwqUxPIc9la1c3uxQzswvioJ9Dn0fcmFlKOOjnUPSIGzNLCQf9LE6PTjBwesz982aWCg76WXjEjZmliYN+FsUBj7gxs/Rw0M+iOFim0JFj6yXdzS7FzOyCOehn0TcwxI6NPeRzHnFjZu3PQT+L4mCZ3b7ZiJmlhIO+zpnxKfpPnPHtA80sNRz0dQ4PlonANwQ3s9Rw0Ncp+ho3ZpYyDvo6xVKZfE5s39DT7FLMzJaEg75OX2mIbRu6KXT4R2Nm6eA0q1Mslf2NWDNLlYaCXtL1kp6QVJR06yzrPy3p0eTxpKSTyfKrJT0s6ZCkv5b07qU+gKU0PlnhqeMj7p83s1TpmK+BpDxwJ3Ad0A8ckLQvIh6bbhMRH6lp/0HgmmR2BPiHEdEn6XLgEUkPRMTJpTyIpXL0+DBTlWC3r1ppZinSyBn9tUAxIo5ExDhwL3DjedrvBe4BiIgnI6IvmX4GKAG9F1by8vFdpcwsjRoJ+s3AsZr5/mTZOSRtA64EHppl3bVAATi88DJXRrFURoKd/rKUmaVII0E/2wVfYo62NwH3R8TUizYgXQZ8EfjNiKicswPp/ZIOSjo4ODjYQEnLo69UZvO6LroK+abVYGa21BoJ+n5ga838FuCZOdreRNJtM03SxcCfA5+MiO/N9qKIuCsi9kTEnt7e5vXseMSNmaVRI0F/ANgt6UpJBaphvq++kaSrgPXAwzXLCsD/BL4QEf99aUpeHlOV4PBg2f3zZpY68wZ9REwCtwAPAI8D90XEIUl3SLqhpule4N6IqO3WeRfwJuDmmuGXVy9h/Uum/8QI45MVj7gxs9SZd3glQETsB/bXLbutbv72WV73JeBLF1DfiulL7iq102f0ZpYy/mZsojjooZVmlk4O+kTfQJlNF61ibVdns0sxM1tSDvpE9a5SPps3s/Rx0AMRweFS2XeVMrNUctADz50epTw2yS7fJ9bMUshBz8yIG5/Rm1kaOeiZuX2g++jNLI0c9FSvcbOuu5MNPYVml2JmtuQc9MDh5Bo30mzXbzMza28Oeqr3ifUXpcwsrTIf9MfLY5wYmWCXr3FjZimV+aD3XaXMLO0yH/RnR9w46M0spRz0pTI9hTyXrV3d7FLMzJaFg75UvdmIR9yYWVplPuj7SkO+Br2ZpVqmg/706AQDp8d8VykzS7VMB33RI27MLAMc9HjEjZmlW+aDvtCRY+sl3c0uxcxs2WQ+6Hds7CGf84gbM0uvTAe9r3FjZlmQ2aA/Mz5F/4kzHnFjZqmX2aA/PFgmwiNuzCz9Mh304LtKmVn6ZTbo+wbK5HNi+4aeZpdiZrasMhv0xVKZbRu6KXRk9kdgZhmR2ZTrKw2xq9fdNmaWfpkM+vHJCkePj7h/3swyIZNBf/T4MJOV8IgbM8uETAb9zDVuPIbezNKvoaCXdL2kJyQVJd06y/pPS3o0eTwp6WTNuvdJ6kse71vK4hdr+j6xO3o94sbM0q9jvgaS8sCdwHVAP3BA0r6IeGy6TUR8pKb9B4FrkulLgN8D9gABPJK89sSSHsUCFUtltqzvorsw7+GbmbW9Rs7orwWKEXEkIsaBe4Ebz9N+L3BPMv0rwIMR8UIS7g8C119IwUuhL7l9oJlZFjQS9JuBYzXz/cmyc0jaBlwJPLSQ10p6v6SDkg4ODg42UveiTVWCI4NlX4PezDKjkaCf7Rq+MUfbm4D7I2JqIa+NiLsiYk9E7Ont7W2gpMXrPzHC2GTFZ/RmlhmNBH0/sLVmfgvwzBxtb2Km22ahr10RfQPTtw/0iBszy4ZGgv4AsFvSlZIKVMN8X30jSVcB64GHaxY/APyypPWS1gO/nCxrmuKg7xNrZtky77CTiJiUdAvVgM4Dd0fEIUl3AAcjYjr09wL3RkTUvPYFSb9P9Y8FwB0R8cLSHsLC9A2U2XTRKtZ2dTazDDOzFdPQ+MKI2A/sr1t2W9387XO89m7g7kXWt+SKgx5xY2bZkqlvxkYEh0secWNm2ZKpoH/u9CjlsUmf0ZtZpmQq6D3ixsyyKFNBP30xM5/Rm1mWZCro+0pl1nV3snFNodmlmJmtmEwF/eFSmV29a5Bm+8KumVk6ZSro+0pDvquUmWVOZoL+eHmMEyMT7PR9Ys0sYzIT9NM3G9l9qUfcmFm2ZCboPeLGzLIqU0HfU8hz+drVzS7FzGxFZSrod27yiBszy57MBH1facjdNmaWSZkI+tOjEwycHnPQm1kmZSLopz+I3e1r3JhZBmUs6H1Gb2bZk5mgL3Tk2HpJd7NLMTNbcZkJ+h0be8jnPOLGzLInE0HvETdmlmWpD/oz41P0nzjjD2LNLLNSH/SHB8tE+NIHZpZdmQh6wJcnNrPMSn3Q9w2UyefE9g09zS7FzKwpUh/0xVKZbRu6KXSk/lDNzGaV+vTrKw2xyzcbMbMMS3XQj09WOHp8xP3zZpZpqQ76o8eHmayER9yYWaalOuh9MTMzs5QH/fR9Ynf0esSNmWVXqoO+WCqzZX0X3YWOZpdiZtY0DQW9pOslPSGpKOnWOdq8S9Jjkg5J+krN8n+bLHtc0h9pBe/l11cqu3/ezDJv3lNdSXngTuA6oB84IGlfRDxW02Y38LvA6yPihKRNyfLXAa8HXpE0/S7wZuA7S3kQs5mqBEcGy7xh14bl3pWZWUtr5Iz+WqAYEUciYhy4F7ixrs0/Ae6MiBMAEVFKlgewGigAq4BOYGApCp9P/4kRxiYrPqM3s8xrJOg3A8dq5vuTZbVeCrxU0l9K+p6k6wEi4mHg28CzyeOBiHi8fgeS3i/poKSDg4ODizmOc0yPuNnlETdmlnGNBP1sfepRN98B7AbeAuwFPitpnaRdwMuALVT/OLxV0pvO2VjEXRGxJyL29Pb2LqT+OfWdDXqf0ZtZtjUS9P3A1pr5LcAzs7T5ekRMRMRPgCeoBv/fB74XEeWIKAP/C3jNhZc9v2KpzKaLVrG2q3Mldmdm1rIaCfoDwG5JV0oqADcB++rafA34RQBJG6l25RwBfgq8WVKHpE6qH8Se03WzHDzixsysat6gj4hJ4BbgAaohfV9EHJJ0h6QbkmYPAMclPUa1T/7jEXEcuB84DPwY+BHwo4j4s2U4jvqaOVwqs9tBb2Y2//BKgIjYD+yvW3ZbzXQA/yJ51LaZAv7phZe5MM+dHqU8NukzejMzUvrNWI+4MTObkcqg7xvwiBszs2mpDPriYJl13Z1sXFNodilmZk2XzqAfKLOrdw0reFkdM7OWlc6gHyz7rlJmZonUBf3x8hgvDI+z0/eJNTMDUhj0Z+8qdalH3JiZQQqD3te4MTN7sdQFfbFUpqeQ5/K1q5tdiplZS0hl0O/c5BE3ZmbTUhn07rYxM5uRqqA/PTrBc6dHHfRmZjVSFfSHp0fc+Bo3ZmZnpSroPeLGzOxcqQr6w6UyhY4cW9d3NbsUM7OWkaqg7yuV2bGxh458qg7LzOyCpCoRPeLGzOxcqQn60Ykpjp0YcdCbmdVJTdCXxyZ5+ysu51Xb1je7FDOzltLQPWPbwcY1q/ijvdc0uwwzs5aTmjN6MzObnYPezCzlHPRmZinnoDczSzkHvZlZyjnozcxSzkFvZpZyDnozs5RTRDS7hheRNAgcvYBNbASeX6Jy2kXWjjlrxws+5qy4kGPeFhG9s61ouaC/UJIORsSeZtexkrJ2zFk7XvAxZ8VyHbO7bszMUs5Bb2aWcmkM+ruaXUATZO2Ys3a84GPOimU55tT10ZuZ2Yul8YzezMxqOOjNzFIuNUEv6XpJT0gqSrq12fUsN0lbJX1b0uOSDkn6ULNrWimS8pJ+KOkbza5lJUhaJ+l+SX+b/Hu/ttk1LTdJH0n+X/+NpHskrW52TUtN0t2SSpL+pmbZJZIelNSXPC/JLfNSEfSS8sCdwN8FXg7slfTy5la17CaBj0bEy4DXAL+dgWOe9iHg8WYXsYL+I/C/I+JngJ8n5ccuaTPwz4E9EfFzQB64qblVLYvPAdfXLbsV+FZE7Aa+lcxfsFQEPXAtUIyIIxExDtwL3NjkmpZVRDwbET9Ipoeo/vJvbm5Vy0/SFuBXgc82u5aVIOli4E3AfwGIiPGIONncqlZEB9AlqQPoBp5pcj1LLiL+L/BC3eIbgc8n058H3rEU+0pL0G8GjtXM95OB0JsmaTtwDfD95layIv4D8DtApdmFrJAdwCDwX5Puqs9K6ml2UcspIp4G/j3wU+BZ4FREfLO5Va2YSyPiWaiezAGblmKjaQl6zbIsE+NGJa0B/gfw4Yg43ex6lpOkXwNKEfFIs2tZQR3AK4E/iYhrgGGW6O18q0r6pW8ErgQuB3ok/Xpzq2pvaQn6fmBrzfwWUvhWr56kTqoh/+WI+Gqz61kBrwdukPQU1e65t0r6UnNLWnb9QH9ETL9bu59q8KfZ24CfRMRgREwAXwVe1+SaVsqApMsAkufSUmw0LUF/ANgt6UpJBaof3Oxrck3LSpKo9ts+HhF/2Ox6VkJE/G5EbImI7VT/jR+KiFSf6UXEc8AxSVcli34JeKyJJa2EnwKvkdSd/D//JVL+AXSNfcD7kun3AV9fio12LMVGmi0iJiXdAjxA9RP6uyPiUJPLWm6vB34D+LGkR5Nln4iI/U2syZbHB4EvJycxR4DfbHI9yyoivi/pfuAHVEeX/ZAUXg5B0j3AW4CNkvqB3wP+ALhP0j+m+gfvnUuyL18Cwcws3dLSdWNmZnNw0JuZpZyD3sws5Rz0ZmYp56A3M0s5B72ZWco56M3MUu7/Azk8nzy3PpqDAAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from itertools import repeat\n",
    "\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import hypergrad as hg\n",
    "import numpy as np\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.preprocessing import MinMaxScaler\n",
    "from sklearn.datasets import fetch_20newsgroups_vectorized\n",
    "import time\n",
    "import pdb\n",
    "\n",
    "import torch.nn.functional as F\n",
    "\n",
    "experiment_name = 'logistic_20news'\n",
    "algs = {'ZOJ': hg.hgvzoj, 'HOZOG': hg.hozog}\n",
    "\n",
    "# Helper functions to deal with cuda\n",
    "cuda = True and torch.cuda.is_available()\n",
    "\n",
    "default_tensor_str = 'torch.cuda.FloatTensor' if cuda else 'torch.FloatTensor'\n",
    "\n",
    "kwargs = {'num_workers': 4, 'pin_memory': True} if cuda else {}\n",
    "torch.set_default_tensor_type(default_tensor_str)\n",
    "\n",
    "\n",
    "# torch.multiprocessing.set_start_method('forkserver')\n",
    "\n",
    "def frnp(x): return torch.from_numpy(x).cuda().float() if cuda else torch.from_numpy(x).float()\n",
    "\n",
    "\n",
    "def tonp(x, cuda=cuda): return x.detach().cpu().numpy() if cuda else x.detach().numpy()\n",
    "\n",
    "\n",
    "seed = 0\n",
    "torch.manual_seed(seed)\n",
    "np.random.seed(seed)\n",
    "\n",
    "# load twentynews and preprocess\n",
    "val_size = 0.5\n",
    "X, y = fetch_20newsgroups_vectorized(subset='train', return_X_y=True,\n",
    "                                     # remove=('headers', 'footers', 'quotes')\n",
    "                                     )\n",
    "x_test, y_test = fetch_20newsgroups_vectorized(subset='test', return_X_y=True,\n",
    "                                               # remove=('headers', 'footers', 'quotes')\n",
    "                                               )\n",
    "\n",
    "x_train, x_val, y_train, y_val = train_test_split(X, y, stratify=y, test_size=val_size)\n",
    "\n",
    "train_samples, n_features = x_train.shape\n",
    "test_samples, n_features = x_test.shape\n",
    "val_samples, n_features = x_val.shape\n",
    "n_classes = np.unique(y_train).shape[0]\n",
    "\n",
    "print('Dataset 20newsgroup, train_samples=%i, val_samples=%i, test_samples=%i, n_features=%i, n_classes=%i'\n",
    "      % (train_samples, val_samples, test_samples, n_features, n_classes))\n",
    "\n",
    "ys = [frnp(y_train).long(), frnp(y_val).long(), frnp(y_test).long()] # long tensors used for ys\n",
    "xs = [x_train, x_val, x_test]\n",
    "\n",
    "\n",
    "def from_sparse(x):\n",
    "    x = x.tocoo() # tocoo() is a numpy function?\n",
    "    values = x.data\n",
    "    indices = np.vstack((x.row, x.col))\n",
    "\n",
    "    i = torch.LongTensor(indices)\n",
    "    v = torch.FloatTensor(values)\n",
    "    shape = x.shape\n",
    "\n",
    "    return torch.sparse.FloatTensor(i, v, torch.Size(shape)) # float tensors used for xs\n",
    "\n",
    "\n",
    "if cuda:\n",
    "    xs = [from_sparse(x).cuda() for x in xs]\n",
    "else:\n",
    "    xs = [from_sparse(x) for x in xs]\n",
    "\n",
    "x_train, x_val, x_test = xs\n",
    "y_train, y_val, y_test = ys\n",
    "\n",
    "\n",
    "class CustomTensorIterator:\n",
    "    def __init__(self, tensor_list, batch_size, **loader_kwargs):\n",
    "        self.loader = DataLoader(TensorDataset(*tensor_list), batch_size=batch_size, **loader_kwargs)\n",
    "        self.iterator = iter(self.loader)\n",
    "\n",
    "    def __next__(self, *args):\n",
    "        try:\n",
    "            idx = next(self.iterator)\n",
    "        except StopIteration:\n",
    "            self.iterator = iter(self.loader)\n",
    "            idx = next(self.iterator)\n",
    "        return idx\n",
    "\n",
    "\n",
    "# torch.DataLoader has problems with sparse tensor on GPU\n",
    "train_batch_size = len(y_train)\n",
    "val_batch_size = len(y_val)\n",
    "\n",
    "iterators = []\n",
    "for bs, x, y in [(train_batch_size, x_train, y_train), (val_batch_size, x_val, y_val)]:\n",
    "    if bs < len(y):\n",
    "        print('making iterator with batch size ', bs)\n",
    "        iterators.append(CustomTensorIterator([x, y], batch_size=bs, shuffle=True, **kwargs))\n",
    "    else: # only else statement will be executed because of the way they defined train_batch_size and val_batch_size\n",
    "        iterators.append(repeat([x, y]))\n",
    "\n",
    "train_iterator, val_iterator = iterators\n",
    "\n",
    "# HPO set up\n",
    "prefix = 'ZOJ'\n",
    "reg_type = 'ONE' # or EACH\n",
    "mu = 0.01\n",
    "n_steps = 500\n",
    "outer_lr, outer_mu = 0.02, 0.0  # nice with 100.0, 0.0 (torch.SGD) tested with T, K = 5, 10 and CG\n",
    "inner_lr, inner_mu = 100.0, 0.9  # nice with 100., 0.9 (HeavyBall) tested with T, K = 5, 10 and CG\n",
    "T, K = 20, 20\n",
    "tol = 1e-12\n",
    "warm_start = True\n",
    "bias = False  # without bias outer_lr can be bigger (much faster convergence)\n",
    "\n",
    "hyperAlgo = algs[prefix]\n",
    "\n",
    "train_log_interval = 100\n",
    "val_log_interval = 50\n",
    "print(experiment_name + ' with ' + prefix + ', mu = ' + str(mu) + ', beta = ' + str(outer_lr) + ' and T = ' + str(T))\n",
    "\n",
    "if reg_type=='EACH':\n",
    "    l2_reg_params = torch.zeros(n_features).requires_grad_(True)  # one hp per feature\n",
    "    l1_reg_params = (-1. * torch.ones(n_features)).requires_grad_(True)\n",
    "else:\n",
    "    l2_reg_params = (-1.*torch.ones(1)).requires_grad_(True)  # one l2 hp only (best when really low)\n",
    "    l1_reg_params = (0. * torch.ones(1)).requires_grad_(True)  # one l1 hp only (best when really low)\n",
    "\n",
    "\n",
    "hparams = [l2_reg_params]\n",
    "\n",
    "ones_dxc = torch.ones(n_features, n_classes)\n",
    "\n",
    "\n",
    "def reg_f(params, l2_reg_params, l1_reg_params=None):\n",
    "    r = 0.5 * ((params[0] ** 2) * torch.exp(l2_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    if l1_reg_params is not None:\n",
    "        r += (params[0].abs() * torch.exp(l1_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    return r\n",
    "\n",
    "\n",
    "outer_opt = torch.optim.SGD(lr=outer_lr, momentum=outer_mu, params=hparams)\n",
    "# outer_opt = torch.optim.Adam(lr=0.01, params=hparams)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "val_losses, val_accs = [], []\n",
    "test_losses, test_accs = [], []\n",
    "\n",
    "w = torch.zeros(n_features, n_classes).requires_grad_(True)\n",
    "parameters = [w]\n",
    "\n",
    "if bias:\n",
    "    b = torch.zeros(n_classes).requires_grad_(True)\n",
    "    parameters.append(b)\n",
    "\n",
    "initial_parameters = []\n",
    "for param in parameters:\n",
    "    initial_parameters.append(param.clone())\n",
    "\n",
    "def out_f(x, params):\n",
    "    out = x @ params[0]\n",
    "    out += params[1] if len(params) == 2 else 0\n",
    "    return out\n",
    "\n",
    "\n",
    "def train_loss(params, hparams, data):\n",
    "    x_mb, y_mb = data\n",
    "    out = out_f(x_mb, params)\n",
    "    return F.cross_entropy(out, y_mb) + reg_f(params, *hparams)\n",
    "\n",
    "\n",
    "def val_loss(params, hparams, more=False, verbose=False):\n",
    "    x_mb, y_mb = next(val_iterator)\n",
    "    out = out_f(x_mb, params)\n",
    "    val_loss = F.cross_entropy(out, y_mb)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y_mb.view_as(pred)).sum().item() / len(y_mb)\n",
    "\n",
    "    #val_losses.append(tonp(val_loss)) # get rid of it\n",
    "    #val_accs.append(acc) # get rid of it\n",
    "\n",
    "    if more:\n",
    "        return val_loss, acc\n",
    "    else:\n",
    "        return val_loss\n",
    "\n",
    "\n",
    "def eval(params, x, y):\n",
    "    out = out_f(x, params)\n",
    "    loss = F.cross_entropy(out, y)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y.view_as(pred)).sum().item() / len(y)\n",
    "\n",
    "    return loss, acc\n",
    "\n",
    "inner_opt = torch.optim.SGD(lr=inner_lr, momentum=inner_mu, params=parameters)\n",
    "\n",
    "def update_tensor_grads(params, grads):\n",
    "    for l, g in zip(params, grads):\n",
    "        if l.grad is None:\n",
    "            l.grad = torch.zeros_like(l)\n",
    "        if g is not None:\n",
    "            l.grad += g\n",
    "\n",
    "def inner_solver(hparams, steps, optim, params0=None):\n",
    "\n",
    "    if params0 is not None:\n",
    "        for param, param0 in zip(parameters, params0):\n",
    "            param.data = param0.data\n",
    "\n",
    "    for t in range(steps):\n",
    "        data = next(train_iterator)\n",
    "        loss = train_loss(parameters, hparams, data)\n",
    "        optim.zero_grad()\n",
    "        grads = torch.autograd.grad(loss, parameters)\n",
    "        update_tensor_grads(parameters, grads)\n",
    "        optim.step()\n",
    "        #print('inner grad norm:', torch.norm(grads[0]).item())\n",
    "        #print('current train loss:', loss.item())\n",
    "\n",
    "    return [param.clone() for param in parameters]\n",
    "\n",
    "\n",
    "# final_params = inner_solver(hparams, 1000, initial_parameters)\n",
    "# pdb.set_trace()\n",
    "\n",
    "params_history = []\n",
    "total_time = 0\n",
    "running_time = []\n",
    "for o_step in range(n_steps):\n",
    "    start_time = time.time()\n",
    "\n",
    "    final_params = inner_solver(hparams, T, optim=inner_opt, params0=initial_parameters)\n",
    "\n",
    "    # pdb.set_trace()\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    _, vloss, vacc = hyperAlgo(final_params, hparams, val_loss, inner_solver, params0=initial_parameters, optim=inner_opt, mu=mu, T=T, p=1, set_grad=True, more=True)\n",
    "    outer_opt.step()\n",
    "    val_losses.append(tonp(vloss))\n",
    "    val_accs.append(vacc)\n",
    "\n",
    "    # for hparam in hparams:\n",
    "    #     hparam.data.clamp_(min=-5.0, max=5.0)\n",
    "    #pdb.set_trace()\n",
    "\n",
    "    for init_p, new_p in zip(initial_parameters, final_params):\n",
    "        if warm_start:\n",
    "            init_p.data = new_p\n",
    "        else:\n",
    "            init_p.data = torch.zeros_like(init_p)\n",
    "\n",
    "    iter_time = time.time() - start_time\n",
    "    total_time += iter_time\n",
    "    running_time.append(total_time)\n",
    "\n",
    "    if o_step % val_log_interval == 0 or o_step == n_steps - 1:\n",
    "        test_loss, test_acc = eval(final_params, x_test, y_test)\n",
    "        test_losses.append(test_loss)\n",
    "        test_accs.append(test_acc)\n",
    "        #print('step train los is: ', step_train_loss)\n",
    "        print('o_step={} ({:.2e}s) Val loss: {:.4e}, Val Acc: {:.2f}%'.format(o_step, iter_time, val_losses[-1],\n",
    "                                                                              100 * val_accs[-1]))\n",
    "        print('          Test loss: {:.4e}, Test Acc: {:.2f}%'.format(test_loss, 100 * test_acc))\n",
    "        print('          l2_hp norm: {:.4e}'.format(torch.norm(hparams[0])))\n",
    "        print('          l2_hp: {:.4e}'.format(hparams[0].item()))\n",
    "        if len(hparams) == 2:\n",
    "            print('          l1_hp : ', torch.norm(hparams[1]))\n",
    "\n",
    "print('HPO ended in {:.2e} seconds\\n'.format(total_time))\n",
    "\n",
    "plt.title('val_accuracy')\n",
    "plt.plot(running_time, val_accs)\n",
    "#figname = 'logistic_plots/VAL_ACC_' + experiment_name + '_' + prefix + '_' + reg_type + '_' + str(T) + '_' + str(mu) + '_' + str(outer_lr) + '.png'\n",
    "#plt.savefig(figname, bbox_inches='tight')\n",
    "#plt.close()\n",
    "plt.show()\n",
    "\n",
    "\n",
    "plt.title('test_accuracy')\n",
    "plt.plot(test_accs)\n",
    "# figname = 'logistic_plots/TEST_ACC_' + experiment_name + '_' + prefix + '_' + reg_type + '_' + str(T) + '_' + str(mu) + '_' + str(outer_lr) + '.png'\n",
    "# plt.savefig(figname, bbox_inches='tight')\n",
    "# plt.close()\n",
    "plt.show()\n",
    "\n",
    "val_zoj = val_accs\n",
    "run_zoj = running_time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dataset 20newsgroup, train_samples=5657, val_samples=5657, test_samples=7532, n_features=130107, n_classes=20\n",
      "logistic_20news with HOZOG, mu = 0.01, beta = 0.08 and T = 20\n",
      "o_step=0 (2.87e-01s) Val loss: 8.9652e-01, Val Acc: 76.21%\n",
      "          Test loss: 1.1634e+00, Test Acc: 66.81%\n",
      "          l2_hp norm: 3.0037e+00\n",
      "          l2_hp: -3.0037e+00\n",
      "o_step=50 (2.79e-01s) Val loss: 5.1947e-01, Val Acc: 85.86%\n",
      "          Test loss: 8.5389e-01, Test Acc: 76.31%\n",
      "          l2_hp norm: 2.0484e+00\n",
      "          l2_hp: -2.0484e+00\n",
      "o_step=100 (2.78e-01s) Val loss: 5.1922e-01, Val Acc: 86.23%\n",
      "          Test loss: 8.6140e-01, Test Acc: 76.57%\n",
      "          l2_hp norm: 2.0492e+00\n",
      "          l2_hp: -2.0492e+00\n",
      "o_step=150 (2.79e-01s) Val loss: 5.1934e-01, Val Acc: 86.34%\n",
      "          Test loss: 8.6464e-01, Test Acc: 76.70%\n",
      "          l2_hp norm: 2.0488e+00\n",
      "          l2_hp: -2.0488e+00\n",
      "o_step=200 (2.79e-01s) Val loss: 5.1906e-01, Val Acc: 86.44%\n",
      "          Test loss: 8.6568e-01, Test Acc: 76.77%\n",
      "          l2_hp norm: 2.0488e+00\n",
      "          l2_hp: -2.0488e+00\n",
      "o_step=250 (2.79e-01s) Val loss: 5.1850e-01, Val Acc: 86.49%\n",
      "          Test loss: 8.6556e-01, Test Acc: 76.77%\n",
      "          l2_hp norm: 2.0486e+00\n",
      "          l2_hp: -2.0486e+00\n",
      "o_step=300 (2.79e-01s) Val loss: 5.1778e-01, Val Acc: 86.53%\n",
      "          Test loss: 8.6484e-01, Test Acc: 76.74%\n",
      "          l2_hp norm: 2.0488e+00\n",
      "          l2_hp: -2.0488e+00\n",
      "o_step=350 (2.79e-01s) Val loss: 5.1698e-01, Val Acc: 86.57%\n",
      "          Test loss: 8.6379e-01, Test Acc: 76.78%\n",
      "          l2_hp norm: 2.0478e+00\n",
      "          l2_hp: -2.0478e+00\n",
      "o_step=400 (2.79e-01s) Val loss: 5.1617e-01, Val Acc: 86.57%\n",
      "          Test loss: 8.6261e-01, Test Acc: 76.83%\n",
      "          l2_hp norm: 2.0470e+00\n",
      "          l2_hp: -2.0470e+00\n",
      "o_step=450 (2.79e-01s) Val loss: 5.1537e-01, Val Acc: 86.57%\n",
      "          Test loss: 8.6137e-01, Test Acc: 76.82%\n",
      "          l2_hp norm: 2.0454e+00\n",
      "          l2_hp: -2.0454e+00\n",
      "o_step=499 (2.79e-01s) Val loss: 5.1462e-01, Val Acc: 86.67%\n",
      "          Test loss: 8.6017e-01, Test Acc: 76.85%\n",
      "          l2_hp norm: 2.0446e+00\n",
      "          l2_hp: -2.0446e+00\n",
      "HPO ended in 1.39e+02 seconds\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from itertools import repeat\n",
    "\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import hypergrad as hg\n",
    "import numpy as np\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.preprocessing import MinMaxScaler\n",
    "from sklearn.datasets import fetch_20newsgroups_vectorized\n",
    "import time\n",
    "import pdb\n",
    "\n",
    "import torch.nn.functional as F\n",
    "\n",
    "experiment_name = 'logistic_20news'\n",
    "algs = {'ZOJ': hg.hgvzoj, 'HOZOG': hg.hozog}\n",
    "\n",
    "# Helper functions to deal with cuda\n",
    "cuda = True and torch.cuda.is_available()\n",
    "\n",
    "default_tensor_str = 'torch.cuda.FloatTensor' if cuda else 'torch.FloatTensor'\n",
    "\n",
    "kwargs = {'num_workers': 4, 'pin_memory': True} if cuda else {}\n",
    "torch.set_default_tensor_type(default_tensor_str)\n",
    "\n",
    "\n",
    "# torch.multiprocessing.set_start_method('forkserver')\n",
    "\n",
    "def frnp(x): return torch.from_numpy(x).cuda().float() if cuda else torch.from_numpy(x).float()\n",
    "\n",
    "\n",
    "def tonp(x, cuda=cuda): return x.detach().cpu().numpy() if cuda else x.detach().numpy()\n",
    "\n",
    "\n",
    "seed = 0\n",
    "torch.manual_seed(seed)\n",
    "np.random.seed(seed)\n",
    "\n",
    "# load twentynews and preprocess\n",
    "val_size = 0.5\n",
    "X, y = fetch_20newsgroups_vectorized(subset='train', return_X_y=True,\n",
    "                                     # remove=('headers', 'footers', 'quotes')\n",
    "                                     )\n",
    "x_test, y_test = fetch_20newsgroups_vectorized(subset='test', return_X_y=True,\n",
    "                                               # remove=('headers', 'footers', 'quotes')\n",
    "                                               )\n",
    "\n",
    "x_train, x_val, y_train, y_val = train_test_split(X, y, stratify=y, test_size=val_size)\n",
    "\n",
    "train_samples, n_features = x_train.shape\n",
    "test_samples, n_features = x_test.shape\n",
    "val_samples, n_features = x_val.shape\n",
    "n_classes = np.unique(y_train).shape[0]\n",
    "\n",
    "print('Dataset 20newsgroup, train_samples=%i, val_samples=%i, test_samples=%i, n_features=%i, n_classes=%i'\n",
    "      % (train_samples, val_samples, test_samples, n_features, n_classes))\n",
    "\n",
    "ys = [frnp(y_train).long(), frnp(y_val).long(), frnp(y_test).long()] # long tensors used for ys\n",
    "xs = [x_train, x_val, x_test]\n",
    "\n",
    "\n",
    "def from_sparse(x):\n",
    "    x = x.tocoo() # tocoo() is a numpy function?\n",
    "    values = x.data\n",
    "    indices = np.vstack((x.row, x.col))\n",
    "\n",
    "    i = torch.LongTensor(indices)\n",
    "    v = torch.FloatTensor(values)\n",
    "    shape = x.shape\n",
    "\n",
    "    return torch.sparse.FloatTensor(i, v, torch.Size(shape)) # float tensors used for xs\n",
    "\n",
    "\n",
    "if cuda:\n",
    "    xs = [from_sparse(x).cuda() for x in xs]\n",
    "else:\n",
    "    xs = [from_sparse(x) for x in xs]\n",
    "\n",
    "x_train, x_val, x_test = xs\n",
    "y_train, y_val, y_test = ys\n",
    "\n",
    "\n",
    "class CustomTensorIterator:\n",
    "    def __init__(self, tensor_list, batch_size, **loader_kwargs):\n",
    "        self.loader = DataLoader(TensorDataset(*tensor_list), batch_size=batch_size, **loader_kwargs)\n",
    "        self.iterator = iter(self.loader)\n",
    "\n",
    "    def __next__(self, *args):\n",
    "        try:\n",
    "            idx = next(self.iterator)\n",
    "        except StopIteration:\n",
    "            self.iterator = iter(self.loader)\n",
    "            idx = next(self.iterator)\n",
    "        return idx\n",
    "\n",
    "\n",
    "# torch.DataLoader has problems with sparse tensor on GPU\n",
    "train_batch_size = len(y_train)\n",
    "val_batch_size = len(y_val)\n",
    "\n",
    "iterators = []\n",
    "for bs, x, y in [(train_batch_size, x_train, y_train), (val_batch_size, x_val, y_val)]:\n",
    "    if bs < len(y):\n",
    "        print('making iterator with batch size ', bs)\n",
    "        iterators.append(CustomTensorIterator([x, y], batch_size=bs, shuffle=True, **kwargs))\n",
    "    else: # only else statement will be executed because of the way they defined train_batch_size and val_batch_size\n",
    "        iterators.append(repeat([x, y]))\n",
    "\n",
    "train_iterator, val_iterator = iterators\n",
    "\n",
    "# HPO set up\n",
    "prefix = 'HOZOG'\n",
    "reg_type = 'ONE' # or EACH\n",
    "mu = 0.01\n",
    "n_steps = 500\n",
    "outer_lr, outer_mu = 0.08, 0.0  # nice with 100.0, 0.0 (torch.SGD) tested with T, K = 5, 10 and CG\n",
    "inner_lr, inner_mu = 100.0, 0.9  # nice with 100., 0.9 (HeavyBall) tested with T, K = 5, 10 and CG\n",
    "T, K = 20, 20\n",
    "tol = 1e-12\n",
    "warm_start = True\n",
    "bias = False  # without bias outer_lr can be bigger (much faster convergence)\n",
    "\n",
    "hyperAlgo = algs[prefix]\n",
    "\n",
    "train_log_interval = 100\n",
    "val_log_interval = 50\n",
    "print(experiment_name + ' with ' + prefix + ', mu = ' + str(mu) + ', beta = ' + str(outer_lr) + ' and T = ' + str(T))\n",
    "\n",
    "if reg_type=='EACH':\n",
    "    l2_reg_params = torch.zeros(n_features).requires_grad_(True)  # one hp per feature\n",
    "    l1_reg_params = (-1. * torch.ones(n_features)).requires_grad_(True)\n",
    "else:\n",
    "    l2_reg_params = (-1.*torch.ones(1)).requires_grad_(True)  # one l2 hp only (best when really low)\n",
    "    l1_reg_params = (0. * torch.ones(1)).requires_grad_(True)  # one l1 hp only (best when really low)\n",
    "\n",
    "\n",
    "hparams = [l2_reg_params]\n",
    "\n",
    "ones_dxc = torch.ones(n_features, n_classes)\n",
    "\n",
    "\n",
    "def reg_f(params, l2_reg_params, l1_reg_params=None):\n",
    "    r = 0.5 * ((params[0] ** 2) * torch.exp(l2_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    if l1_reg_params is not None:\n",
    "        r += (params[0].abs() * torch.exp(l1_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    return r\n",
    "\n",
    "\n",
    "outer_opt = torch.optim.SGD(lr=outer_lr, momentum=outer_mu, params=hparams)\n",
    "# outer_opt = torch.optim.Adam(lr=0.01, params=hparams)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "val_losses, val_accs = [], []\n",
    "test_losses, test_accs = [], []\n",
    "\n",
    "w = torch.zeros(n_features, n_classes).requires_grad_(True)\n",
    "parameters = [w]\n",
    "\n",
    "if bias:\n",
    "    b = torch.zeros(n_classes).requires_grad_(True)\n",
    "    parameters.append(b)\n",
    "\n",
    "initial_parameters = []\n",
    "for param in parameters:\n",
    "    initial_parameters.append(param.clone())\n",
    "\n",
    "def out_f(x, params):\n",
    "    out = x @ params[0]\n",
    "    out += params[1] if len(params) == 2 else 0\n",
    "    return out\n",
    "\n",
    "\n",
    "def train_loss(params, hparams, data):\n",
    "    x_mb, y_mb = data\n",
    "    out = out_f(x_mb, params)\n",
    "    return F.cross_entropy(out, y_mb) + reg_f(params, *hparams)\n",
    "\n",
    "\n",
    "def val_loss(params, hparams, more=False, verbose=False):\n",
    "    x_mb, y_mb = next(val_iterator)\n",
    "    out = out_f(x_mb, params)\n",
    "    val_loss = F.cross_entropy(out, y_mb)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y_mb.view_as(pred)).sum().item() / len(y_mb)\n",
    "\n",
    "    #val_losses.append(tonp(val_loss)) # get rid of it\n",
    "    #val_accs.append(acc) # get rid of it\n",
    "\n",
    "    if more:\n",
    "        return val_loss, acc\n",
    "    else:\n",
    "        return val_loss\n",
    "\n",
    "\n",
    "def eval(params, x, y):\n",
    "    out = out_f(x, params)\n",
    "    loss = F.cross_entropy(out, y)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y.view_as(pred)).sum().item() / len(y)\n",
    "\n",
    "    return loss, acc\n",
    "\n",
    "inner_opt = torch.optim.SGD(lr=inner_lr, momentum=inner_mu, params=parameters)\n",
    "\n",
    "def update_tensor_grads(params, grads):\n",
    "    for l, g in zip(params, grads):\n",
    "        if l.grad is None:\n",
    "            l.grad = torch.zeros_like(l)\n",
    "        if g is not None:\n",
    "            l.grad += g\n",
    "\n",
    "def inner_solver(hparams, steps, optim, params0=None):\n",
    "\n",
    "    if params0 is not None:\n",
    "        for param, param0 in zip(parameters, params0):\n",
    "            param.data = param0.data\n",
    "\n",
    "    for t in range(steps):\n",
    "        data = next(train_iterator)\n",
    "        loss = train_loss(parameters, hparams, data)\n",
    "        optim.zero_grad()\n",
    "        grads = torch.autograd.grad(loss, parameters)\n",
    "        update_tensor_grads(parameters, grads)\n",
    "        optim.step()\n",
    "        #print('inner grad norm:', torch.norm(grads[0]).item())\n",
    "        #print('current train loss:', loss.item())\n",
    "\n",
    "    return [param.clone() for param in parameters]\n",
    "\n",
    "\n",
    "# final_params = inner_solver(hparams, 1000, initial_parameters)\n",
    "# pdb.set_trace()\n",
    "\n",
    "params_history = []\n",
    "total_time = 0\n",
    "running_time = []\n",
    "for o_step in range(n_steps):\n",
    "    start_time = time.time()\n",
    "\n",
    "    final_params = inner_solver(hparams, T, optim=inner_opt, params0=initial_parameters)\n",
    "\n",
    "    # pdb.set_trace()\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    _, vloss, vacc = hyperAlgo(final_params, hparams, val_loss, inner_solver, params0=initial_parameters, optim=inner_opt, mu=mu, T=T, p=1, set_grad=True, more=True)\n",
    "    outer_opt.step()\n",
    "    val_losses.append(tonp(vloss))\n",
    "    val_accs.append(vacc)\n",
    "\n",
    "    # for hparam in hparams:\n",
    "    #     hparam.data.clamp_(min=-5.0, max=5.0)\n",
    "    #pdb.set_trace()\n",
    "\n",
    "    for init_p, new_p in zip(initial_parameters, final_params):\n",
    "        if warm_start:\n",
    "            init_p.data = new_p\n",
    "        else:\n",
    "            init_p.data = torch.zeros_like(init_p)\n",
    "\n",
    "    iter_time = time.time() - start_time\n",
    "    total_time += iter_time\n",
    "    running_time.append(total_time)\n",
    "\n",
    "    if o_step % val_log_interval == 0 or o_step == n_steps - 1:\n",
    "        test_loss, test_acc = eval(final_params, x_test, y_test)\n",
    "        test_losses.append(test_loss)\n",
    "        test_accs.append(test_acc)\n",
    "        #print('step train los is: ', step_train_loss)\n",
    "        print('o_step={} ({:.2e}s) Val loss: {:.4e}, Val Acc: {:.2f}%'.format(o_step, iter_time, val_losses[-1],\n",
    "                                                                              100 * val_accs[-1]))\n",
    "        print('          Test loss: {:.4e}, Test Acc: {:.2f}%'.format(test_loss, 100 * test_acc))\n",
    "        print('          l2_hp norm: {:.4e}'.format(torch.norm(hparams[0])))\n",
    "        print('          l2_hp: {:.4e}'.format(hparams[0].item()))\n",
    "        if len(hparams) == 2:\n",
    "            print('          l1_hp : ', torch.norm(hparams[1]))\n",
    "\n",
    "print('HPO ended in {:.2e} seconds\\n'.format(total_time))\n",
    "\n",
    "plt.title('val_accuracy')\n",
    "plt.plot(running_time, val_accs)\n",
    "#figname = 'logistic_plots/VAL_ACC_' + experiment_name + '_' + prefix + '_' + reg_type + '_' + str(T) + '_' + str(mu) + '_' + str(outer_lr) + '.png'\n",
    "#plt.savefig(figname, bbox_inches='tight')\n",
    "#plt.close()\n",
    "plt.show()\n",
    "\n",
    "\n",
    "plt.title('test_accuracy')\n",
    "plt.plot(test_accs)\n",
    "# figname = 'logistic_plots/TEST_ACC_' + experiment_name + '_' + prefix + '_' + reg_type + '_' + str(T) + '_' + str(mu) + '_' + str(outer_lr) + '.png'\n",
    "# plt.savefig(figname, bbox_inches='tight')\n",
    "# plt.close()\n",
    "plt.show()\n",
    "\n",
    "val_hozog = val_accs\n",
    "run_hozog = running_time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dataset 20newsgroup, train_samples=5657, val_samples=5657, test_samples=7532, n_features=130107, n_classes=20\n",
      "o_step=0 (3.11e-01s) Val loss: 8.9652e-01, Val Acc: 76.21%\n",
      "          Test loss: 1.1634e+00, Test Acc: 66.81%\n",
      "          l2_hp norm: 1.0544e+00\n",
      "o_step=50 (3.26e-01s) Val loss: 5.2041e-01, Val Acc: 85.82%\n",
      "          Test loss: 8.5389e-01, Test Acc: 76.35%\n",
      "          l2_hp norm: 1.2992e+00\n",
      "o_step=100 (3.82e-01s) Val loss: 5.1676e-01, Val Acc: 86.14%\n",
      "          Test loss: 8.5187e-01, Test Acc: 76.55%\n",
      "          l2_hp norm: 9.8552e-01\n",
      "o_step=150 (4.30e-01s) Val loss: 5.1314e-01, Val Acc: 86.35%\n",
      "          Test loss: 8.4321e-01, Test Acc: 76.62%\n",
      "          l2_hp norm: 6.5394e-01\n",
      "o_step=200 (4.99e-01s) Val loss: 5.1085e-01, Val Acc: 86.51%\n",
      "          Test loss: 8.3650e-01, Test Acc: 76.62%\n",
      "          l2_hp norm: 6.2455e-01\n",
      "o_step=250 (5.56e-01s) Val loss: 5.0903e-01, Val Acc: 86.51%\n",
      "          Test loss: 8.3406e-01, Test Acc: 76.57%\n",
      "          l2_hp norm: 7.5160e-01\n",
      "o_step=300 (5.94e-01s) Val loss: 5.0754e-01, Val Acc: 86.51%\n",
      "          Test loss: 8.3307e-01, Test Acc: 76.61%\n",
      "          l2_hp norm: 8.3820e-01\n",
      "o_step=350 (6.64e-01s) Val loss: 5.0647e-01, Val Acc: 86.53%\n",
      "          Test loss: 8.3233e-01, Test Acc: 76.71%\n",
      "          l2_hp norm: 8.7737e-01\n",
      "o_step=400 (8.30e-01s) Val loss: 5.0566e-01, Val Acc: 86.57%\n",
      "          Test loss: 8.3156e-01, Test Acc: 76.67%\n",
      "          l2_hp norm: 8.9535e-01\n",
      "o_step=450 (8.36e-01s) Val loss: 5.0502e-01, Val Acc: 86.60%\n",
      "          Test loss: 8.3084e-01, Test Acc: 76.74%\n",
      "          l2_hp norm: 9.0796e-01\n",
      "o_step=499 (9.09e-01s) Val loss: 5.0450e-01, Val Acc: 86.65%\n",
      "          Test loss: 8.3022e-01, Test Acc: 76.74%\n",
      "          l2_hp norm: 9.2005e-01\n",
      "HPO ended in 2.83e+02 seconds\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from itertools import repeat\n",
    "\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import hypergrad as hg\n",
    "import numpy as np\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.preprocessing import MinMaxScaler\n",
    "from sklearn.datasets import fetch_20newsgroups_vectorized\n",
    "import time\n",
    "\n",
    "import torch.nn.functional as F\n",
    "\n",
    "\n",
    "# Helper functions to deal with cuda\n",
    "cuda = True and torch.cuda.is_available()\n",
    "\n",
    "default_tensor_str = 'torch.cuda.FloatTensor' if cuda else 'torch.FloatTensor'\n",
    "\n",
    "kwargs = {'num_workers': 1, 'pin_memory': True} if cuda else {}\n",
    "torch.set_default_tensor_type(default_tensor_str)\n",
    "#torch.multiprocessing.set_start_method('forkserver')\n",
    "\n",
    "def frnp(x): return torch.from_numpy(x).cuda().float() if cuda else torch.from_numpy(x).float()\n",
    "def tonp(x, cuda=cuda): return x.detach().cpu().numpy() if cuda else x.detach().numpy()\n",
    "\n",
    "\n",
    "seed = 0\n",
    "torch.manual_seed(seed)\n",
    "np.random.seed(seed)\n",
    "\n",
    "\n",
    "# load twentynews and preprocess\n",
    "val_size = 0.5\n",
    "X, y = fetch_20newsgroups_vectorized(subset='train', return_X_y=True,\n",
    "                                     #remove=('headers', 'footers', 'quotes')\n",
    "                                     )\n",
    "x_test, y_test = fetch_20newsgroups_vectorized(subset='test', return_X_y=True,\n",
    "                                               #remove=('headers', 'footers', 'quotes')\n",
    "                                               )\n",
    "\n",
    "\n",
    "x_train, x_val, y_train, y_val = train_test_split(X, y, stratify=y, test_size=val_size)\n",
    "\n",
    "\n",
    "train_samples, n_features = x_train.shape\n",
    "test_samples, n_features = x_test.shape\n",
    "val_samples, n_features = x_val.shape\n",
    "n_classes = np.unique(y_train).shape[0]\n",
    "\n",
    "print('Dataset 20newsgroup, train_samples=%i, val_samples=%i, test_samples=%i, n_features=%i, n_classes=%i'\n",
    "      % (train_samples, val_samples, test_samples, n_features, n_classes))\n",
    "\n",
    "\n",
    "ys = [frnp(y_train).long(), frnp(y_val).long(), frnp(y_test).long()]\n",
    "xs = [x_train, x_val, x_test]\n",
    "\n",
    "\n",
    "def from_sparse(x):\n",
    "    x = x.tocoo()\n",
    "    values = x.data\n",
    "    indices = np.vstack((x.row, x.col))\n",
    "\n",
    "    i = torch.LongTensor(indices)\n",
    "    v = torch.FloatTensor(values)\n",
    "    shape = x.shape\n",
    "\n",
    "    return torch.sparse.FloatTensor(i, v, torch.Size(shape))\n",
    "\n",
    "\n",
    "if cuda:\n",
    "    xs = [from_sparse(x).cuda() for x in xs]\n",
    "else:\n",
    "    xs = [from_sparse(x) for x in xs]\n",
    "\n",
    "x_train, x_val, x_test = xs\n",
    "y_train, y_val, y_test = ys\n",
    "\n",
    "\n",
    "class CustomTensorIterator:\n",
    "    def __init__(self, tensor_list, batch_size, **loader_kwargs):\n",
    "        self.loader = DataLoader(TensorDataset(*tensor_list), batch_size=batch_size, **loader_kwargs)\n",
    "        self.iterator = iter(self.loader)\n",
    "\n",
    "    def __next__(self, *args):\n",
    "        try:\n",
    "            idx = next(self.iterator)\n",
    "        except StopIteration:\n",
    "            self.iterator = iter(self.loader)\n",
    "            idx = next(self.iterator)\n",
    "        return idx\n",
    "\n",
    "\n",
    "# torch.DataLoader has problems with sparse tensor on GPU\n",
    "train_batch_size = len(y_train)\n",
    "val_batch_size = len(y_val)\n",
    "\n",
    "iterators = []\n",
    "for bs, x, y in [(train_batch_size, x_train, y_train), (val_batch_size, x_val, y_val)]:\n",
    "    if bs < len(y):\n",
    "        print('making iterator with batch size ', bs)\n",
    "        iterators.append(CustomTensorIterator([x, y], batch_size=bs, shuffle=True, **kwargs))\n",
    "    else:\n",
    "        iterators.append(repeat([x, y]))\n",
    "\n",
    "train_iterator, val_iterator = iterators\n",
    "\n",
    "# HPO set up\n",
    "n_steps = 500\n",
    "outer_lr, outer_mu = 100.0, 0.0  # nice with 100.0, 0.0 (torch.SGD) tested with T, K = 5, 10 and CG\n",
    "inner_lr, inner_mu = 100., 0.9   # nice with 100., 0.9 (HeavyBall) tested with T, K = 5, 10 and CG\n",
    "T, K = 20, 20\n",
    "tol = 1e-12\n",
    "warm_start = True\n",
    "bias = False  # without bias outer_lr can be bigger (much faster convergence)\n",
    "\n",
    "train_log_interval = 100\n",
    "val_log_interval = 50\n",
    "\n",
    "#l2_reg_params = torch.zeros(n_features).requires_grad_(True)  # one hp per feature\n",
    "l2_reg_params = (-1.*torch.ones(1)).requires_grad_(True)  # one l2 hp only (best when really low)\n",
    "l1_reg_params = (0.*torch.ones(1)).requires_grad_(True)  # one l1 hp only (best when really low)\n",
    "#l1_reg_params = (-1.*torch.ones(n_features)).requires_grad_(True)\n",
    "\n",
    "hparams = [l2_reg_params]\n",
    "\n",
    "ones_dxc = torch.ones(n_features, n_classes)\n",
    "\n",
    "\n",
    "def reg_f(params, l2_reg_params, l1_reg_params=None):\n",
    "    r = 0.5 * ((params[0] ** 2) * torch.exp(l2_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    if l1_reg_params is not None:\n",
    "        r += (params[0].abs() * torch.exp(l1_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    return r\n",
    "\n",
    "\n",
    "outer_opt = torch.optim.SGD(lr=outer_lr, momentum=outer_mu, params=hparams)\n",
    "#outer_opt = torch.optim.Adam(lr=0.01, params=hparams)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "val_losses, val_accs = [], []\n",
    "test_losses, test_accs = [], []\n",
    "\n",
    "w = torch.zeros(n_features, n_classes).requires_grad_(True)\n",
    "parameters = [w]\n",
    "\n",
    "if bias:\n",
    "    b = torch.zeros(n_classes).requires_grad_(True)\n",
    "    parameters.append(b)\n",
    "\n",
    "\n",
    "def out_f(x, params):\n",
    "    out = x @ params[0]\n",
    "    out += params[1] if len(params) == 2 else 0\n",
    "    return out\n",
    "\n",
    "\n",
    "def train_loss(params, hparams, data):\n",
    "    x_mb, y_mb = data\n",
    "    out = out_f(x_mb,  params)\n",
    "    return F.cross_entropy(out, y_mb) + reg_f(params, *hparams)\n",
    "\n",
    "\n",
    "def val_loss(opt_params, hparams):\n",
    "    x_mb, y_mb = next(val_iterator)\n",
    "    out = out_f(x_mb,  opt_params[:len(parameters)])\n",
    "    val_loss = F.cross_entropy(out, y_mb)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y_mb.view_as(pred)).sum().item() / len(y_mb)\n",
    "\n",
    "    val_losses.append(tonp(val_loss))\n",
    "    val_accs.append(acc)\n",
    "    return val_loss\n",
    "\n",
    "\n",
    "def eval(params, x, y):\n",
    "    out = out_f(x,  params)\n",
    "    loss = F.cross_entropy(out, y)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y.view_as(pred)).sum().item() / len(y)\n",
    "\n",
    "    return loss, acc\n",
    "\n",
    "\n",
    "if inner_mu > 0:\n",
    "    #inner_opt = hg.Momentum(train_loss, inner_lr, inner_mu, data_or_iter=train_iterator)\n",
    "    inner_opt = hg.HeavyBall(train_loss, inner_lr, inner_mu, data_or_iter=train_iterator)\n",
    "else:\n",
    "    inner_opt = hg.GradientDescent(train_loss, inner_lr, data_or_iter=train_iterator)\n",
    "\n",
    "inner_opt_cg = hg.GradientDescent(train_loss, 1., data_or_iter=train_iterator)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "total_time = 0\n",
    "running_time = []\n",
    "for o_step in range(n_steps):\n",
    "    start_time = time.time()\n",
    "\n",
    "    inner_losses = []\n",
    "    if params_history:\n",
    "        params_history = [params_history[-1]]\n",
    "    else:\n",
    "        params_history = [inner_opt.get_opt_params(parameters)]\n",
    "    for t in range(T):\n",
    "        params_history.append(inner_opt(params_history[-1], hparams, create_graph=False))\n",
    "        inner_losses.append(inner_opt.curr_loss)\n",
    "\n",
    "#         if t % train_log_interval == 0 or t == T-1:\n",
    "#             print('t={} loss: {}'.format(t, inner_losses[-1]))\n",
    "\n",
    "    final_params = params_history[-1]\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    #hg.reverse(params_history[-K-1:], hparams, [inner_opt]*K, val_loss)\n",
    "    hg.fixed_point(final_params, hparams, K, inner_opt, val_loss, stochastic=False, tol=tol)\n",
    "    #hg.CG(final_params[:len(parameters)], hparams, K, inner_opt_cg, val_loss, stochastic=False, tol=tol)\n",
    "    outer_opt.step()\n",
    "\n",
    "    for p, new_p in zip(parameters, final_params[:len(parameters)]):\n",
    "        if warm_start:\n",
    "            p.data = new_p\n",
    "        else:\n",
    "            p.data = torch.zeros_like(p)\n",
    "\n",
    "    iter_time = time.time() - start_time\n",
    "    total_time += iter_time\n",
    "    running_time.append(total_time)\n",
    "    if o_step % val_log_interval == 0 or o_step == n_steps-1:\n",
    "        test_loss, test_acc = eval(final_params[:len(parameters)], x_test, y_test)\n",
    "        test_losses.append(test_loss)\n",
    "        test_accs.append(test_acc)\n",
    "        print('o_step={} ({:.2e}s) Val loss: {:.4e}, Val Acc: {:.2f}%'.format(o_step, iter_time, val_losses[-1],\n",
    "                                                                              100*val_accs[-1]))\n",
    "        print('          Test loss: {:.4e}, Test Acc: {:.2f}%'.format(test_loss, 100*test_acc))\n",
    "        print('          l2_hp norm: {:.4e}'.format(torch.norm(hparams[0])))\n",
    "        if len(hparams) == 2:\n",
    "            print('          l1_hp : ', torch.norm(hparams[1]))\n",
    "\n",
    "print('HPO ended in {:.2e} seconds\\n'.format(total_time))\n",
    "\n",
    "plt.title('val_accuracy')\n",
    "plt.plot(running_time, val_accs)\n",
    "plt.show()\n",
    "\n",
    "plt.title('test_accuracy')\n",
    "plt.plot(test_accs)\n",
    "plt.show()\n",
    "\n",
    "val_fp = val_accs\n",
    "run_fp = running_time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 49,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dataset 20newsgroup, train_samples=5657, val_samples=5657, test_samples=7532, n_features=130107, n_classes=20\n",
      "o_step=0 (3.12e-01s) Val loss: 8.9652e-01, Val Acc: 76.21%\n",
      "          Test loss: 1.1634e+00, Test Acc: 66.81%\n",
      "          l2_hp norm: 1.4383e+00\n",
      "o_step=50 (3.36e-01s) Val loss: 5.2042e-01, Val Acc: 85.84%\n",
      "          Test loss: 8.4565e-01, Test Acc: 76.33%\n",
      "          l2_hp norm: 3.6759e+00\n",
      "o_step=100 (3.92e-01s) Val loss: 5.1485e-01, Val Acc: 86.23%\n",
      "          Test loss: 8.4635e-01, Test Acc: 76.47%\n",
      "          l2_hp norm: 3.6934e+00\n",
      "o_step=150 (4.29e-01s) Val loss: 5.1238e-01, Val Acc: 86.46%\n",
      "          Test loss: 8.4594e-01, Test Acc: 76.57%\n",
      "          l2_hp norm: 9.0676e-01\n",
      "o_step=200 (4.99e-01s) Val loss: 5.1387e-01, Val Acc: 86.51%\n",
      "          Test loss: 8.3047e-01, Test Acc: 76.59%\n",
      "          l2_hp norm: 3.4714e+00\n",
      "o_step=250 (5.57e-01s) Val loss: 5.1177e-01, Val Acc: 86.55%\n",
      "          Test loss: 8.2837e-01, Test Acc: 76.66%\n",
      "          l2_hp norm: 4.1199e+00\n",
      "o_step=300 (6.01e-01s) Val loss: 5.0765e-01, Val Acc: 86.53%\n",
      "          Test loss: 8.4038e-01, Test Acc: 76.70%\n",
      "          l2_hp norm: 4.9799e+00\n",
      "o_step=350 (6.66e-01s) Val loss: 5.0624e-01, Val Acc: 86.55%\n",
      "          Test loss: 8.3528e-01, Test Acc: 76.74%\n",
      "          l2_hp norm: 4.9105e+00\n",
      "o_step=400 (7.90e-01s) Val loss: 5.0651e-01, Val Acc: 86.57%\n",
      "          Test loss: 8.2901e-01, Test Acc: 76.69%\n",
      "          l2_hp norm: 6.3503e+00\n",
      "o_step=450 (8.28e-01s) Val loss: 5.0720e-01, Val Acc: 86.60%\n",
      "          Test loss: 8.4408e-01, Test Acc: 76.75%\n",
      "          l2_hp norm: 5.6431e+00\n",
      "o_step=499 (8.74e-01s) Val loss: 5.0473e-01, Val Acc: 86.60%\n",
      "          Test loss: 8.3238e-01, Test Acc: 76.75%\n",
      "          l2_hp norm: 5.2627e+00\n",
      "HPO ended in 2.83e+02 seconds\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAXoAAAEICAYAAABRSj9aAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4xLjEsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy8QZhcZAAAcxUlEQVR4nO3dfZRcd33f8fdnZ/ZZkiVLK2EkIcm2cDAUcCzMUxII4MRpgk3PacCCpDjtKW2IKRAgx+FwHB/npE0fCG0an7SO6/Js13UpOKDWOJi0h8RQyWBCZMfWSLaRbKEZS5a0s9I+zMy3f8zd1Wi1D7OPd+fO53XOnJl772/ufO/K/uxvfve39yoiMDOz7OpIuwAzM1taDnozs4xz0JuZZZyD3sws4xz0ZmYZ56A3M8s4B72ZWcY56G3FkPSMpHcscB83SfrOYtVklgUOerOUSMqnXYO1Bwe9rQiSvgC8DPhzSWVJvyPpDZL+WtJJST+U9NaG9jdJOiRpUNLTkt4n6RXAfwLemOzj5Cyf+cuSfiDptKTDkm6btP1nGj7/sKSbkvW9kj4t6VlJpyR9J1n3VklHJu1j4luKpNsk3S/pi5JOAzdJukbSI8lnHJX0J5K6Gt7/SkkPSToh6ZikT0p6iaQzktY3tLtaUklS5/z+BSzTIsIPP1bEA3gGeEfyejNwHPj71Dsk1ybLA0A/cBq4Iml7CfDK5PVNwHea/Ly3An8v2f+rgWPAu5JtLwMGgd1AJ7AeeG2y7Q7gL5Mac8CbgO5kf0dmOKbbgDHgXcln9gJXA28A8sB24AngI0n71cBR4GNAT7L8+mTbHuA3Gz7nM8B/TPvf0I+V+XCP3laqXwP2RMSeiKhFxEPAPurBD1ADXiWpNyKORsT+uX5ARPxlRPwo2f/fAPcAb0k2vw/4i4i4JyLGIuJ4RDwmqQP4x8CHI+K5iKhGxF9HxEiTH/tIRHw1+cyzEfFoRHw3IioR8Qzwnxtq+BXgJxHx6YgYjojBiPhesu1zyc8ISTnqv5C+MNefgbUHB72tVNuAX02GNE4mwzA/A1wSEUPAe4B/DhyV9A1JPzXXD5D0eknfToY8TiX725Bs3gocnOJtG6j3rqfa1ozDk2p4uaSvS/pJMpzzL5uoAeBrwJWSLqX+bedURPy/edZkGeegt5Wk8VKqh4EvRMTahkd/RPwhQEQ8GBHXUh+2+Tvgz6bYx2y+DDwAbI2Ii6iP76vh8y+b4j0vAMPTbBsC+sYXkp72wAzHCPCnSf07I2IN8MkmaiAihoH7qH/z+HXcm7cZOOhtJTkGXJq8/iLwTkm/KCknqSc52blF0iZJ10vqB0aAMlBt2MeWxhOaM1gNnIiIYUnXAO9t2PYl4B2S3i0pL2m9pNdGRA24G/gjSS9NanujpG7gKaAnOcnbCXyK+tj9bDWcBsrJt5LfbNj2deAlkj4iqVvSakmvb9j+eernJK5Pfl5mU3LQ20ryr4BPJcM07wFuoN7DLVHv3X6C+n+zHdRPUD4PnKA+pv3BZB8PA/uBn0h6YZbP+yBwu6RB4FbqPWQAIuLH1M8HfCz5jMeA1ySbPw78CNibbPvXQEdEnEr2eRfwHPUe/nmzcKbwceq/YAapfyv5bw01DFIflnkn8BPgAPDzDdv/ivq5iu8n4/tmU1KEbzxi1qokPQx8OSLuSrsWW7kc9GYtStLrgIeon2MYTLseW7k8dGOZJml/8sdTkx/vS7u2hZD0OeAvqM+5d8jbjNyjNzPLOPfozcwybsVdVGnDhg2xffv2tMswM2spjz766AsRMfnvNoAVGPTbt29n3759aZdhZtZSJD073TYP3ZiZZZyD3sws4xz0ZmYZ56A3M8s4B72ZWcY56M3MMs5Bb2aWcStuHr2ZLY9aLRit1hit1hir1J9HKzXGqjVGKuOvg3xO9ORz9HR20NOZoztff+7pzJHr0OwflLKIYKRS4+xolTNjVc6OVjgzWk0e516fHX8eq0JKl4Z5yUW9vPf1L1v0/TrozZZQRFCpBWNJiI6H6XiI1tdVGa3EeUF7YduGddOtn9hH9dy+G4O72tC+UqNSW3iYdSa/BLo7p/pF0JH8gsjRnWyrt+047xfHufedv61DSoL3wjA+P6ArEwHduG5oon2FuR6qUvr99dqtax30ZgsVEROB0BgaZ0erDDW8PhcclfPajlSqE8E6VglGpugNjwfpeCgvZudQgq5cR/2Rrz86x1/nOujMd9Cd66C3M8eannzSJkdnTnSPt5n03u7G/eTOX9+Z62CsVmNkrMrwWI2RSv15OFkerlQnXo+MVRmpJNuSdifPjJ3bXqkykrxnrLqwH4oEfZ05ervy9HXl6OvK0Zs8r+vrmljXl2zvnVhO3tPZ+J78ee17OjtQWkm/RBz0tuQigmqt3rOt1IJqNRir1agmPd2JbdWgUqslz0GlcVvD+rFq7bwwHhppCONJX83r6yqcGWn4Wj4HuQ4lgVIPgu58riEkxUVdnUk46oIQ7ZoUwI3L5wV0sq96EOfoTPY1Vdt8hzIRQpXkW0b9l0L9eaThF8fIWI1axHlB3NuZ7TBeSg56o1YLyqMVTp8dY3C44Xn43PLE64Z1w2O1icAeD+l6eMcFIb6UJofxeDCs7smzaU03fV35+rbOcz26/u7x4Jihx9edoyvnQFkK+VwH+VwH/d2OoOXgn3IGjFSq0wb0+PJ5rydtK49UZh1e6OnsYE1PJ6t78qzp7eSivi5e0ln/nzXfIfIdyXNOyXN9OdfwemJbRwf5XH1bZ0dH0ubc+snv78yJ3Hn77zjv67rD2GxmDvoVaqxa43h5lOLgMKXBEYqDI5SSR+O6F8ojDI/VZtyXBKu76wG9uqeTNT15tl7cVw/tZLm+LZ+EeSdrevMTwb66p5OuvGfimrUqB/0yiggGRyr1kD49Qqk8QvH0MKXyuRAfD/ATQ6NT7mNtXycbV3czsLqb122/mA2ruriot/O8oD4/tPP0d+XpaIFpcGa2NBz0iyAiOHb6wt72eT3wJMyn6n135ToYSMJ768V9XL1t3cTyxtU9yXM361d10Z3PpXCEZtbKHPSL4A++8QR3fefpC9Y39r6vftk6Nq7pYWBVNxvXdDOw6lyQr+nNe4zZzJaMg34R7H3mBK+4ZA2/fe3LJ3riG9z7NrMVwkG/QBFBoVjmH169hWuv3JR2OWZmF/BUigU6emqYodEql29clXYpZmZTairoJV0n6UlJBUm3TLH9M5IeSx5PSTrZsO1lkr4p6QlJj0vavnjlp69QLANw+cbVKVdiZja1WYduJOWAO4BrgSPAXkkPRMTj420i4qMN7T8EXNWwi88DfxARD0laBcw86bvFnAt69+jNbGVqpkd/DVCIiEMRMQrcC9wwQ/vdwD0Akq4E8hHxEEBElCPizAJrXlEKpTIX9XayYVVX2qWYmU2pmaDfDBxuWD6SrLuApG3ADuDhZNXLgZOSviLpB5L+bfINYfL7PiBpn6R9pVJpbkeQskKxzOUbV3l6pJmtWM0E/VQJNt2VUW4E7o+I8UsE5oGfBT4OvA64FLjpgp1F3BkRuyJi18DAQBMlrRwHi2UuH/CwjZmtXM0E/RFga8PyFuD5adreSDJs0/DeHyTDPhXgq8BPz6fQlejE0CjHh0Y9Pm9mK1ozQb8X2Clph6Qu6mH+wORGkq4A1gGPTHrvOknj3fS3AY9Pfm+rmjgRu8lBb2Yr16xBn/TEbwYeBJ4A7ouI/ZJul3R9Q9PdwL0R5y54mwzhfBz4lqQfUR8G+rPFPIA0TQS9h27MbAVr6i9jI2IPsGfSulsnLd82zXsfAl49z/pWtEKxTG9njs1re9MuxcxsWv7L2AUolMpcOtDvSwCb2YrmoF+Ag8nUSjOzlcxBP09DIxWeO3mWnQ56M1vhHPTzdLDkSx+YWWtw0M+Tr3FjZq3CQT9PhWKZfIfYtr4/7VLMzGbkoJ+nQrHMtvV9dOb8IzSzlc0pNU8Fz7gxsxbhoJ+H0UqNZ0+cYadvNmJmLcBBPw/PHB+iWgv36M2sJTjo58EzbsyslTjo52E86C8d8IwbM1v5HPTzUCiW2by2l76upq4JZ2aWKgf9PBwoltnpa9CbWYtw0M9RtRYcKvn2gWbWOhz0c/Tci2cZqdR8ItbMWoaDfo4KpUHAM27MrHU46OfIUyvNrNU46OeoUCyzYVUXa/u60i7FzKwpDvo5OuBr3JhZi3HQz0FE+GJmZtZyHPRzUBocYXC44qmVZtZSHPRzcO5ErK9aaWatw0E/BwXfJ9bMWpCDfg4KxTKruvNsWtOddilmZk1z0M/BgWP1E7GS0i7FzKxpDvo5KJQ848bMWo+Dvkmnzo5RGhxx0JtZy3HQN2lixo2nVppZi3HQN+mgr3FjZi3KQd+kA8VBuvIdbL24L+1SzMzmpKmgl3SdpCclFSTdMsX2z0h6LHk8JenkpO1rJD0n6U8Wq/DlViiWuXRDP7kOz7gxs9Yy601PJeWAO4BrgSPAXkkPRMTj420i4qMN7T8EXDVpN78P/J9FqTglhVKZ12xZm3YZZmZz1kyP/hqgEBGHImIUuBe4YYb2u4F7xhckXQ1sAr65kELTNDxW5ciLZz0+b2YtqZmg3wwcblg+kqy7gKRtwA7g4WS5A/g08ImZPkDSByTtk7SvVCo1U/eyOlgqE+ETsWbWmpoJ+qkGpWOatjcC90dENVn+ILAnIg5P076+s4g7I2JXROwaGBhooqTl5btKmVkrm3WMnnoPfmvD8hbg+Wna3gj8VsPyG4GflfRBYBXQJakcERec0F3JCsUyHYIdG/rTLsXMbM6aCfq9wE5JO4DnqIf5eyc3knQFsA54ZHxdRLyvYftNwK5WC3moB/229f1053Npl2JmNmezDt1ERAW4GXgQeAK4LyL2S7pd0vUNTXcD90bEdMM6LatQLHOZ/yLWzFpUMz16ImIPsGfSulsnLd82yz4+C3x2TtWtAJVqjWeOD/H2V2xKuxQzs3nxX8bO4tkTZxirhk/EmlnLctDPwjNuzKzVOehn4aA3s1bnoJ9FoVjmkot6WNXd1OkMM7MVx0E/i0LRd5Uys9bmoJ9BrRYcLHlqpZm1Ngf9DI6eHubMaNU9ejNraQ76GRw4NgjATge9mbUwB/0MPOPGzLLAQT+Dg6Uy6/o6Wb+qO+1SzMzmzUE/A8+4MbMscNDPwEFvZlngoJ/G8fIIL54Z89RKM2t5DvppHEhOxO7ctDrlSszMFsZBPw3PuDGzrHDQT6NQLNPXleOlF/WkXYqZ2YI46KcxfukDaap7o5uZtQ4H/TQ848bMssJBP4XB4TGOnhp20JtZJjjop3CwNAT4RKyZZYODfgqecWNmWeKgn0KhWKYzJ7Zd3Jd2KWZmC+agn0KhWGb7+n7yOf94zKz1OcmmcLDkGTdmlh0O+kmGx6o8e3zINxsxs8xw0E/yzPEhagGXOejNLCMc9JN4xo2ZZY2DfpJCsYyEL09sZpnhoJ+kUCyzZV0vPZ25tEsxM1sUDvpJCsUyl7s3b2YZ4qBvUK0Fh14Y8s1GzCxTmgp6SddJelJSQdItU2z/jKTHksdTkk4m618r6RFJ+yX9jaT3LPYBLKbDJ84wWqm5R29mmZKfrYGkHHAHcC1wBNgr6YGIeHy8TUR8tKH9h4CrksUzwD+KiAOSXgo8KunBiDi5mAexWMZn3HhqpZllSTM9+muAQkQciohR4F7ghhna7wbuAYiIpyLiQPL6eaAIDCys5KVTKHlqpZllTzNBvxk43LB8JFl3AUnbgB3Aw1NsuwboAg5Ose0DkvZJ2lcqlZqpe0kUimUGVndzUW9najWYmS22ZoJ+qnvpxTRtbwTuj4jqeTuQLgG+APxGRNQu2FnEnRGxKyJ2DQyk1+EvFMu+9IGZZU4zQX8E2NqwvAV4fpq2N5IM24yTtAb4BvCpiPjufIpcDhHh2weaWSY1E/R7gZ2Sdkjqoh7mD0xuJOkKYB3wSMO6LuB/Ap+PiP++OCUvjWOnRyiPVBz0ZpY5swZ9RFSAm4EHgSeA+yJiv6TbJV3f0HQ3cG9ENA7rvBv4OeCmhumXr13E+hfNxDVuPLXSzDJm1umVABGxB9gzad2tk5Zvm+J9XwS+uID6lk2hOAh4xo2ZZY//MjZRKJVZ3ZNnYHV32qWYmS0qB31ifMaNNNUkIzOz1uWgT3jGjZlllYMeOHlmlBfKow56M8skBz2+q5SZZZuDnsaplb48sZllj4OeetB35zvYvK437VLMzBadgx44UCxz2cAqch2ecWNm2eOgxzNuzCzb2j7oz4xWeO7kWQe9mWVW2wf9odIQ4Bk3ZpZdbR/0nlppZlnnoC+WyXWI7ev70y7FzGxJtH3QHygOsm19H135tv9RmFlGtX26FYplX4PezDKtrYN+rFrj2eNnPD5vZpnW1kH/7PEhKrVw0JtZprV10HvGjZm1Awc9cJnH6M0sw9o66A8Uy2xe20t/d1O3zjUza0ltHfSFYpnLPGxjZhnXtkFfqwUHS55aaWbZ17ZB/9zJswyP1Xwi1swyr22DvlCqn4jduclBb2bZ1r5Bf2z89oEOejPLtvYN+mKZ9f1drOvvSrsUM7Ml1b5BX/KMGzNrD20Z9BHh2weaWdtoy6B/oTzKqbNjHp83s7bQlkE/fukDz7gxs3bQVNBLuk7Sk5IKkm6ZYvtnJD2WPJ6SdLJh2/slHUge71/M4uerUBwEfDEzM2sPs17kRVIOuAO4FjgC7JX0QEQ8Pt4mIj7a0P5DwFXJ64uB3wN2AQE8mrz3xUU9ijkqFMus6s7zkjU9aZZhZrYsmunRXwMUIuJQRIwC9wI3zNB+N3BP8voXgYci4kQS7g8B1y2k4MVQKJW5bKAfSWmXYma25JoJ+s3A4YblI8m6C0jaBuwAHp7LeyV9QNI+SftKpVIzdS+IL2ZmZu2kmaCfqtsb07S9Ebg/IqpzeW9E3BkRuyJi18DAQBMlzd/p4TGOnR7x+LyZtY1mgv4IsLVheQvw/DRtb+TcsM1c37ssDo7PuNm4Os0yzMyWTTNBvxfYKWmHpC7qYf7A5EaSrgDWAY80rH4Q+AVJ6yStA34hWZeaA759oJm1mVln3URERdLN1AM6B9wdEfsl3Q7si4jx0N8N3BsR0fDeE5J+n/ovC4DbI+LE4h7C3BwslunKdbB1XW+aZZiZLZum7qEXEXuAPZPW3Tpp+bZp3ns3cPc861t0hWKZHRv6yefa8m/FzKwNtV3aFUq+xo2ZtZe2CvrhsSqHT5xx0JtZW2mroH/6hSFq4ROxZtZe2iroPePGzNpRWwV9oVimQ7BjQ3/apZiZLZu2CvqDxTJbL+6jpzOXdilmZsumrYK+UCz7ZiNm1nbaJugr1RpPvzDE5b7ZiJm1mbYJ+h+fOMNoteYevZm1nbYJ+oJn3JhZm2qfoC/Vg97XoTezdtM+QV8ss2lNN2t6OtMuxcxsWbVN0B8s+ho3Ztae2iLoI4JCseybjZhZW2qLoD96apih0arH582sLbVF0E/MuPHUSjNrQ+0V9O7Rm1kbao+gL5W5qLeTDau60i7FzGzZtUfQF8vs3LgKSWmXYma27Noi6D210szaWeaD/sTQKMeHRh30Zta2Mh/04ydiPbXSzNpV2wS9p1aaWbtqi6Dv7cyxeW1v2qWYmaUi+0FfKnPZxn46OjzjxszaU/aD/tigh23MrK1lOuiHRio8f2rYM27MrK1lOugPlnzpAzOzTAe9r3FjZtYGQZ/vENvW96ddiplZajIf9Ns39NOZy/RhmpnNqKkElHSdpCclFSTdMk2bd0t6XNJ+SV9uWP9vknVPSPpjLeOVxQrFsmfcmFnbmzXoJeWAO4BfAq4Edku6clKbncDvAm+OiFcCH0nWvwl4M/Bq4FXA64C3LOYBTGe0UuPZE2c8Pm9mba+ZHv01QCEiDkXEKHAvcMOkNv8UuCMiXgSIiGKyPoAeoAvoBjqBY4tR+GyeOT5EtRYOejNre80E/WbgcMPykWRdo5cDL5f0V5K+K+k6gIh4BPg2cDR5PBgRT0z+AEkfkLRP0r5SqTSf47iAZ9yYmdU1E/RTjanHpOU8sBN4K7AbuEvSWkmXA68AtlD/5fA2ST93wc4i7oyIXRGxa2BgYC71T6tQLCPBZR6jN7M210zQHwG2NixvAZ6fos3XImIsIp4GnqQe/P8A+G5ElCOiDPwv4A0LL3t2hWKZzWt76e3KLcfHmZmtWM0E/V5gp6QdkrqAG4EHJrX5KvDzAJI2UB/KOQT8GHiLpLykTuonYi8YulkKB3xXKTMzoImgj4gKcDPwIPWQvi8i9ku6XdL1SbMHgeOSHqc+Jv+JiDgO3A8cBH4E/BD4YUT8+RIcx3mqteBQyVMrzcygPrY+q4jYA+yZtO7WhtcB/HbyaGxTBf7Zwsucm+dePMtIpeYevZkZGf3L2EJpEPCMGzMzyGrQe2qlmdmETAb9gWNlNqzqZm1fV9qlmJmlLpNBXyiVuXyjr1hpZgYZDPqIqF/MzMM2ZmZABoO+NDjC4HDFUyvNzBKZC/rxE7E7N61OuRIzs5Uhe0Hv+8SamZ0nc0F/4FiZ1d15Nq7uTrsUM7MVIXNBXyiWuWzjKpbxRlZmZita9oK+5Bk3ZmaNMhX0p86OURoccdCbmTXIVNBPzLhx0JuZTchU0B/0NW7MzC6QqaA/UBykK9/BlnV9aZdiZrZiZCroC8Uyl27oJ9fhGTdmZuOyFfSecWNmdoHMBP3wWJUjL5510JuZTZKZoC+PVHjnq1/K1dvWpV2KmdmK0tQ9Y1vBhlXd/PHuq9Iuw8xsxclMj97MzKbmoDczyzgHvZlZxjnozcwyzkFvZpZxDnozs4xz0JuZZZyD3sws4xQRaddwHkkl4NkF7GID8MIildMq2u2Y2+14wcfcLhZyzNsiYmCqDSsu6BdK0r6I2JV2Hcup3Y653Y4XfMztYqmO2UM3ZmYZ56A3M8u4LAb9nWkXkIJ2O+Z2O17wMbeLJTnmzI3Rm5nZ+bLYozczswYOejOzjMtM0Eu6TtKTkgqSbkm7nqUmaaukb0t6QtJ+SR9Ou6blIikn6QeSvp52LctB0lpJ90v6u+Tf+41p17TUJH00+e/6byXdI6kn7ZoWm6S7JRUl/W3DuoslPSTpQPK8KLfMy0TQS8oBdwC/BFwJ7JZ0ZbpVLbkK8LGIeAXwBuC32uCYx30YeCLtIpbRfwD+d0T8FPAaMn7skjYD/wLYFRGvAnLAjelWtSQ+C1w3ad0twLciYifwrWR5wTIR9MA1QCEiDkXEKHAvcEPKNS2piDgaEd9PXg9S/59/c7pVLT1JW4BfBu5Ku5blIGkN8HPAfwGIiNGIOJluVcsiD/RKygN9wPMp17PoIuL/Aicmrb4B+Fzy+nPAuxbjs7IS9JuBww3LR2iD0BsnaTtwFfC9dCtZFv8e+B2glnYhy+RSoAT812S46i5J/WkXtZQi4jng3wE/Bo4CpyLim+lWtWw2RcRRqHfmgI2LsdOsBL2mWNcW80YlrQL+B/CRiDiddj1LSdKvAMWIeDTtWpZRHvhp4E8j4ipgiEX6Or9SJePSNwA7gJcC/ZJ+Ld2qWltWgv4IsLVheQsZ/Ko3maRO6iH/pYj4Str1LIM3A9dLeob68NzbJH0x3ZKW3BHgSESMf1u7n3rwZ9k7gKcjohQRY8BXgDelXNNyOSbpEoDkubgYO81K0O8FdkraIamL+ombB1KuaUlJEvVx2yci4o/Srmc5RMTvRsSWiNhO/d/44YjIdE8vIn4CHJZ0RbLq7cDjKZa0HH4MvEFSX/Lf+dvJ+AnoBg8A709evx/42mLsNL8YO0lbRFQk3Qw8SP0M/d0RsT/lspbam4FfB34k6bFk3ScjYk+KNdnS+BDwpaQTcwj4jZTrWVIR8T1J9wPfpz677Adk8HIIku4B3gpskHQE+D3gD4H7JP0T6r/wfnVRPsuXQDAzy7asDN2Ymdk0HPRmZhnnoDczyzgHvZlZxjnozcwyzkFvZpZxDnozs4z7/wKxtaUquneYAAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from itertools import repeat\n",
    "\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import hypergrad as hg\n",
    "import numpy as np\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.preprocessing import MinMaxScaler\n",
    "from sklearn.datasets import fetch_20newsgroups_vectorized\n",
    "import time\n",
    "\n",
    "import torch.nn.functional as F\n",
    "\n",
    "\n",
    "# Helper functions to deal with cuda\n",
    "cuda = True and torch.cuda.is_available()\n",
    "\n",
    "default_tensor_str = 'torch.cuda.FloatTensor' if cuda else 'torch.FloatTensor'\n",
    "\n",
    "kwargs = {'num_workers': 1, 'pin_memory': True} if cuda else {}\n",
    "torch.set_default_tensor_type(default_tensor_str)\n",
    "#torch.multiprocessing.set_start_method('forkserver')\n",
    "\n",
    "def frnp(x): return torch.from_numpy(x).cuda().float() if cuda else torch.from_numpy(x).float()\n",
    "def tonp(x, cuda=cuda): return x.detach().cpu().numpy() if cuda else x.detach().numpy()\n",
    "\n",
    "\n",
    "seed = 0\n",
    "torch.manual_seed(seed)\n",
    "np.random.seed(seed)\n",
    "\n",
    "\n",
    "# load twentynews and preprocess\n",
    "val_size = 0.5\n",
    "X, y = fetch_20newsgroups_vectorized(subset='train', return_X_y=True,\n",
    "                                     #remove=('headers', 'footers', 'quotes')\n",
    "                                     )\n",
    "x_test, y_test = fetch_20newsgroups_vectorized(subset='test', return_X_y=True,\n",
    "                                               #remove=('headers', 'footers', 'quotes')\n",
    "                                               )\n",
    "\n",
    "\n",
    "x_train, x_val, y_train, y_val = train_test_split(X, y, stratify=y, test_size=val_size)\n",
    "\n",
    "\n",
    "train_samples, n_features = x_train.shape\n",
    "test_samples, n_features = x_test.shape\n",
    "val_samples, n_features = x_val.shape\n",
    "n_classes = np.unique(y_train).shape[0]\n",
    "\n",
    "print('Dataset 20newsgroup, train_samples=%i, val_samples=%i, test_samples=%i, n_features=%i, n_classes=%i'\n",
    "      % (train_samples, val_samples, test_samples, n_features, n_classes))\n",
    "\n",
    "\n",
    "ys = [frnp(y_train).long(), frnp(y_val).long(), frnp(y_test).long()]\n",
    "xs = [x_train, x_val, x_test]\n",
    "\n",
    "\n",
    "def from_sparse(x):\n",
    "    x = x.tocoo()\n",
    "    values = x.data\n",
    "    indices = np.vstack((x.row, x.col))\n",
    "\n",
    "    i = torch.LongTensor(indices)\n",
    "    v = torch.FloatTensor(values)\n",
    "    shape = x.shape\n",
    "\n",
    "    return torch.sparse.FloatTensor(i, v, torch.Size(shape))\n",
    "\n",
    "\n",
    "if cuda:\n",
    "    xs = [from_sparse(x).cuda() for x in xs]\n",
    "else:\n",
    "    xs = [from_sparse(x) for x in xs]\n",
    "\n",
    "x_train, x_val, x_test = xs\n",
    "y_train, y_val, y_test = ys\n",
    "\n",
    "\n",
    "class CustomTensorIterator:\n",
    "    def __init__(self, tensor_list, batch_size, **loader_kwargs):\n",
    "        self.loader = DataLoader(TensorDataset(*tensor_list), batch_size=batch_size, **loader_kwargs)\n",
    "        self.iterator = iter(self.loader)\n",
    "\n",
    "    def __next__(self, *args):\n",
    "        try:\n",
    "            idx = next(self.iterator)\n",
    "        except StopIteration:\n",
    "            self.iterator = iter(self.loader)\n",
    "            idx = next(self.iterator)\n",
    "        return idx\n",
    "\n",
    "\n",
    "# torch.DataLoader has problems with sparse tensor on GPU\n",
    "train_batch_size = len(y_train)\n",
    "val_batch_size = len(y_val)\n",
    "\n",
    "iterators = []\n",
    "for bs, x, y in [(train_batch_size, x_train, y_train), (val_batch_size, x_val, y_val)]:\n",
    "    if bs < len(y):\n",
    "        print('making iterator with batch size ', bs)\n",
    "        iterators.append(CustomTensorIterator([x, y], batch_size=bs, shuffle=True, **kwargs))\n",
    "    else:\n",
    "        iterators.append(repeat([x, y]))\n",
    "\n",
    "train_iterator, val_iterator = iterators\n",
    "\n",
    "# HPO set up\n",
    "n_steps = 500\n",
    "outer_lr, outer_mu = 100., 0.0  # nice with 100.0, 0.0 (torch.SGD) tested with T, K = 5, 10 and CG\n",
    "inner_lr, inner_mu = 100., 0.9   # nice with 100., 0.9 (HeavyBall) tested with T, K = 5, 10 and CG\n",
    "T, K = 20, 20\n",
    "tol = 1e-12\n",
    "warm_start = True\n",
    "bias = False  # without bias outer_lr can be bigger (much faster convergence)\n",
    "\n",
    "train_log_interval = 100\n",
    "val_log_interval = 50\n",
    "\n",
    "#l2_reg_params = torch.zeros(n_features).requires_grad_(True)  # one hp per feature\n",
    "l2_reg_params = (-1.*torch.ones(1)).requires_grad_(True)  # one l2 hp only (best when really low)\n",
    "l1_reg_params = (0.*torch.ones(1)).requires_grad_(True)  # one l1 hp only (best when really low)\n",
    "#l1_reg_params = (-1.*torch.ones(n_features)).requires_grad_(True)\n",
    "\n",
    "hparams = [l2_reg_params]\n",
    "\n",
    "ones_dxc = torch.ones(n_features, n_classes)\n",
    "\n",
    "\n",
    "def reg_f(params, l2_reg_params, l1_reg_params=None):\n",
    "    r = 0.5 * ((params[0] ** 2) * torch.exp(l2_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    if l1_reg_params is not None:\n",
    "        r += (params[0].abs() * torch.exp(l1_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    return r\n",
    "\n",
    "\n",
    "outer_opt = torch.optim.SGD(lr=outer_lr, momentum=outer_mu, params=hparams)\n",
    "#outer_opt = torch.optim.Adam(lr=0.01, params=hparams)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "val_losses, val_accs = [], []\n",
    "test_losses, test_accs = [], []\n",
    "\n",
    "w = torch.zeros(n_features, n_classes).requires_grad_(True)\n",
    "parameters = [w]\n",
    "\n",
    "if bias:\n",
    "    b = torch.zeros(n_classes).requires_grad_(True)\n",
    "    parameters.append(b)\n",
    "\n",
    "\n",
    "def out_f(x, params):\n",
    "    out = x @ params[0]\n",
    "    out += params[1] if len(params) == 2 else 0\n",
    "    return out\n",
    "\n",
    "\n",
    "def train_loss(params, hparams, data):\n",
    "    x_mb, y_mb = data\n",
    "    out = out_f(x_mb,  params)\n",
    "    return F.cross_entropy(out, y_mb) + reg_f(params, *hparams)\n",
    "\n",
    "\n",
    "def val_loss(opt_params, hparams):\n",
    "    x_mb, y_mb = next(val_iterator)\n",
    "    out = out_f(x_mb,  opt_params[:len(parameters)])\n",
    "    val_loss = F.cross_entropy(out, y_mb)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y_mb.view_as(pred)).sum().item() / len(y_mb)\n",
    "\n",
    "    val_losses.append(tonp(val_loss))\n",
    "    val_accs.append(acc)\n",
    "    return val_loss\n",
    "\n",
    "\n",
    "def eval(params, x, y):\n",
    "    out = out_f(x,  params)\n",
    "    loss = F.cross_entropy(out, y)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y.view_as(pred)).sum().item() / len(y)\n",
    "\n",
    "    return loss, acc\n",
    "\n",
    "\n",
    "if inner_mu > 0:\n",
    "    #inner_opt = hg.Momentum(train_loss, inner_lr, inner_mu, data_or_iter=train_iterator)\n",
    "    inner_opt = hg.HeavyBall(train_loss, inner_lr, inner_mu, data_or_iter=train_iterator)\n",
    "else:\n",
    "    inner_opt = hg.GradientDescent(train_loss, inner_lr, data_or_iter=train_iterator)\n",
    "\n",
    "inner_opt_cg = hg.GradientDescent(train_loss, 1., data_or_iter=train_iterator)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "total_time = 0\n",
    "running_time = []\n",
    "for o_step in range(n_steps):\n",
    "    start_time = time.time()\n",
    "\n",
    "    inner_losses = []\n",
    "    if params_history:\n",
    "        params_history = [params_history[-1]]\n",
    "    else:\n",
    "        params_history = [inner_opt.get_opt_params(parameters)]\n",
    "    for t in range(T):\n",
    "        params_history.append(inner_opt(params_history[-1], hparams, create_graph=False))\n",
    "        inner_losses.append(inner_opt.curr_loss)\n",
    "\n",
    "#         if t % train_log_interval == 0 or t == T-1:\n",
    "#             print('t={} loss: {}'.format(t, inner_losses[-1]))\n",
    "\n",
    "    final_params = params_history[-1]\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    #hg.reverse(params_history[-K-1:], hparams, [inner_opt]*K, val_loss)\n",
    "    #hg.fixed_point(final_params, hparams, K, inner_opt, val_loss, stochastic=False, tol=tol)\n",
    "    hg.CG(final_params[:len(parameters)], hparams, K, inner_opt_cg, val_loss, stochastic=False, tol=tol)\n",
    "    outer_opt.step()\n",
    "\n",
    "    for p, new_p in zip(parameters, final_params[:len(parameters)]):\n",
    "        if warm_start:\n",
    "            p.data = new_p\n",
    "        else:\n",
    "            p.data = torch.zeros_like(p)\n",
    "\n",
    "    iter_time = time.time() - start_time\n",
    "    total_time += iter_time\n",
    "    running_time.append(total_time)\n",
    "    if o_step % val_log_interval == 0 or o_step == n_steps-1:\n",
    "        test_loss, test_acc = eval(final_params[:len(parameters)], x_test, y_test)\n",
    "        test_losses.append(test_loss)\n",
    "        test_accs.append(test_acc)\n",
    "        print('o_step={} ({:.2e}s) Val loss: {:.4e}, Val Acc: {:.2f}%'.format(o_step, iter_time, val_losses[-1],\n",
    "                                                                              100*val_accs[-1]))\n",
    "        print('          Test loss: {:.4e}, Test Acc: {:.2f}%'.format(test_loss, 100*test_acc))\n",
    "        print('          l2_hp norm: {:.4e}'.format(torch.norm(hparams[0])))\n",
    "        if len(hparams) == 2:\n",
    "            print('          l1_hp : ', torch.norm(hparams[1]))\n",
    "\n",
    "print('HPO ended in {:.2e} seconds\\n'.format(total_time))\n",
    "\n",
    "plt.title('val_accuracy')\n",
    "plt.plot(running_time, val_accs)\n",
    "plt.show()\n",
    "\n",
    "plt.title('test_accuracy')\n",
    "plt.plot(test_accs)\n",
    "plt.show()\n",
    "\n",
    "val_cg = val_accs\n",
    "run_cg = running_time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 50,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dataset 20newsgroup, train_samples=5657, val_samples=5657, test_samples=7532, n_features=130107, n_classes=20\n",
      "o_step=0 (4.34e-01s) Val loss: 8.9652e-01, Val Acc: 76.21%\n",
      "          Test loss: 1.1634e+00, Test Acc: 66.81%\n",
      "          l2_hp norm: 1.0127e+00\n",
      "o_step=50 (4.60e-01s) Val loss: 5.2029e-01, Val Acc: 85.84%\n",
      "          Test loss: 8.5343e-01, Test Acc: 76.34%\n",
      "          l2_hp norm: 1.2556e+00\n",
      "o_step=100 (5.21e-01s) Val loss: 5.1651e-01, Val Acc: 86.16%\n",
      "          Test loss: 8.5102e-01, Test Acc: 76.55%\n",
      "          l2_hp norm: 9.4819e-01\n",
      "o_step=150 (5.56e-01s) Val loss: 5.1297e-01, Val Acc: 86.37%\n",
      "          Test loss: 8.4251e-01, Test Acc: 76.62%\n",
      "          l2_hp norm: 6.4354e-01\n",
      "o_step=200 (6.22e-01s) Val loss: 5.1074e-01, Val Acc: 86.51%\n",
      "          Test loss: 8.3630e-01, Test Acc: 76.62%\n",
      "          l2_hp norm: 6.3619e-01\n",
      "o_step=250 (6.76e-01s) Val loss: 5.0893e-01, Val Acc: 86.51%\n",
      "          Test loss: 8.3402e-01, Test Acc: 76.57%\n",
      "          l2_hp norm: 7.5957e-01\n",
      "o_step=300 (7.30e-01s) Val loss: 5.0748e-01, Val Acc: 86.51%\n",
      "          Test loss: 8.3304e-01, Test Acc: 76.61%\n",
      "          l2_hp norm: 8.4095e-01\n",
      "o_step=350 (7.88e-01s) Val loss: 5.0642e-01, Val Acc: 86.53%\n",
      "          Test loss: 8.3228e-01, Test Acc: 76.71%\n",
      "          l2_hp norm: 8.7810e-01\n",
      "o_step=400 (9.07e-01s) Val loss: 5.0562e-01, Val Acc: 86.57%\n",
      "          Test loss: 8.3151e-01, Test Acc: 76.69%\n",
      "          l2_hp norm: 8.9579e-01\n",
      "o_step=450 (9.59e-01s) Val loss: 5.0498e-01, Val Acc: 86.60%\n",
      "          Test loss: 8.3079e-01, Test Acc: 76.73%\n",
      "          l2_hp norm: 9.0858e-01\n",
      "o_step=499 (1.01e+00s) Val loss: 5.0447e-01, Val Acc: 86.65%\n",
      "          Test loss: 8.3019e-01, Test Acc: 76.74%\n",
      "          l2_hp norm: 9.2077e-01\n",
      "HPO ended in 3.46e+02 seconds\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from itertools import repeat\n",
    "\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import hypergrad as hg\n",
    "import numpy as np\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.preprocessing import MinMaxScaler\n",
    "from sklearn.datasets import fetch_20newsgroups_vectorized\n",
    "import time\n",
    "\n",
    "import torch.nn.functional as F\n",
    "\n",
    "\n",
    "# Helper functions to deal with cuda\n",
    "cuda = True and torch.cuda.is_available()\n",
    "\n",
    "default_tensor_str = 'torch.cuda.FloatTensor' if cuda else 'torch.FloatTensor'\n",
    "\n",
    "kwargs = {'num_workers': 1, 'pin_memory': True} if cuda else {}\n",
    "torch.set_default_tensor_type(default_tensor_str)\n",
    "#torch.multiprocessing.set_start_method('forkserver')\n",
    "\n",
    "def frnp(x): return torch.from_numpy(x).cuda().float() if cuda else torch.from_numpy(x).float()\n",
    "def tonp(x, cuda=cuda): return x.detach().cpu().numpy() if cuda else x.detach().numpy()\n",
    "\n",
    "\n",
    "seed = 0\n",
    "torch.manual_seed(seed)\n",
    "np.random.seed(seed)\n",
    "\n",
    "\n",
    "# load twentynews and preprocess\n",
    "val_size = 0.5\n",
    "X, y = fetch_20newsgroups_vectorized(subset='train', return_X_y=True,\n",
    "                                     #remove=('headers', 'footers', 'quotes')\n",
    "                                     )\n",
    "x_test, y_test = fetch_20newsgroups_vectorized(subset='test', return_X_y=True,\n",
    "                                               #remove=('headers', 'footers', 'quotes')\n",
    "                                               )\n",
    "\n",
    "\n",
    "x_train, x_val, y_train, y_val = train_test_split(X, y, stratify=y, test_size=val_size)\n",
    "\n",
    "\n",
    "train_samples, n_features = x_train.shape\n",
    "test_samples, n_features = x_test.shape\n",
    "val_samples, n_features = x_val.shape\n",
    "n_classes = np.unique(y_train).shape[0]\n",
    "\n",
    "print('Dataset 20newsgroup, train_samples=%i, val_samples=%i, test_samples=%i, n_features=%i, n_classes=%i'\n",
    "      % (train_samples, val_samples, test_samples, n_features, n_classes))\n",
    "\n",
    "\n",
    "ys = [frnp(y_train).long(), frnp(y_val).long(), frnp(y_test).long()]\n",
    "xs = [x_train, x_val, x_test]\n",
    "\n",
    "\n",
    "def from_sparse(x):\n",
    "    x = x.tocoo()\n",
    "    values = x.data\n",
    "    indices = np.vstack((x.row, x.col))\n",
    "\n",
    "    i = torch.LongTensor(indices)\n",
    "    v = torch.FloatTensor(values)\n",
    "    shape = x.shape\n",
    "\n",
    "    return torch.sparse.FloatTensor(i, v, torch.Size(shape))\n",
    "\n",
    "\n",
    "if cuda:\n",
    "    xs = [from_sparse(x).cuda() for x in xs]\n",
    "else:\n",
    "    xs = [from_sparse(x) for x in xs]\n",
    "\n",
    "x_train, x_val, x_test = xs\n",
    "y_train, y_val, y_test = ys\n",
    "\n",
    "\n",
    "class CustomTensorIterator:\n",
    "    def __init__(self, tensor_list, batch_size, **loader_kwargs):\n",
    "        self.loader = DataLoader(TensorDataset(*tensor_list), batch_size=batch_size, **loader_kwargs)\n",
    "        self.iterator = iter(self.loader)\n",
    "\n",
    "    def __next__(self, *args):\n",
    "        try:\n",
    "            idx = next(self.iterator)\n",
    "        except StopIteration:\n",
    "            self.iterator = iter(self.loader)\n",
    "            idx = next(self.iterator)\n",
    "        return idx\n",
    "\n",
    "\n",
    "# torch.DataLoader has problems with sparse tensor on GPU\n",
    "train_batch_size = len(y_train)\n",
    "val_batch_size = len(y_val)\n",
    "\n",
    "iterators = []\n",
    "for bs, x, y in [(train_batch_size, x_train, y_train), (val_batch_size, x_val, y_val)]:\n",
    "    if bs < len(y):\n",
    "        print('making iterator with batch size ', bs)\n",
    "        iterators.append(CustomTensorIterator([x, y], batch_size=bs, shuffle=True, **kwargs))\n",
    "    else:\n",
    "        iterators.append(repeat([x, y]))\n",
    "\n",
    "train_iterator, val_iterator = iterators\n",
    "\n",
    "# HPO set up\n",
    "n_steps = 500\n",
    "outer_lr, outer_mu = 100.0, 0.0  # nice with 100.0, 0.0 (torch.SGD) tested with T, K = 5, 10 and CG\n",
    "inner_lr, inner_mu = 100.0, 0.9   # nice with 100., 0.9 (HeavyBall) tested with T, K = 5, 10 and CG\n",
    "T, K = 20, 20\n",
    "tol = 1e-12\n",
    "warm_start = True\n",
    "bias = False  # without bias outer_lr can be bigger (much faster convergence)\n",
    "\n",
    "train_log_interval = 100\n",
    "val_log_interval = 50\n",
    "\n",
    "#l2_reg_params = torch.zeros(n_features).requires_grad_(True)  # one hp per feature\n",
    "l2_reg_params = (-1.*torch.ones(1)).requires_grad_(True)  # one l2 hp only (best when really low)\n",
    "l1_reg_params = (0.*torch.ones(1)).requires_grad_(True)  # one l1 hp only (best when really low)\n",
    "#l1_reg_params = (-1.*torch.ones(n_features)).requires_grad_(True)\n",
    "\n",
    "hparams = [l2_reg_params]\n",
    "\n",
    "ones_dxc = torch.ones(n_features, n_classes)\n",
    "\n",
    "\n",
    "def reg_f(params, l2_reg_params, l1_reg_params=None):\n",
    "    r = 0.5 * ((params[0] ** 2) * torch.exp(l2_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    if l1_reg_params is not None:\n",
    "        r += (params[0].abs() * torch.exp(l1_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    return r\n",
    "\n",
    "\n",
    "outer_opt = torch.optim.SGD(lr=outer_lr, momentum=outer_mu, params=hparams)\n",
    "#outer_opt = torch.optim.Adam(lr=0.01, params=hparams)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "val_losses, val_accs = [], []\n",
    "test_losses, test_accs = [], []\n",
    "\n",
    "w = torch.zeros(n_features, n_classes).requires_grad_(True)\n",
    "parameters = [w]\n",
    "\n",
    "if bias:\n",
    "    b = torch.zeros(n_classes).requires_grad_(True)\n",
    "    parameters.append(b)\n",
    "\n",
    "\n",
    "def out_f(x, params):\n",
    "    out = x @ params[0]\n",
    "    out += params[1] if len(params) == 2 else 0\n",
    "    return out\n",
    "\n",
    "\n",
    "def train_loss(params, hparams, data):\n",
    "    x_mb, y_mb = data\n",
    "    out = out_f(x_mb,  params)\n",
    "    return F.cross_entropy(out, y_mb) + reg_f(params, *hparams)\n",
    "\n",
    "\n",
    "def val_loss(opt_params, hparams):\n",
    "    x_mb, y_mb = next(val_iterator)\n",
    "    out = out_f(x_mb,  opt_params[:len(parameters)])\n",
    "    val_loss = F.cross_entropy(out, y_mb)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y_mb.view_as(pred)).sum().item() / len(y_mb)\n",
    "\n",
    "    val_losses.append(tonp(val_loss))\n",
    "    val_accs.append(acc)\n",
    "    return val_loss\n",
    "\n",
    "\n",
    "def eval(params, x, y):\n",
    "    out = out_f(x,  params)\n",
    "    loss = F.cross_entropy(out, y)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y.view_as(pred)).sum().item() / len(y)\n",
    "\n",
    "    return loss, acc\n",
    "\n",
    "\n",
    "if inner_mu > 0:\n",
    "    #inner_opt = hg.Momentum(train_loss, inner_lr, inner_mu, data_or_iter=train_iterator)\n",
    "    inner_opt = hg.HeavyBall(train_loss, inner_lr, inner_mu, data_or_iter=train_iterator)\n",
    "else:\n",
    "    inner_opt = hg.GradientDescent(train_loss, inner_lr, data_or_iter=train_iterator)\n",
    "\n",
    "inner_opt_cg = hg.GradientDescent(train_loss, 1., data_or_iter=train_iterator)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "total_time = 0\n",
    "running_time = []\n",
    "for o_step in range(n_steps):\n",
    "    start_time = time.time()\n",
    "\n",
    "    inner_losses = []\n",
    "    if params_history:\n",
    "        params_history = [params_history[-1]]\n",
    "    else:\n",
    "        params_history = [inner_opt.get_opt_params(parameters)]\n",
    "    for t in range(T):\n",
    "        params_history.append(inner_opt(params_history[-1], hparams, create_graph=False))\n",
    "        inner_losses.append(inner_opt.curr_loss)\n",
    "\n",
    "#         if t % train_log_interval == 0 or t == T-1:\n",
    "#             print('t={} loss: {}'.format(t, inner_losses[-1]))\n",
    "\n",
    "    final_params = params_history[-1]\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    hg.reverse(params_history[-K-1:], hparams, [inner_opt]*K, val_loss)\n",
    "    #hg.fixed_point(final_params, hparams, K, inner_opt, val_loss, stochastic=False, tol=tol)\n",
    "    #hg.CG(final_params[:len(parameters)], hparams, K, inner_opt_cg, val_loss, stochastic=False, tol=tol)\n",
    "    outer_opt.step()\n",
    "\n",
    "    for p, new_p in zip(parameters, final_params[:len(parameters)]):\n",
    "        if warm_start:\n",
    "            p.data = new_p\n",
    "        else:\n",
    "            p.data = torch.zeros_like(p)\n",
    "\n",
    "    iter_time = time.time() - start_time\n",
    "    total_time += iter_time\n",
    "    running_time.append(total_time)\n",
    "    if o_step % val_log_interval == 0 or o_step == n_steps-1:\n",
    "        test_loss, test_acc = eval(final_params[:len(parameters)], x_test, y_test)\n",
    "        test_losses.append(test_loss)\n",
    "        test_accs.append(test_acc)\n",
    "        print('o_step={} ({:.2e}s) Val loss: {:.4e}, Val Acc: {:.2f}%'.format(o_step, iter_time, val_losses[-1],\n",
    "                                                                              100*val_accs[-1]))\n",
    "        print('          Test loss: {:.4e}, Test Acc: {:.2f}%'.format(test_loss, 100*test_acc))\n",
    "        print('          l2_hp norm: {:.4e}'.format(torch.norm(hparams[0])))\n",
    "        if len(hparams) == 2:\n",
    "            print('          l1_hp : ', torch.norm(hparams[1]))\n",
    "\n",
    "print('HPO ended in {:.2e} seconds\\n'.format(total_time))\n",
    "\n",
    "plt.title('val_accuracy')\n",
    "plt.plot(running_time, val_accs)\n",
    "plt.show()\n",
    "\n",
    "plt.title('test_accuracy')\n",
    "plt.plot(test_accs)\n",
    "plt.show()\n",
    "\n",
    "val_rv = val_accs\n",
    "run_rv = running_time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 60,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 1080x576 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "s = 10 \n",
    "plt.plot(run_zoj[:s], val_zoj[:s], '-', label='zoj')\n",
    "plt.plot(run_hozog[:s], val_hozog[:s], '-', label='hozog')\n",
    "plt.plot(run_cg[:s], val_cg[:s], '-', label='cg')\n",
    "plt.plot(run_fp[:s], val_fp[:s], '-', label='fp')\n",
    "plt.plot(run_rv[:s], val_rv[:s], '-', label='rv')\n",
    "plt.xlabel('Running Time')\n",
    "plt.ylabel('outer loss')\n",
    "plt.legend(loc='lower right')\n",
    "plt.gcf().set_size_inches(15, 8)\n",
    "plt.legend()\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import pickle\n",
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 52,
   "metadata": {},
   "outputs": [],
   "source": [
    "with open(os.path.join(\"logistic_logs\", \"zoj.txt\"), \"ab\") as f:\n",
    "            pickle.dump([run_zoj, val_zoj], f)\n",
    "with open(os.path.join(\"logistic_logs\", \"hozog.txt\"), \"ab\") as f:\n",
    "            pickle.dump([run_hozog, val_hozog], f)\n",
    "with open(os.path.join(\"logistic_logs\", \"fp.txt\"), \"ab\") as f:\n",
    "            pickle.dump([run_fp, val_fp], f)\n",
    "with open(os.path.join(\"logistic_logs\", \"cg.txt\"), \"ab\") as f:\n",
    "            pickle.dump([run_cg, val_cg], f)\n",
    "with open(os.path.join(\"logistic_logs\", \"rv.txt\"), \"ab\") as f:\n",
    "            pickle.dump([run_rv, val_rv], f)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 57,
   "metadata": {},
   "outputs": [],
   "source": [
    "with open(os.path.join(\"logistic_logs\", \"zoj.txt\"), \"rb\") as f:\n",
    "            run_zoj, val_zoj = pickle.load(f)\n",
    "with open(os.path.join(\"logistic_logs\", \"hozog.txt\"), \"rb\") as f:\n",
    "            run_hozog, val_hozog = pickle.load(f)\n",
    "with open(os.path.join(\"logistic_logs\", \"fp.txt\"), \"rb\") as f:\n",
    "            run_fp, val_fp = pickle.load(f)\n",
    "with open(os.path.join(\"logistic_logs\", \"cg.txt\"), \"rb\") as f:\n",
    "            run_cg, val_cg = pickle.load(f)        \n",
    "with open(os.path.join(\"logistic_logs\", \"rv.txt\"), \"rb\") as f:\n",
    "            run_rv, val_rv = pickle.load(f)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 58,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 1080x576 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.plot(run_zoj, val_zoj, '-', label='zoj')\n",
    "plt.plot(run_hozog, val_hozog, '-', label='hozog')\n",
    "plt.plot(run_cg, val_cg, '-', label='cg')\n",
    "plt.plot(run_fp, val_fp, '-', label='fp')\n",
    "plt.plot(run_rv, val_rv, '-', label='rv')\n",
    "plt.xlabel('Running Time')\n",
    "plt.ylabel('outer loss')\n",
    "plt.legend(loc='lower right')\n",
    "plt.gcf().set_size_inches(15, 8)\n",
    "plt.legend()\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# MANY"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# ZOJ"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dataset 20newsgroup, train_samples=5657, val_samples=5657, test_samples=7532, n_features=130107, n_classes=20\n",
      "logistic_20news with ZOJ, mu = 0.01, beta = 0.01 and T = 5\n",
      "o_step=0 (8.24e-02s) Val loss: 2.1360e+00, Val Acc: 48.65%\n",
      "          Test loss: 2.2243e+00, Test Acc: 43.26%\n",
      "          l2_hp norm: 1.0639e+00\n",
      "o_step=50 (7.83e-02s) Val loss: 5.3847e-01, Val Acc: 85.36%\n",
      "          Test loss: 8.4916e-01, Test Acc: 75.86%\n",
      "          l2_hp norm: 1.1880e+00\n",
      "o_step=100 (7.84e-02s) Val loss: 5.2564e-01, Val Acc: 85.70%\n",
      "          Test loss: 8.4189e-01, Test Acc: 76.01%\n",
      "          l2_hp norm: 1.1880e+00\n",
      "o_step=150 (7.83e-02s) Val loss: 5.2125e-01, Val Acc: 85.93%\n",
      "          Test loss: 8.3902e-01, Test Acc: 76.12%\n",
      "          l2_hp norm: 1.1880e+00\n",
      "o_step=200 (7.85e-02s) Val loss: 5.1897e-01, Val Acc: 86.05%\n",
      "          Test loss: 8.3700e-01, Test Acc: 76.27%\n",
      "          l2_hp norm: 1.1880e+00\n",
      "o_step=250 (7.84e-02s) Val loss: 5.1752e-01, Val Acc: 86.18%\n",
      "          Test loss: 8.3537e-01, Test Acc: 76.35%\n",
      "          l2_hp norm: 1.1880e+00\n",
      "o_step=300 (7.83e-02s) Val loss: 5.1649e-01, Val Acc: 86.28%\n",
      "          Test loss: 8.3401e-01, Test Acc: 76.42%\n",
      "          l2_hp norm: 1.1880e+00\n",
      "o_step=350 (7.82e-02s) Val loss: 5.1571e-01, Val Acc: 86.34%\n",
      "          Test loss: 8.3287e-01, Test Acc: 76.41%\n",
      "          l2_hp norm: 1.1880e+00\n",
      "o_step=400 (7.83e-02s) Val loss: 5.1508e-01, Val Acc: 86.32%\n",
      "          Test loss: 8.3190e-01, Test Acc: 76.39%\n",
      "          l2_hp norm: 1.1880e+00\n",
      "o_step=450 (7.84e-02s) Val loss: 5.1455e-01, Val Acc: 86.37%\n",
      "          Test loss: 8.3107e-01, Test Acc: 76.43%\n",
      "          l2_hp norm: 1.1880e+00\n",
      "o_step=499 (7.83e-02s) Val loss: 5.1411e-01, Val Acc: 86.37%\n",
      "          Test loss: 8.3037e-01, Test Acc: 76.47%\n",
      "          l2_hp norm: 1.1880e+00\n",
      "HPO ended in 3.92e+01 seconds\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from itertools import repeat\n",
    "\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import hypergrad as hg\n",
    "import numpy as np\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.preprocessing import MinMaxScaler\n",
    "from sklearn.datasets import fetch_20newsgroups_vectorized\n",
    "import time\n",
    "import pdb\n",
    "\n",
    "import torch.nn.functional as F\n",
    "\n",
    "experiment_name = 'logistic_20news'\n",
    "algs = {'ZOJ': hg.hgvzoj, 'HOZOG': hg.hozog}\n",
    "\n",
    "# Helper functions to deal with cuda\n",
    "cuda = True and torch.cuda.is_available()\n",
    "\n",
    "default_tensor_str = 'torch.cuda.FloatTensor' if cuda else 'torch.FloatTensor'\n",
    "\n",
    "kwargs = {'num_workers': 4, 'pin_memory': True} if cuda else {}\n",
    "torch.set_default_tensor_type(default_tensor_str)\n",
    "\n",
    "\n",
    "# torch.multiprocessing.set_start_method('forkserver')\n",
    "\n",
    "def frnp(x): return torch.from_numpy(x).cuda().float() if cuda else torch.from_numpy(x).float()\n",
    "\n",
    "\n",
    "def tonp(x, cuda=cuda): return x.detach().cpu().numpy() if cuda else x.detach().numpy()\n",
    "\n",
    "\n",
    "seed = 0\n",
    "torch.manual_seed(seed)\n",
    "np.random.seed(seed)\n",
    "\n",
    "# load twentynews and preprocess\n",
    "val_size = 0.5\n",
    "X, y = fetch_20newsgroups_vectorized(subset='train', return_X_y=True,\n",
    "                                     # remove=('headers', 'footers', 'quotes')\n",
    "                                     )\n",
    "x_test, y_test = fetch_20newsgroups_vectorized(subset='test', return_X_y=True,\n",
    "                                               # remove=('headers', 'footers', 'quotes')\n",
    "                                               )\n",
    "\n",
    "x_train, x_val, y_train, y_val = train_test_split(X, y, stratify=y, test_size=val_size)\n",
    "\n",
    "train_samples, n_features = x_train.shape\n",
    "test_samples, n_features = x_test.shape\n",
    "val_samples, n_features = x_val.shape\n",
    "n_classes = np.unique(y_train).shape[0]\n",
    "\n",
    "print('Dataset 20newsgroup, train_samples=%i, val_samples=%i, test_samples=%i, n_features=%i, n_classes=%i'\n",
    "      % (train_samples, val_samples, test_samples, n_features, n_classes))\n",
    "\n",
    "ys = [frnp(y_train).long(), frnp(y_val).long(), frnp(y_test).long()] # long tensors used for ys\n",
    "xs = [x_train, x_val, x_test]\n",
    "\n",
    "\n",
    "def from_sparse(x):\n",
    "    x = x.tocoo() # tocoo() is a numpy function?\n",
    "    values = x.data\n",
    "    indices = np.vstack((x.row, x.col))\n",
    "\n",
    "    i = torch.LongTensor(indices)\n",
    "    v = torch.FloatTensor(values)\n",
    "    shape = x.shape\n",
    "\n",
    "    return torch.sparse.FloatTensor(i, v, torch.Size(shape)) # float tensors used for xs\n",
    "\n",
    "\n",
    "if cuda:\n",
    "    xs = [from_sparse(x).cuda() for x in xs]\n",
    "else:\n",
    "    xs = [from_sparse(x) for x in xs]\n",
    "\n",
    "x_train, x_val, x_test = xs\n",
    "y_train, y_val, y_test = ys\n",
    "\n",
    "\n",
    "class CustomTensorIterator:\n",
    "    def __init__(self, tensor_list, batch_size, **loader_kwargs):\n",
    "        self.loader = DataLoader(TensorDataset(*tensor_list), batch_size=batch_size, **loader_kwargs)\n",
    "        self.iterator = iter(self.loader)\n",
    "\n",
    "    def __next__(self, *args):\n",
    "        try:\n",
    "            idx = next(self.iterator)\n",
    "        except StopIteration:\n",
    "            self.iterator = iter(self.loader)\n",
    "            idx = next(self.iterator)\n",
    "        return idx\n",
    "\n",
    "\n",
    "# torch.DataLoader has problems with sparse tensor on GPU\n",
    "train_batch_size = len(y_train)\n",
    "val_batch_size = len(y_val)\n",
    "\n",
    "iterators = []\n",
    "for bs, x, y in [(train_batch_size, x_train, y_train), (val_batch_size, x_val, y_val)]:\n",
    "    if bs < len(y):\n",
    "        print('making iterator with batch size ', bs)\n",
    "        iterators.append(CustomTensorIterator([x, y], batch_size=bs, shuffle=True, **kwargs))\n",
    "    else: # only else statement will be executed because of the way they defined train_batch_size and val_batch_size\n",
    "        iterators.append(repeat([x, y]))\n",
    "\n",
    "train_iterator, val_iterator = iterators\n",
    "\n",
    "# HPO set up\n",
    "prefix = 'ZOJ'\n",
    "reg_type = 'EACH' # or EACH\n",
    "mu = 0.01\n",
    "n_steps = 500\n",
    "outer_lr, outer_mu = 0.01, 0.0  # nice with 100.0, 0.0 (torch.SGD) tested with T, K = 5, 10 and CG\n",
    "inner_lr, inner_mu = 100.0, 0.9  # nice with 100., 0.9 (HeavyBall) tested with T, K = 5, 10 and CG\n",
    "T, K = 5, 10\n",
    "tol = 1e-12\n",
    "warm_start = True\n",
    "bias = False  # without bias outer_lr can be bigger (much faster convergence)\n",
    "\n",
    "hyperAlgo = algs[prefix]\n",
    "\n",
    "train_log_interval = 100\n",
    "val_log_interval = 50\n",
    "print(experiment_name + ' with ' + prefix + ', mu = ' + str(mu) + ', beta = ' + str(outer_lr) + ' and T = ' + str(T))\n",
    "\n",
    "if reg_type=='EACH':\n",
    "    l2_reg_params = torch.zeros(n_features).requires_grad_(True)  # one hp per feature\n",
    "    l1_reg_params = (-10. * torch.ones(n_features)).requires_grad_(True)\n",
    "else:\n",
    "    l2_reg_params = (-10.*torch.ones(1)).requires_grad_(True)  # one l2 hp only (best when really low)\n",
    "    l1_reg_params = (0. * torch.ones(1)).requires_grad_(True)  # one l1 hp only (best when really low)\n",
    "\n",
    "\n",
    "hparams = [l2_reg_params]\n",
    "\n",
    "ones_dxc = torch.ones(n_features, n_classes)\n",
    "\n",
    "\n",
    "def reg_f(params, l2_reg_params, l1_reg_params=None):\n",
    "    r = 0.5 * ((params[0] ** 2) * torch.exp(l2_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    if l1_reg_params is not None:\n",
    "        r += (params[0].abs() * torch.exp(l1_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    return r\n",
    "\n",
    "\n",
    "outer_opt = torch.optim.SGD(lr=outer_lr, momentum=outer_mu, params=hparams)\n",
    "# outer_opt = torch.optim.Adam(lr=0.01, params=hparams)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "val_losses, val_accs = [], []\n",
    "test_losses, test_accs = [], []\n",
    "\n",
    "w = torch.zeros(n_features, n_classes).requires_grad_(True)\n",
    "parameters = [w]\n",
    "\n",
    "if bias:\n",
    "    b = torch.zeros(n_classes).requires_grad_(True)\n",
    "    parameters.append(b)\n",
    "\n",
    "initial_parameters = []\n",
    "for param in parameters:\n",
    "    initial_parameters.append(param.clone())\n",
    "\n",
    "def out_f(x, params):\n",
    "    out = x @ params[0]\n",
    "    out += params[1] if len(params) == 2 else 0\n",
    "    return out\n",
    "\n",
    "\n",
    "def train_loss(params, hparams, data):\n",
    "    x_mb, y_mb = data\n",
    "    out = out_f(x_mb, params)\n",
    "    return F.cross_entropy(out, y_mb) + reg_f(params, *hparams)\n",
    "\n",
    "\n",
    "def val_loss(params, hparams, more=False, verbose=False):\n",
    "    x_mb, y_mb = next(val_iterator)\n",
    "    out = out_f(x_mb, params)\n",
    "    val_loss = F.cross_entropy(out, y_mb)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y_mb.view_as(pred)).sum().item() / len(y_mb)\n",
    "\n",
    "    #val_losses.append(tonp(val_loss)) # get rid of it\n",
    "    #val_accs.append(acc) # get rid of it\n",
    "\n",
    "    if more:\n",
    "        return val_loss, acc\n",
    "    else:\n",
    "        return val_loss\n",
    "\n",
    "\n",
    "def eval(params, x, y):\n",
    "    out = out_f(x, params)\n",
    "    loss = F.cross_entropy(out, y)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y.view_as(pred)).sum().item() / len(y)\n",
    "\n",
    "    return loss, acc\n",
    "\n",
    "inner_opt = torch.optim.SGD(lr=inner_lr, momentum=inner_mu, params=parameters)\n",
    "\n",
    "def update_tensor_grads(params, grads):\n",
    "    for l, g in zip(params, grads):\n",
    "        if l.grad is None:\n",
    "            l.grad = torch.zeros_like(l)\n",
    "        if g is not None:\n",
    "            l.grad += g\n",
    "\n",
    "def inner_solver(hparams, steps, optim, params0=None):\n",
    "\n",
    "    if params0 is not None:\n",
    "        for param, param0 in zip(parameters, params0):\n",
    "            param.data = param0.data\n",
    "\n",
    "    for t in range(steps):\n",
    "        data = next(train_iterator)\n",
    "        loss = train_loss(parameters, hparams, data)\n",
    "        optim.zero_grad()\n",
    "        grads = torch.autograd.grad(loss, parameters)\n",
    "        update_tensor_grads(parameters, grads)\n",
    "        optim.step()\n",
    "        #print('inner grad norm:', torch.norm(grads[0]).item())\n",
    "        #print('current train loss:', loss.item())\n",
    "\n",
    "    return [param.clone() for param in parameters]\n",
    "\n",
    "\n",
    "# final_params = inner_solver(hparams, 1000, initial_parameters)\n",
    "# pdb.set_trace()\n",
    "\n",
    "params_history = []\n",
    "total_time = 0\n",
    "running_time = []\n",
    "for o_step in range(n_steps):\n",
    "    start_time = time.time()\n",
    "\n",
    "    final_params = inner_solver(hparams, T, optim=inner_opt, params0=initial_parameters)\n",
    "\n",
    "    # pdb.set_trace()\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    _, vloss, vacc = hyperAlgo(final_params, hparams, val_loss, inner_solver, params0=initial_parameters, optim=inner_opt, mu=mu, T=T, p=1, set_grad=True, more=True)\n",
    "    outer_opt.step()\n",
    "    val_losses.append(tonp(vloss))\n",
    "    val_accs.append(vacc)\n",
    "\n",
    "    # for hparam in hparams:\n",
    "    #     hparam.data.clamp_(min=-5.0, max=5.0)\n",
    "    #pdb.set_trace()\n",
    "\n",
    "    for init_p, new_p in zip(initial_parameters, final_params):\n",
    "        if warm_start:\n",
    "            init_p.data = new_p\n",
    "        else:\n",
    "            init_p.data = torch.zeros_like(init_p)\n",
    "\n",
    "    iter_time = time.time() - start_time\n",
    "    total_time += iter_time\n",
    "    running_time.append(total_time)\n",
    "\n",
    "    if o_step % val_log_interval == 0 or o_step == n_steps - 1:\n",
    "        test_loss, test_acc = eval(final_params, x_test, y_test)\n",
    "        test_losses.append(test_loss)\n",
    "        test_accs.append(test_acc)\n",
    "        #print('step train los is: ', step_train_loss)\n",
    "        print('o_step={} ({:.2e}s) Val loss: {:.4e}, Val Acc: {:.2f}%'.format(o_step, iter_time, val_losses[-1],\n",
    "                                                                              100 * val_accs[-1]))\n",
    "        print('          Test loss: {:.4e}, Test Acc: {:.2f}%'.format(test_loss, 100 * test_acc))\n",
    "        print('          l2_hp norm: {:.4e}'.format(torch.norm(hparams[0])))\n",
    "#         print('          l2_hp: {:.4e}'.format(hparams[0].item()))\n",
    "        if len(hparams) == 2:\n",
    "            print('          l1_hp : ', torch.norm(hparams[1]))\n",
    "\n",
    "print('HPO ended in {:.2e} seconds\\n'.format(total_time))\n",
    "\n",
    "plt.title('val_accuracy')\n",
    "plt.plot(running_time, val_accs)\n",
    "#figname = 'logistic_plots/VAL_ACC_' + experiment_name + '_' + prefix + '_' + reg_type + '_' + str(T) + '_' + str(mu) + '_' + str(outer_lr) + '.png'\n",
    "#plt.savefig(figname, bbox_inches='tight')\n",
    "#plt.close()\n",
    "plt.show()\n",
    "\n",
    "\n",
    "plt.title('test_accuracy')\n",
    "plt.plot(test_accs)\n",
    "# figname = 'logistic_plots/TEST_ACC_' + experiment_name + '_' + prefix + '_' + reg_type + '_' + str(T) + '_' + str(mu) + '_' + str(outer_lr) + '.png'\n",
    "# plt.savefig(figname, bbox_inches='tight')\n",
    "# plt.close()\n",
    "plt.show()\n",
    "\n",
    "val_zoj = val_accs\n",
    "run_zoj = running_time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 43,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dataset 20newsgroup, train_samples=5657, val_samples=5657, test_samples=7532, n_features=130107, n_classes=20\n",
      "logistic_20news with HOZOG, mu = 0.01, beta = 0.02 and T = 5\n",
      "o_step=0 (7.83e-02s) Val loss: 2.2558e+00, Val Acc: 48.40%\n",
      "          Test loss: 2.3280e+00, Test Acc: 42.98%\n",
      "          l2_hp norm: 1.3608e+00\n",
      "o_step=50 (7.44e-02s) Val loss: 5.4522e-01, Val Acc: 85.20%\n",
      "          Test loss: 8.5310e-01, Test Acc: 75.84%\n",
      "          l2_hp norm: 1.6060e+00\n",
      "o_step=100 (7.44e-02s) Val loss: 5.2884e-01, Val Acc: 85.68%\n",
      "          Test loss: 8.4345e-01, Test Acc: 75.89%\n",
      "          l2_hp norm: 1.6060e+00\n",
      "o_step=150 (7.44e-02s) Val loss: 5.2335e-01, Val Acc: 85.86%\n",
      "          Test loss: 8.4029e-01, Test Acc: 76.13%\n",
      "          l2_hp norm: 1.6060e+00\n",
      "o_step=200 (7.45e-02s) Val loss: 5.2056e-01, Val Acc: 86.02%\n",
      "          Test loss: 8.3832e-01, Test Acc: 76.22%\n",
      "          l2_hp norm: 1.6060e+00\n",
      "o_step=250 (7.44e-02s) Val loss: 5.1884e-01, Val Acc: 86.07%\n",
      "          Test loss: 8.3678e-01, Test Acc: 76.30%\n",
      "          l2_hp norm: 1.6060e+00\n",
      "o_step=300 (7.45e-02s) Val loss: 5.1765e-01, Val Acc: 86.19%\n",
      "          Test loss: 8.3548e-01, Test Acc: 76.33%\n",
      "          l2_hp norm: 1.6060e+00\n",
      "o_step=350 (7.44e-02s) Val loss: 5.1676e-01, Val Acc: 86.21%\n",
      "          Test loss: 8.3435e-01, Test Acc: 76.39%\n",
      "          l2_hp norm: 1.6060e+00\n",
      "o_step=400 (7.44e-02s) Val loss: 5.1605e-01, Val Acc: 86.30%\n",
      "          Test loss: 8.3338e-01, Test Acc: 76.43%\n",
      "          l2_hp norm: 1.6060e+00\n",
      "o_step=450 (7.44e-02s) Val loss: 5.1548e-01, Val Acc: 86.32%\n",
      "          Test loss: 8.3253e-01, Test Acc: 76.39%\n",
      "          l2_hp norm: 1.6060e+00\n",
      "o_step=499 (7.44e-02s) Val loss: 5.1500e-01, Val Acc: 86.32%\n",
      "          Test loss: 8.3179e-01, Test Acc: 76.38%\n",
      "          l2_hp norm: 1.6060e+00\n",
      "HPO ended in 3.72e+01 seconds\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from itertools import repeat\n",
    "\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import hypergrad as hg\n",
    "import numpy as np\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.preprocessing import MinMaxScaler\n",
    "from sklearn.datasets import fetch_20newsgroups_vectorized\n",
    "import time\n",
    "import pdb\n",
    "\n",
    "import torch.nn.functional as F\n",
    "\n",
    "experiment_name = 'logistic_20news'\n",
    "algs = {'ZOJ': hg.hgvzoj, 'HOZOG': hg.hozog}\n",
    "\n",
    "# Helper functions to deal with cuda\n",
    "cuda = True and torch.cuda.is_available()\n",
    "\n",
    "default_tensor_str = 'torch.cuda.FloatTensor' if cuda else 'torch.FloatTensor'\n",
    "\n",
    "kwargs = {'num_workers': 4, 'pin_memory': True} if cuda else {}\n",
    "torch.set_default_tensor_type(default_tensor_str)\n",
    "\n",
    "\n",
    "# torch.multiprocessing.set_start_method('forkserver')\n",
    "\n",
    "def frnp(x): return torch.from_numpy(x).cuda().float() if cuda else torch.from_numpy(x).float()\n",
    "\n",
    "\n",
    "def tonp(x, cuda=cuda): return x.detach().cpu().numpy() if cuda else x.detach().numpy()\n",
    "\n",
    "\n",
    "seed = 0\n",
    "torch.manual_seed(seed)\n",
    "np.random.seed(seed)\n",
    "\n",
    "# load twentynews and preprocess\n",
    "val_size = 0.5\n",
    "X, y = fetch_20newsgroups_vectorized(subset='train', return_X_y=True,\n",
    "                                     # remove=('headers', 'footers', 'quotes')\n",
    "                                     )\n",
    "x_test, y_test = fetch_20newsgroups_vectorized(subset='test', return_X_y=True,\n",
    "                                               # remove=('headers', 'footers', 'quotes')\n",
    "                                               )\n",
    "\n",
    "x_train, x_val, y_train, y_val = train_test_split(X, y, stratify=y, test_size=val_size)\n",
    "\n",
    "train_samples, n_features = x_train.shape\n",
    "test_samples, n_features = x_test.shape\n",
    "val_samples, n_features = x_val.shape\n",
    "n_classes = np.unique(y_train).shape[0]\n",
    "\n",
    "print('Dataset 20newsgroup, train_samples=%i, val_samples=%i, test_samples=%i, n_features=%i, n_classes=%i'\n",
    "      % (train_samples, val_samples, test_samples, n_features, n_classes))\n",
    "\n",
    "ys = [frnp(y_train).long(), frnp(y_val).long(), frnp(y_test).long()] # long tensors used for ys\n",
    "xs = [x_train, x_val, x_test]\n",
    "\n",
    "\n",
    "def from_sparse(x):\n",
    "    x = x.tocoo() # tocoo() is a numpy function?\n",
    "    values = x.data\n",
    "    indices = np.vstack((x.row, x.col))\n",
    "\n",
    "    i = torch.LongTensor(indices)\n",
    "    v = torch.FloatTensor(values)\n",
    "    shape = x.shape\n",
    "\n",
    "    return torch.sparse.FloatTensor(i, v, torch.Size(shape)) # float tensors used for xs\n",
    "\n",
    "\n",
    "if cuda:\n",
    "    xs = [from_sparse(x).cuda() for x in xs]\n",
    "else:\n",
    "    xs = [from_sparse(x) for x in xs]\n",
    "\n",
    "x_train, x_val, x_test = xs\n",
    "y_train, y_val, y_test = ys\n",
    "\n",
    "\n",
    "class CustomTensorIterator:\n",
    "    def __init__(self, tensor_list, batch_size, **loader_kwargs):\n",
    "        self.loader = DataLoader(TensorDataset(*tensor_list), batch_size=batch_size, **loader_kwargs)\n",
    "        self.iterator = iter(self.loader)\n",
    "\n",
    "    def __next__(self, *args):\n",
    "        try:\n",
    "            idx = next(self.iterator)\n",
    "        except StopIteration:\n",
    "            self.iterator = iter(self.loader)\n",
    "            idx = next(self.iterator)\n",
    "        return idx\n",
    "\n",
    "\n",
    "# torch.DataLoader has problems with sparse tensor on GPU\n",
    "train_batch_size = len(y_train)\n",
    "val_batch_size = len(y_val)\n",
    "\n",
    "iterators = []\n",
    "for bs, x, y in [(train_batch_size, x_train, y_train), (val_batch_size, x_val, y_val)]:\n",
    "    if bs < len(y):\n",
    "        print('making iterator with batch size ', bs)\n",
    "        iterators.append(CustomTensorIterator([x, y], batch_size=bs, shuffle=True, **kwargs))\n",
    "    else: # only else statement will be executed because of the way they defined train_batch_size and val_batch_size\n",
    "        iterators.append(repeat([x, y]))\n",
    "\n",
    "train_iterator, val_iterator = iterators\n",
    "\n",
    "# HPO set up\n",
    "prefix = 'HOZOG'\n",
    "reg_type = 'EACH' # or EACH\n",
    "mu = 0.01\n",
    "n_steps = 500\n",
    "outer_lr, outer_mu = 0.02, 0.0  # nice with 100.0, 0.0 (torch.SGD) tested with T, K = 5, 10 and CG\n",
    "inner_lr, inner_mu = 80.0, 0.9  # nice with 100., 0.9 (HeavyBall) tested with T, K = 5, 10 and CG\n",
    "T, K = 5, 10\n",
    "tol = 1e-12\n",
    "warm_start = True\n",
    "bias = False  # without bias outer_lr can be bigger (much faster convergence)\n",
    "\n",
    "hyperAlgo = algs[prefix]\n",
    "\n",
    "train_log_interval = 100\n",
    "val_log_interval = 50\n",
    "print(experiment_name + ' with ' + prefix + ', mu = ' + str(mu) + ', beta = ' + str(outer_lr) + ' and T = ' + str(T))\n",
    "\n",
    "if reg_type=='EACH':\n",
    "    l2_reg_params = torch.zeros(n_features).requires_grad_(True)  # one hp per feature\n",
    "    l1_reg_params = (-10. * torch.ones(n_features)).requires_grad_(True)\n",
    "else:\n",
    "    l2_reg_params = (-10.*torch.ones(1)).requires_grad_(True)  # one l2 hp only (best when really low)\n",
    "    l1_reg_params = (0. * torch.ones(1)).requires_grad_(True)  # one l1 hp only (best when really low)\n",
    "\n",
    "\n",
    "hparams = [l2_reg_params]\n",
    "\n",
    "ones_dxc = torch.ones(n_features, n_classes)\n",
    "\n",
    "\n",
    "def reg_f(params, l2_reg_params, l1_reg_params=None):\n",
    "    r = 0.5 * ((params[0] ** 2) * torch.exp(l2_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    if l1_reg_params is not None:\n",
    "        r += (params[0].abs() * torch.exp(l1_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    return r\n",
    "\n",
    "\n",
    "outer_opt = torch.optim.SGD(lr=outer_lr, momentum=outer_mu, params=hparams)\n",
    "# outer_opt = torch.optim.Adam(lr=0.01, params=hparams)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "val_losses, val_accs = [], []\n",
    "test_losses, test_accs = [], []\n",
    "\n",
    "w = torch.zeros(n_features, n_classes).requires_grad_(True)\n",
    "parameters = [w]\n",
    "\n",
    "if bias:\n",
    "    b = torch.zeros(n_classes).requires_grad_(True)\n",
    "    parameters.append(b)\n",
    "\n",
    "initial_parameters = []\n",
    "for param in parameters:\n",
    "    initial_parameters.append(param.clone())\n",
    "\n",
    "def out_f(x, params):\n",
    "    out = x @ params[0]\n",
    "    out += params[1] if len(params) == 2 else 0\n",
    "    return out\n",
    "\n",
    "\n",
    "def train_loss(params, hparams, data):\n",
    "    x_mb, y_mb = data\n",
    "    out = out_f(x_mb, params)\n",
    "    return F.cross_entropy(out, y_mb) + reg_f(params, *hparams)\n",
    "\n",
    "\n",
    "def val_loss(params, hparams, more=False, verbose=False):\n",
    "    x_mb, y_mb = next(val_iterator)\n",
    "    out = out_f(x_mb, params)\n",
    "    val_loss = F.cross_entropy(out, y_mb)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y_mb.view_as(pred)).sum().item() / len(y_mb)\n",
    "\n",
    "    #val_losses.append(tonp(val_loss)) # get rid of it\n",
    "    #val_accs.append(acc) # get rid of it\n",
    "\n",
    "    if more:\n",
    "        return val_loss, acc\n",
    "    else:\n",
    "        return val_loss\n",
    "\n",
    "\n",
    "def eval(params, x, y):\n",
    "    out = out_f(x, params)\n",
    "    loss = F.cross_entropy(out, y)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y.view_as(pred)).sum().item() / len(y)\n",
    "\n",
    "    return loss, acc\n",
    "\n",
    "inner_opt = torch.optim.SGD(lr=inner_lr, momentum=inner_mu, params=parameters)\n",
    "\n",
    "def update_tensor_grads(params, grads):\n",
    "    for l, g in zip(params, grads):\n",
    "        if l.grad is None:\n",
    "            l.grad = torch.zeros_like(l)\n",
    "        if g is not None:\n",
    "            l.grad += g\n",
    "\n",
    "def inner_solver(hparams, steps, optim, params0=None):\n",
    "\n",
    "    if params0 is not None:\n",
    "        for param, param0 in zip(parameters, params0):\n",
    "            param.data = param0.data\n",
    "\n",
    "    for t in range(steps):\n",
    "        data = next(train_iterator)\n",
    "        loss = train_loss(parameters, hparams, data)\n",
    "        optim.zero_grad()\n",
    "        grads = torch.autograd.grad(loss, parameters)\n",
    "        update_tensor_grads(parameters, grads)\n",
    "        optim.step()\n",
    "        #print('inner grad norm:', torch.norm(grads[0]).item())\n",
    "        #print('current train loss:', loss.item())\n",
    "\n",
    "    return [param.clone() for param in parameters]\n",
    "\n",
    "\n",
    "# final_params = inner_solver(hparams, 1000, initial_parameters)\n",
    "# pdb.set_trace()\n",
    "\n",
    "params_history = []\n",
    "total_time = 0\n",
    "running_time = []\n",
    "for o_step in range(n_steps):\n",
    "    start_time = time.time()\n",
    "\n",
    "    final_params = inner_solver(hparams, T, optim=inner_opt, params0=initial_parameters)\n",
    "\n",
    "    # pdb.set_trace()\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    _, vloss, vacc = hyperAlgo(final_params, hparams, val_loss, inner_solver, params0=initial_parameters, optim=inner_opt, mu=mu, T=T, p=1, set_grad=True, more=True)\n",
    "    outer_opt.step()\n",
    "    val_losses.append(tonp(vloss))\n",
    "    val_accs.append(vacc)\n",
    "\n",
    "    # for hparam in hparams:\n",
    "    #     hparam.data.clamp_(min=-5.0, max=5.0)\n",
    "    #pdb.set_trace()\n",
    "\n",
    "    for init_p, new_p in zip(initial_parameters, final_params):\n",
    "        if warm_start:\n",
    "            init_p.data = new_p\n",
    "        else:\n",
    "            init_p.data = torch.zeros_like(init_p)\n",
    "\n",
    "    iter_time = time.time() - start_time\n",
    "    total_time += iter_time\n",
    "    running_time.append(total_time)\n",
    "\n",
    "    if o_step % val_log_interval == 0 or o_step == n_steps - 1:\n",
    "        test_loss, test_acc = eval(final_params, x_test, y_test)\n",
    "        test_losses.append(test_loss)\n",
    "        test_accs.append(test_acc)\n",
    "        #print('step train los is: ', step_train_loss)\n",
    "        print('o_step={} ({:.2e}s) Val loss: {:.4e}, Val Acc: {:.2f}%'.format(o_step, iter_time, val_losses[-1],\n",
    "                                                                              100 * val_accs[-1]))\n",
    "        print('          Test loss: {:.4e}, Test Acc: {:.2f}%'.format(test_loss, 100 * test_acc))\n",
    "        print('          l2_hp norm: {:.4e}'.format(torch.norm(hparams[0])))\n",
    "#         print('          l2_hp: {:.4e}'.format(hparams[0].item()))\n",
    "        if len(hparams) == 2:\n",
    "            print('          l1_hp : ', torch.norm(hparams[1]))\n",
    "\n",
    "print('HPO ended in {:.2e} seconds\\n'.format(total_time))\n",
    "\n",
    "plt.title('val_accuracy')\n",
    "plt.plot(running_time, val_accs)\n",
    "#figname = 'logistic_plots/VAL_ACC_' + experiment_name + '_' + prefix + '_' + reg_type + '_' + str(T) + '_' + str(mu) + '_' + str(outer_lr) + '.png'\n",
    "#plt.savefig(figname, bbox_inches='tight')\n",
    "#plt.close()\n",
    "plt.show()\n",
    "\n",
    "\n",
    "plt.title('test_accuracy')\n",
    "plt.plot(test_accs)\n",
    "# figname = 'logistic_plots/TEST_ACC_' + experiment_name + '_' + prefix + '_' + reg_type + '_' + str(T) + '_' + str(mu) + '_' + str(outer_lr) + '.png'\n",
    "# plt.savefig(figname, bbox_inches='tight')\n",
    "# plt.close()\n",
    "plt.show()\n",
    "\n",
    "val_hozog = val_accs\n",
    "run_hozog = running_time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dataset 20newsgroup, train_samples=5657, val_samples=5657, test_samples=7532, n_features=130107, n_classes=20\n",
      "o_step=0 (1.32e-01s) Val loss: 2.1360e+00, Val Acc: 48.65%\n",
      "          Test loss: 2.2243e+00, Test Acc: 43.26%\n",
      "          l2_hp norm: 5.0090e-03\n",
      "o_step=50 (1.22e-01s) Val loss: 5.3738e-01, Val Acc: 85.35%\n",
      "          Test loss: 8.5174e-01, Test Acc: 75.88%\n",
      "          l2_hp norm: 2.1239e-01\n",
      "o_step=100 (1.23e-01s) Val loss: 5.2627e-01, Val Acc: 85.68%\n",
      "          Test loss: 8.4479e-01, Test Acc: 76.08%\n",
      "          l2_hp norm: 4.2821e-01\n",
      "o_step=150 (1.26e-01s) Val loss: 5.2150e-01, Val Acc: 85.84%\n",
      "          Test loss: 8.4086e-01, Test Acc: 76.26%\n",
      "          l2_hp norm: 6.5130e-01\n",
      "o_step=200 (1.30e-01s) Val loss: 5.1839e-01, Val Acc: 85.95%\n",
      "          Test loss: 8.3763e-01, Test Acc: 76.38%\n",
      "          l2_hp norm: 8.7862e-01\n",
      "o_step=250 (1.33e-01s) Val loss: 5.1594e-01, Val Acc: 86.18%\n",
      "          Test loss: 8.3480e-01, Test Acc: 76.43%\n",
      "          l2_hp norm: 1.1093e+00\n",
      "o_step=300 (1.36e-01s) Val loss: 5.1383e-01, Val Acc: 86.32%\n",
      "          Test loss: 8.3226e-01, Test Acc: 76.49%\n",
      "          l2_hp norm: 1.3434e+00\n",
      "o_step=350 (1.39e-01s) Val loss: 5.1189e-01, Val Acc: 86.44%\n",
      "          Test loss: 8.2994e-01, Test Acc: 76.58%\n",
      "          l2_hp norm: 1.5811e+00\n",
      "o_step=400 (1.45e-01s) Val loss: 5.1005e-01, Val Acc: 86.51%\n",
      "          Test loss: 8.2779e-01, Test Acc: 76.57%\n",
      "          l2_hp norm: 1.8231e+00\n",
      "o_step=450 (1.46e-01s) Val loss: 5.0828e-01, Val Acc: 86.53%\n",
      "          Test loss: 8.2579e-01, Test Acc: 76.59%\n",
      "          l2_hp norm: 2.0700e+00\n",
      "o_step=499 (1.49e-01s) Val loss: 5.0659e-01, Val Acc: 86.53%\n",
      "          Test loss: 8.2392e-01, Test Acc: 76.61%\n",
      "          l2_hp norm: 2.3174e+00\n",
      "HPO ended in 6.68e+01 seconds\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from itertools import repeat\n",
    "\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import hypergrad as hg\n",
    "import numpy as np\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.preprocessing import MinMaxScaler\n",
    "from sklearn.datasets import fetch_20newsgroups_vectorized\n",
    "import time\n",
    "\n",
    "import torch.nn.functional as F\n",
    "\n",
    "\n",
    "# Helper functions to deal with cuda\n",
    "cuda = True and torch.cuda.is_available()\n",
    "\n",
    "default_tensor_str = 'torch.cuda.FloatTensor' if cuda else 'torch.FloatTensor'\n",
    "\n",
    "kwargs = {'num_workers': 1, 'pin_memory': True} if cuda else {}\n",
    "torch.set_default_tensor_type(default_tensor_str)\n",
    "#torch.multiprocessing.set_start_method('forkserver')\n",
    "\n",
    "def frnp(x): return torch.from_numpy(x).cuda().float() if cuda else torch.from_numpy(x).float()\n",
    "def tonp(x, cuda=cuda): return x.detach().cpu().numpy() if cuda else x.detach().numpy()\n",
    "\n",
    "\n",
    "seed = 0\n",
    "torch.manual_seed(seed)\n",
    "np.random.seed(seed)\n",
    "\n",
    "\n",
    "# load twentynews and preprocess\n",
    "val_size = 0.5\n",
    "X, y = fetch_20newsgroups_vectorized(subset='train', return_X_y=True,\n",
    "                                     #remove=('headers', 'footers', 'quotes')\n",
    "                                     )\n",
    "x_test, y_test = fetch_20newsgroups_vectorized(subset='test', return_X_y=True,\n",
    "                                               #remove=('headers', 'footers', 'quotes')\n",
    "                                               )\n",
    "\n",
    "\n",
    "x_train, x_val, y_train, y_val = train_test_split(X, y, stratify=y, test_size=val_size)\n",
    "\n",
    "\n",
    "train_samples, n_features = x_train.shape\n",
    "test_samples, n_features = x_test.shape\n",
    "val_samples, n_features = x_val.shape\n",
    "n_classes = np.unique(y_train).shape[0]\n",
    "\n",
    "print('Dataset 20newsgroup, train_samples=%i, val_samples=%i, test_samples=%i, n_features=%i, n_classes=%i'\n",
    "      % (train_samples, val_samples, test_samples, n_features, n_classes))\n",
    "\n",
    "\n",
    "ys = [frnp(y_train).long(), frnp(y_val).long(), frnp(y_test).long()]\n",
    "xs = [x_train, x_val, x_test]\n",
    "\n",
    "\n",
    "def from_sparse(x):\n",
    "    x = x.tocoo()\n",
    "    values = x.data\n",
    "    indices = np.vstack((x.row, x.col))\n",
    "\n",
    "    i = torch.LongTensor(indices)\n",
    "    v = torch.FloatTensor(values)\n",
    "    shape = x.shape\n",
    "\n",
    "    return torch.sparse.FloatTensor(i, v, torch.Size(shape))\n",
    "\n",
    "\n",
    "if cuda:\n",
    "    xs = [from_sparse(x).cuda() for x in xs]\n",
    "else:\n",
    "    xs = [from_sparse(x) for x in xs]\n",
    "\n",
    "x_train, x_val, x_test = xs\n",
    "y_train, y_val, y_test = ys\n",
    "\n",
    "\n",
    "class CustomTensorIterator:\n",
    "    def __init__(self, tensor_list, batch_size, **loader_kwargs):\n",
    "        self.loader = DataLoader(TensorDataset(*tensor_list), batch_size=batch_size, **loader_kwargs)\n",
    "        self.iterator = iter(self.loader)\n",
    "\n",
    "    def __next__(self, *args):\n",
    "        try:\n",
    "            idx = next(self.iterator)\n",
    "        except StopIteration:\n",
    "            self.iterator = iter(self.loader)\n",
    "            idx = next(self.iterator)\n",
    "        return idx\n",
    "\n",
    "\n",
    "# torch.DataLoader has problems with sparse tensor on GPU\n",
    "train_batch_size = len(y_train)\n",
    "val_batch_size = len(y_val)\n",
    "\n",
    "iterators = []\n",
    "for bs, x, y in [(train_batch_size, x_train, y_train), (val_batch_size, x_val, y_val)]:\n",
    "    if bs < len(y):\n",
    "        print('making iterator with batch size ', bs)\n",
    "        iterators.append(CustomTensorIterator([x, y], batch_size=bs, shuffle=True, **kwargs))\n",
    "    else:\n",
    "        iterators.append(repeat([x, y]))\n",
    "\n",
    "train_iterator, val_iterator = iterators\n",
    "\n",
    "# HPO set up\n",
    "n_steps = 500\n",
    "outer_lr, outer_mu = 100.0, 0.0  # nice with 100.0, 0.0 (torch.SGD) tested with T, K = 5, 10 and CG\n",
    "inner_lr, inner_mu = 100., 0.9   # nice with 100., 0.9 (HeavyBall) tested with T, K = 5, 10 and CG\n",
    "T, K = 5, 10\n",
    "tol = 1e-12\n",
    "warm_start = True\n",
    "bias = False  # without bias outer_lr can be bigger (much faster convergence)\n",
    "\n",
    "train_log_interval = 100\n",
    "val_log_interval = 50\n",
    "\n",
    "l2_reg_params = torch.zeros(n_features).requires_grad_(True)  # one hp per feature\n",
    "#l2_reg_params = (-1.*torch.ones(1)).requires_grad_(True)  # one l2 hp only (best when really low)\n",
    "l1_reg_params = (0.*torch.ones(1)).requires_grad_(True)  # one l1 hp only (best when really low)\n",
    "#l1_reg_params = (-1.*torch.ones(n_features)).requires_grad_(True)\n",
    "\n",
    "hparams = [l2_reg_params]\n",
    "\n",
    "ones_dxc = torch.ones(n_features, n_classes)\n",
    "\n",
    "\n",
    "def reg_f(params, l2_reg_params, l1_reg_params=None):\n",
    "    r = 0.5 * ((params[0] ** 2) * torch.exp(l2_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    if l1_reg_params is not None:\n",
    "        r += (params[0].abs() * torch.exp(l1_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    return r\n",
    "\n",
    "\n",
    "outer_opt = torch.optim.SGD(lr=outer_lr, momentum=outer_mu, params=hparams)\n",
    "#outer_opt = torch.optim.Adam(lr=0.01, params=hparams)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "val_losses, val_accs = [], []\n",
    "test_losses, test_accs = [], []\n",
    "\n",
    "w = torch.zeros(n_features, n_classes).requires_grad_(True)\n",
    "parameters = [w]\n",
    "\n",
    "if bias:\n",
    "    b = torch.zeros(n_classes).requires_grad_(True)\n",
    "    parameters.append(b)\n",
    "\n",
    "\n",
    "def out_f(x, params):\n",
    "    out = x @ params[0]\n",
    "    out += params[1] if len(params) == 2 else 0\n",
    "    return out\n",
    "\n",
    "\n",
    "def train_loss(params, hparams, data):\n",
    "    x_mb, y_mb = data\n",
    "    out = out_f(x_mb,  params)\n",
    "    return F.cross_entropy(out, y_mb) + reg_f(params, *hparams)\n",
    "\n",
    "\n",
    "def val_loss(opt_params, hparams):\n",
    "    x_mb, y_mb = next(val_iterator)\n",
    "    out = out_f(x_mb,  opt_params[:len(parameters)])\n",
    "    val_loss = F.cross_entropy(out, y_mb)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y_mb.view_as(pred)).sum().item() / len(y_mb)\n",
    "\n",
    "    val_losses.append(tonp(val_loss))\n",
    "    val_accs.append(acc)\n",
    "    return val_loss\n",
    "\n",
    "\n",
    "def eval(params, x, y):\n",
    "    out = out_f(x,  params)\n",
    "    loss = F.cross_entropy(out, y)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y.view_as(pred)).sum().item() / len(y)\n",
    "\n",
    "    return loss, acc\n",
    "\n",
    "\n",
    "if inner_mu > 0:\n",
    "    #inner_opt = hg.Momentum(train_loss, inner_lr, inner_mu, data_or_iter=train_iterator)\n",
    "    inner_opt = hg.HeavyBall(train_loss, inner_lr, inner_mu, data_or_iter=train_iterator)\n",
    "else:\n",
    "    inner_opt = hg.GradientDescent(train_loss, inner_lr, data_or_iter=train_iterator)\n",
    "\n",
    "inner_opt_cg = hg.GradientDescent(train_loss, 1., data_or_iter=train_iterator)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "total_time = 0\n",
    "running_time = []\n",
    "for o_step in range(n_steps):\n",
    "    start_time = time.time()\n",
    "\n",
    "    inner_losses = []\n",
    "    if params_history:\n",
    "        params_history = [params_history[-1]]\n",
    "    else:\n",
    "        params_history = [inner_opt.get_opt_params(parameters)]\n",
    "    for t in range(T):\n",
    "        params_history.append(inner_opt(params_history[-1], hparams, create_graph=False))\n",
    "        inner_losses.append(inner_opt.curr_loss)\n",
    "\n",
    "#         if t % train_log_interval == 0 or t == T-1:\n",
    "#             print('t={} loss: {}'.format(t, inner_losses[-1]))\n",
    "\n",
    "    final_params = params_history[-1]\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    #hg.reverse(params_history[-K-1:], hparams, [inner_opt]*K, val_loss)\n",
    "    hg.fixed_point(final_params, hparams, K, inner_opt, val_loss, stochastic=False, tol=tol)\n",
    "    #hg.CG(final_params[:len(parameters)], hparams, K, inner_opt_cg, val_loss, stochastic=False, tol=tol)\n",
    "    outer_opt.step()\n",
    "\n",
    "    for p, new_p in zip(parameters, final_params[:len(parameters)]):\n",
    "        if warm_start:\n",
    "            p.data = new_p\n",
    "        else:\n",
    "            p.data = torch.zeros_like(p)\n",
    "\n",
    "    iter_time = time.time() - start_time\n",
    "    total_time += iter_time\n",
    "    running_time.append(total_time)\n",
    "    if o_step % val_log_interval == 0 or o_step == n_steps-1:\n",
    "        test_loss, test_acc = eval(final_params[:len(parameters)], x_test, y_test)\n",
    "        test_losses.append(test_loss)\n",
    "        test_accs.append(test_acc)\n",
    "        print('o_step={} ({:.2e}s) Val loss: {:.4e}, Val Acc: {:.2f}%'.format(o_step, iter_time, val_losses[-1],\n",
    "                                                                              100*val_accs[-1]))\n",
    "        print('          Test loss: {:.4e}, Test Acc: {:.2f}%'.format(test_loss, 100*test_acc))\n",
    "        print('          l2_hp norm: {:.4e}'.format(torch.norm(hparams[0])))\n",
    "        if len(hparams) == 2:\n",
    "            print('          l1_hp : ', torch.norm(hparams[1]))\n",
    "\n",
    "print('HPO ended in {:.2e} seconds\\n'.format(total_time))\n",
    "\n",
    "plt.title('val_accuracy')\n",
    "plt.plot(running_time, val_accs)\n",
    "plt.show()\n",
    "\n",
    "plt.title('test_accuracy')\n",
    "plt.plot(test_accs)\n",
    "plt.show()\n",
    "\n",
    "val_fp = val_accs\n",
    "run_fp = running_time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dataset 20newsgroup, train_samples=5657, val_samples=5657, test_samples=7532, n_features=130107, n_classes=20\n",
      "o_step=0 (1.26e-01s) Val loss: 2.1360e+00, Val Acc: 48.65%\n",
      "          Test loss: 2.2243e+00, Test Acc: 43.26%\n",
      "          l2_hp norm: 1.4296e-02\n",
      "o_step=50 (1.23e-01s) Val loss: 5.1978e-01, Val Acc: 86.09%\n",
      "          Test loss: 8.3740e-01, Test Acc: 76.35%\n",
      "          l2_hp norm: 2.1678e+01\n",
      "o_step=100 (1.24e-01s) Val loss: 4.6045e-01, Val Acc: 88.17%\n",
      "          Test loss: 7.7947e-01, Test Acc: 78.07%\n",
      "          l2_hp norm: 4.1669e+01\n",
      "o_step=150 (1.27e-01s) Val loss: 4.3076e-01, Val Acc: 89.08%\n",
      "          Test loss: 7.5694e-01, Test Acc: 78.66%\n",
      "          l2_hp norm: 4.8569e+01\n",
      "o_step=200 (1.31e-01s) Val loss: 4.0998e-01, Val Acc: 89.78%\n",
      "          Test loss: 7.4462e-01, Test Acc: 79.16%\n",
      "          l2_hp norm: 5.3211e+01\n",
      "o_step=250 (1.34e-01s) Val loss: 3.9305e-01, Val Acc: 90.21%\n",
      "          Test loss: 7.3697e-01, Test Acc: 79.58%\n",
      "          l2_hp norm: 5.7543e+01\n",
      "o_step=300 (1.38e-01s) Val loss: 3.7978e-01, Val Acc: 90.74%\n",
      "          Test loss: 7.3190e-01, Test Acc: 79.71%\n",
      "          l2_hp norm: 6.1044e+01\n",
      "o_step=350 (1.41e-01s) Val loss: 3.6898e-01, Val Acc: 91.02%\n",
      "          Test loss: 7.2836e-01, Test Acc: 79.70%\n",
      "          l2_hp norm: 6.3994e+01\n",
      "o_step=400 (1.47e-01s) Val loss: 3.5944e-01, Val Acc: 91.41%\n",
      "          Test loss: 7.2625e-01, Test Acc: 79.62%\n",
      "          l2_hp norm: 6.6742e+01\n",
      "o_step=450 (1.47e-01s) Val loss: 3.5078e-01, Val Acc: 91.67%\n",
      "          Test loss: 7.2498e-01, Test Acc: 79.74%\n",
      "          l2_hp norm: 6.9388e+01\n",
      "o_step=499 (1.50e-01s) Val loss: 3.4304e-01, Val Acc: 91.82%\n",
      "          Test loss: 7.2450e-01, Test Acc: 79.73%\n",
      "          l2_hp norm: 7.1890e+01\n",
      "HPO ended in 6.75e+01 seconds\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from itertools import repeat\n",
    "\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import hypergrad as hg\n",
    "import numpy as np\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.preprocessing import MinMaxScaler\n",
    "from sklearn.datasets import fetch_20newsgroups_vectorized\n",
    "import time\n",
    "\n",
    "import torch.nn.functional as F\n",
    "\n",
    "\n",
    "# Helper functions to deal with cuda\n",
    "cuda = True and torch.cuda.is_available()\n",
    "\n",
    "default_tensor_str = 'torch.cuda.FloatTensor' if cuda else 'torch.FloatTensor'\n",
    "\n",
    "kwargs = {'num_workers': 1, 'pin_memory': True} if cuda else {}\n",
    "torch.set_default_tensor_type(default_tensor_str)\n",
    "#torch.multiprocessing.set_start_method('forkserver')\n",
    "\n",
    "def frnp(x): return torch.from_numpy(x).cuda().float() if cuda else torch.from_numpy(x).float()\n",
    "def tonp(x, cuda=cuda): return x.detach().cpu().numpy() if cuda else x.detach().numpy()\n",
    "\n",
    "\n",
    "seed = 0\n",
    "torch.manual_seed(seed)\n",
    "np.random.seed(seed)\n",
    "\n",
    "\n",
    "# load twentynews and preprocess\n",
    "val_size = 0.5\n",
    "X, y = fetch_20newsgroups_vectorized(subset='train', return_X_y=True,\n",
    "                                     #remove=('headers', 'footers', 'quotes')\n",
    "                                     )\n",
    "x_test, y_test = fetch_20newsgroups_vectorized(subset='test', return_X_y=True,\n",
    "                                               #remove=('headers', 'footers', 'quotes')\n",
    "                                               )\n",
    "\n",
    "\n",
    "x_train, x_val, y_train, y_val = train_test_split(X, y, stratify=y, test_size=val_size)\n",
    "\n",
    "\n",
    "train_samples, n_features = x_train.shape\n",
    "test_samples, n_features = x_test.shape\n",
    "val_samples, n_features = x_val.shape\n",
    "n_classes = np.unique(y_train).shape[0]\n",
    "\n",
    "print('Dataset 20newsgroup, train_samples=%i, val_samples=%i, test_samples=%i, n_features=%i, n_classes=%i'\n",
    "      % (train_samples, val_samples, test_samples, n_features, n_classes))\n",
    "\n",
    "\n",
    "ys = [frnp(y_train).long(), frnp(y_val).long(), frnp(y_test).long()]\n",
    "xs = [x_train, x_val, x_test]\n",
    "\n",
    "\n",
    "def from_sparse(x):\n",
    "    x = x.tocoo()\n",
    "    values = x.data\n",
    "    indices = np.vstack((x.row, x.col))\n",
    "\n",
    "    i = torch.LongTensor(indices)\n",
    "    v = torch.FloatTensor(values)\n",
    "    shape = x.shape\n",
    "\n",
    "    return torch.sparse.FloatTensor(i, v, torch.Size(shape))\n",
    "\n",
    "\n",
    "if cuda:\n",
    "    xs = [from_sparse(x).cuda() for x in xs]\n",
    "else:\n",
    "    xs = [from_sparse(x) for x in xs]\n",
    "\n",
    "x_train, x_val, x_test = xs\n",
    "y_train, y_val, y_test = ys\n",
    "\n",
    "\n",
    "class CustomTensorIterator:\n",
    "    def __init__(self, tensor_list, batch_size, **loader_kwargs):\n",
    "        self.loader = DataLoader(TensorDataset(*tensor_list), batch_size=batch_size, **loader_kwargs)\n",
    "        self.iterator = iter(self.loader)\n",
    "\n",
    "    def __next__(self, *args):\n",
    "        try:\n",
    "            idx = next(self.iterator)\n",
    "        except StopIteration:\n",
    "            self.iterator = iter(self.loader)\n",
    "            idx = next(self.iterator)\n",
    "        return idx\n",
    "\n",
    "\n",
    "# torch.DataLoader has problems with sparse tensor on GPU\n",
    "train_batch_size = len(y_train)\n",
    "val_batch_size = len(y_val)\n",
    "\n",
    "iterators = []\n",
    "for bs, x, y in [(train_batch_size, x_train, y_train), (val_batch_size, x_val, y_val)]:\n",
    "    if bs < len(y):\n",
    "        print('making iterator with batch size ', bs)\n",
    "        iterators.append(CustomTensorIterator([x, y], batch_size=bs, shuffle=True, **kwargs))\n",
    "    else:\n",
    "        iterators.append(repeat([x, y]))\n",
    "\n",
    "train_iterator, val_iterator = iterators\n",
    "\n",
    "# HPO set up\n",
    "n_steps = 500\n",
    "outer_lr, outer_mu = 100., 0.0  # nice with 100.0, 0.0 (torch.SGD) tested with T, K = 5, 10 and CG\n",
    "inner_lr, inner_mu = 100., 0.9   # nice with 100., 0.9 (HeavyBall) tested with T, K = 5, 10 and CG\n",
    "T, K = 5, 10\n",
    "tol = 1e-12\n",
    "warm_start = True\n",
    "bias = False  # without bias outer_lr can be bigger (much faster convergence)\n",
    "\n",
    "train_log_interval = 100\n",
    "val_log_interval = 50\n",
    "\n",
    "l2_reg_params = torch.zeros(n_features).requires_grad_(True)  # one hp per feature\n",
    "#l2_reg_params = (-1.*torch.ones(1)).requires_grad_(True)  # one l2 hp only (best when really low)\n",
    "l1_reg_params = (0.*torch.ones(1)).requires_grad_(True)  # one l1 hp only (best when really low)\n",
    "#l1_reg_params = (-1.*torch.ones(n_features)).requires_grad_(True)\n",
    "\n",
    "hparams = [l2_reg_params]\n",
    "\n",
    "ones_dxc = torch.ones(n_features, n_classes)\n",
    "\n",
    "\n",
    "def reg_f(params, l2_reg_params, l1_reg_params=None):\n",
    "    r = 0.5 * ((params[0] ** 2) * torch.exp(l2_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    if l1_reg_params is not None:\n",
    "        r += (params[0].abs() * torch.exp(l1_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    return r\n",
    "\n",
    "\n",
    "outer_opt = torch.optim.SGD(lr=outer_lr, momentum=outer_mu, params=hparams)\n",
    "#outer_opt = torch.optim.Adam(lr=0.01, params=hparams)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "val_losses, val_accs = [], []\n",
    "test_losses, test_accs = [], []\n",
    "\n",
    "w = torch.zeros(n_features, n_classes).requires_grad_(True)\n",
    "parameters = [w]\n",
    "\n",
    "if bias:\n",
    "    b = torch.zeros(n_classes).requires_grad_(True)\n",
    "    parameters.append(b)\n",
    "\n",
    "\n",
    "def out_f(x, params):\n",
    "    out = x @ params[0]\n",
    "    out += params[1] if len(params) == 2 else 0\n",
    "    return out\n",
    "\n",
    "\n",
    "def train_loss(params, hparams, data):\n",
    "    x_mb, y_mb = data\n",
    "    out = out_f(x_mb,  params)\n",
    "    return F.cross_entropy(out, y_mb) + reg_f(params, *hparams)\n",
    "\n",
    "\n",
    "def val_loss(opt_params, hparams):\n",
    "    x_mb, y_mb = next(val_iterator)\n",
    "    out = out_f(x_mb,  opt_params[:len(parameters)])\n",
    "    val_loss = F.cross_entropy(out, y_mb)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y_mb.view_as(pred)).sum().item() / len(y_mb)\n",
    "\n",
    "    val_losses.append(tonp(val_loss))\n",
    "    val_accs.append(acc)\n",
    "    return val_loss\n",
    "\n",
    "\n",
    "def eval(params, x, y):\n",
    "    out = out_f(x,  params)\n",
    "    loss = F.cross_entropy(out, y)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y.view_as(pred)).sum().item() / len(y)\n",
    "\n",
    "    return loss, acc\n",
    "\n",
    "\n",
    "if inner_mu > 0:\n",
    "    #inner_opt = hg.Momentum(train_loss, inner_lr, inner_mu, data_or_iter=train_iterator)\n",
    "    inner_opt = hg.HeavyBall(train_loss, inner_lr, inner_mu, data_or_iter=train_iterator)\n",
    "else:\n",
    "    inner_opt = hg.GradientDescent(train_loss, inner_lr, data_or_iter=train_iterator)\n",
    "\n",
    "inner_opt_cg = hg.GradientDescent(train_loss, 1., data_or_iter=train_iterator)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "total_time = 0\n",
    "running_time = []\n",
    "for o_step in range(n_steps):\n",
    "    start_time = time.time()\n",
    "\n",
    "    inner_losses = []\n",
    "    if params_history:\n",
    "        params_history = [params_history[-1]]\n",
    "    else:\n",
    "        params_history = [inner_opt.get_opt_params(parameters)]\n",
    "    for t in range(T):\n",
    "        params_history.append(inner_opt(params_history[-1], hparams, create_graph=False))\n",
    "        inner_losses.append(inner_opt.curr_loss)\n",
    "\n",
    "#         if t % train_log_interval == 0 or t == T-1:\n",
    "#             print('t={} loss: {}'.format(t, inner_losses[-1]))\n",
    "\n",
    "    final_params = params_history[-1]\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    #hg.reverse(params_history[-K-1:], hparams, [inner_opt]*K, val_loss)\n",
    "    #hg.fixed_point(final_params, hparams, K, inner_opt, val_loss, stochastic=False, tol=tol)\n",
    "    hg.CG(final_params[:len(parameters)], hparams, K, inner_opt_cg, val_loss, stochastic=False, tol=tol)\n",
    "    outer_opt.step()\n",
    "\n",
    "    for p, new_p in zip(parameters, final_params[:len(parameters)]):\n",
    "        if warm_start:\n",
    "            p.data = new_p\n",
    "        else:\n",
    "            p.data = torch.zeros_like(p)\n",
    "\n",
    "    iter_time = time.time() - start_time\n",
    "    total_time += iter_time\n",
    "    running_time.append(total_time)\n",
    "    if o_step % val_log_interval == 0 or o_step == n_steps-1:\n",
    "        test_loss, test_acc = eval(final_params[:len(parameters)], x_test, y_test)\n",
    "        test_losses.append(test_loss)\n",
    "        test_accs.append(test_acc)\n",
    "        print('o_step={} ({:.2e}s) Val loss: {:.4e}, Val Acc: {:.2f}%'.format(o_step, iter_time, val_losses[-1],\n",
    "                                                                              100*val_accs[-1]))\n",
    "        print('          Test loss: {:.4e}, Test Acc: {:.2f}%'.format(test_loss, 100*test_acc))\n",
    "        print('          l2_hp norm: {:.4e}'.format(torch.norm(hparams[0])))\n",
    "        if len(hparams) == 2:\n",
    "            print('          l1_hp : ', torch.norm(hparams[1]))\n",
    "\n",
    "print('HPO ended in {:.2e} seconds\\n'.format(total_time))\n",
    "\n",
    "plt.title('val_accuracy')\n",
    "plt.plot(running_time, val_accs)\n",
    "plt.show()\n",
    "\n",
    "plt.title('test_accuracy')\n",
    "plt.plot(test_accs)\n",
    "plt.show()\n",
    "\n",
    "val_cg = val_accs\n",
    "run_cg = running_time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 40,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dataset 20newsgroup, train_samples=5657, val_samples=5657, test_samples=7532, n_features=130107, n_classes=20\n",
      "o_step=0 (1.15e-01s) Val loss: 2.1360e+00, Val Acc: 48.65%\n",
      "          Test loss: 2.2243e+00, Test Acc: 43.26%\n",
      "          l2_hp norm: 3.7606e-04\n",
      "o_step=50 (1.11e-01s) Val loss: 5.3747e-01, Val Acc: 85.35%\n",
      "          Test loss: 8.5182e-01, Test Acc: 75.88%\n",
      "          l2_hp norm: 6.8502e-02\n",
      "o_step=100 (1.12e-01s) Val loss: 5.2663e-01, Val Acc: 85.68%\n",
      "          Test loss: 8.4513e-01, Test Acc: 76.06%\n",
      "          l2_hp norm: 1.3810e-01\n",
      "o_step=150 (1.15e-01s) Val loss: 5.2227e-01, Val Acc: 85.82%\n",
      "          Test loss: 8.4159e-01, Test Acc: 76.23%\n",
      "          l2_hp norm: 2.0904e-01\n",
      "o_step=200 (1.20e-01s) Val loss: 5.1969e-01, Val Acc: 85.89%\n",
      "          Test loss: 8.3884e-01, Test Acc: 76.27%\n",
      "          l2_hp norm: 2.8032e-01\n",
      "o_step=250 (1.23e-01s) Val loss: 5.1785e-01, Val Acc: 86.09%\n",
      "          Test loss: 8.3657e-01, Test Acc: 76.30%\n",
      "          l2_hp norm: 3.5162e-01\n",
      "o_step=300 (1.26e-01s) Val loss: 5.1640e-01, Val Acc: 86.16%\n",
      "          Test loss: 8.3463e-01, Test Acc: 76.39%\n",
      "          l2_hp norm: 4.2288e-01\n",
      "o_step=350 (1.29e-01s) Val loss: 5.1519e-01, Val Acc: 86.28%\n",
      "          Test loss: 8.3296e-01, Test Acc: 76.47%\n",
      "          l2_hp norm: 4.9410e-01\n",
      "o_step=400 (1.33e-01s) Val loss: 5.1412e-01, Val Acc: 86.30%\n",
      "          Test loss: 8.3151e-01, Test Acc: 76.49%\n",
      "          l2_hp norm: 5.6533e-01\n",
      "o_step=450 (1.35e-01s) Val loss: 5.1316e-01, Val Acc: 86.41%\n",
      "          Test loss: 8.3022e-01, Test Acc: 76.47%\n",
      "          l2_hp norm: 6.3661e-01\n",
      "o_step=499 (1.37e-01s) Val loss: 5.1228e-01, Val Acc: 86.41%\n",
      "          Test loss: 8.2908e-01, Test Acc: 76.49%\n",
      "          l2_hp norm: 7.0658e-01\n",
      "HPO ended in 6.15e+01 seconds\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from itertools import repeat\n",
    "\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import hypergrad as hg\n",
    "import numpy as np\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.preprocessing import MinMaxScaler\n",
    "from sklearn.datasets import fetch_20newsgroups_vectorized\n",
    "import time\n",
    "\n",
    "import torch.nn.functional as F\n",
    "\n",
    "\n",
    "# Helper functions to deal with cuda\n",
    "cuda = True and torch.cuda.is_available()\n",
    "\n",
    "default_tensor_str = 'torch.cuda.FloatTensor' if cuda else 'torch.FloatTensor'\n",
    "\n",
    "kwargs = {'num_workers': 1, 'pin_memory': True} if cuda else {}\n",
    "torch.set_default_tensor_type(default_tensor_str)\n",
    "#torch.multiprocessing.set_start_method('forkserver')\n",
    "\n",
    "def frnp(x): return torch.from_numpy(x).cuda().float() if cuda else torch.from_numpy(x).float()\n",
    "def tonp(x, cuda=cuda): return x.detach().cpu().numpy() if cuda else x.detach().numpy()\n",
    "\n",
    "\n",
    "seed = 0\n",
    "torch.manual_seed(seed)\n",
    "np.random.seed(seed)\n",
    "\n",
    "\n",
    "# load twentynews and preprocess\n",
    "val_size = 0.5\n",
    "X, y = fetch_20newsgroups_vectorized(subset='train', return_X_y=True,\n",
    "                                     #remove=('headers', 'footers', 'quotes')\n",
    "                                     )\n",
    "x_test, y_test = fetch_20newsgroups_vectorized(subset='test', return_X_y=True,\n",
    "                                               #remove=('headers', 'footers', 'quotes')\n",
    "                                               )\n",
    "\n",
    "\n",
    "x_train, x_val, y_train, y_val = train_test_split(X, y, stratify=y, test_size=val_size)\n",
    "\n",
    "\n",
    "train_samples, n_features = x_train.shape\n",
    "test_samples, n_features = x_test.shape\n",
    "val_samples, n_features = x_val.shape\n",
    "n_classes = np.unique(y_train).shape[0]\n",
    "\n",
    "print('Dataset 20newsgroup, train_samples=%i, val_samples=%i, test_samples=%i, n_features=%i, n_classes=%i'\n",
    "      % (train_samples, val_samples, test_samples, n_features, n_classes))\n",
    "\n",
    "\n",
    "ys = [frnp(y_train).long(), frnp(y_val).long(), frnp(y_test).long()]\n",
    "xs = [x_train, x_val, x_test]\n",
    "\n",
    "\n",
    "def from_sparse(x):\n",
    "    x = x.tocoo()\n",
    "    values = x.data\n",
    "    indices = np.vstack((x.row, x.col))\n",
    "\n",
    "    i = torch.LongTensor(indices)\n",
    "    v = torch.FloatTensor(values)\n",
    "    shape = x.shape\n",
    "\n",
    "    return torch.sparse.FloatTensor(i, v, torch.Size(shape))\n",
    "\n",
    "\n",
    "if cuda:\n",
    "    xs = [from_sparse(x).cuda() for x in xs]\n",
    "else:\n",
    "    xs = [from_sparse(x) for x in xs]\n",
    "\n",
    "x_train, x_val, x_test = xs\n",
    "y_train, y_val, y_test = ys\n",
    "\n",
    "\n",
    "class CustomTensorIterator:\n",
    "    def __init__(self, tensor_list, batch_size, **loader_kwargs):\n",
    "        self.loader = DataLoader(TensorDataset(*tensor_list), batch_size=batch_size, **loader_kwargs)\n",
    "        self.iterator = iter(self.loader)\n",
    "\n",
    "    def __next__(self, *args):\n",
    "        try:\n",
    "            idx = next(self.iterator)\n",
    "        except StopIteration:\n",
    "            self.iterator = iter(self.loader)\n",
    "            idx = next(self.iterator)\n",
    "        return idx\n",
    "\n",
    "\n",
    "# torch.DataLoader has problems with sparse tensor on GPU\n",
    "train_batch_size = len(y_train)\n",
    "val_batch_size = len(y_val)\n",
    "\n",
    "iterators = []\n",
    "for bs, x, y in [(train_batch_size, x_train, y_train), (val_batch_size, x_val, y_val)]:\n",
    "    if bs < len(y):\n",
    "        print('making iterator with batch size ', bs)\n",
    "        iterators.append(CustomTensorIterator([x, y], batch_size=bs, shuffle=True, **kwargs))\n",
    "    else:\n",
    "        iterators.append(repeat([x, y]))\n",
    "\n",
    "train_iterator, val_iterator = iterators\n",
    "\n",
    "# HPO set up\n",
    "n_steps = 500\n",
    "outer_lr, outer_mu = 100.0, 0.0  # nice with 100.0, 0.0 (torch.SGD) tested with T, K = 5, 10 and CG\n",
    "inner_lr, inner_mu = 100.0, 0.9   # nice with 100., 0.9 (HeavyBall) tested with T, K = 5, 10 and CG\n",
    "T, K = 5, 10\n",
    "tol = 1e-12\n",
    "warm_start = True\n",
    "bias = False  # without bias outer_lr can be bigger (much faster convergence)\n",
    "\n",
    "train_log_interval = 100\n",
    "val_log_interval = 50\n",
    "\n",
    "l2_reg_params = torch.zeros(n_features).requires_grad_(True)  # one hp per feature\n",
    "#l2_reg_params = (-1.*torch.ones(1)).requires_grad_(True)  # one l2 hp only (best when really low)\n",
    "l1_reg_params = (0.*torch.ones(1)).requires_grad_(True)  # one l1 hp only (best when really low)\n",
    "#l1_reg_params = (-1.*torch.ones(n_features)).requires_grad_(True)\n",
    "\n",
    "hparams = [l2_reg_params]\n",
    "\n",
    "ones_dxc = torch.ones(n_features, n_classes)\n",
    "\n",
    "\n",
    "def reg_f(params, l2_reg_params, l1_reg_params=None):\n",
    "    r = 0.5 * ((params[0] ** 2) * torch.exp(l2_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    if l1_reg_params is not None:\n",
    "        r += (params[0].abs() * torch.exp(l1_reg_params.unsqueeze(1) * ones_dxc)).mean()\n",
    "    return r\n",
    "\n",
    "\n",
    "outer_opt = torch.optim.SGD(lr=outer_lr, momentum=outer_mu, params=hparams)\n",
    "#outer_opt = torch.optim.Adam(lr=0.01, params=hparams)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "val_losses, val_accs = [], []\n",
    "test_losses, test_accs = [], []\n",
    "\n",
    "w = torch.zeros(n_features, n_classes).requires_grad_(True)\n",
    "parameters = [w]\n",
    "\n",
    "if bias:\n",
    "    b = torch.zeros(n_classes).requires_grad_(True)\n",
    "    parameters.append(b)\n",
    "\n",
    "\n",
    "def out_f(x, params):\n",
    "    out = x @ params[0]\n",
    "    out += params[1] if len(params) == 2 else 0\n",
    "    return out\n",
    "\n",
    "\n",
    "def train_loss(params, hparams, data):\n",
    "    x_mb, y_mb = data\n",
    "    out = out_f(x_mb,  params)\n",
    "    return F.cross_entropy(out, y_mb) + reg_f(params, *hparams)\n",
    "\n",
    "\n",
    "def val_loss(opt_params, hparams):\n",
    "    x_mb, y_mb = next(val_iterator)\n",
    "    out = out_f(x_mb,  opt_params[:len(parameters)])\n",
    "    val_loss = F.cross_entropy(out, y_mb)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y_mb.view_as(pred)).sum().item() / len(y_mb)\n",
    "\n",
    "    val_losses.append(tonp(val_loss))\n",
    "    val_accs.append(acc)\n",
    "    return val_loss\n",
    "\n",
    "\n",
    "def eval(params, x, y):\n",
    "    out = out_f(x,  params)\n",
    "    loss = F.cross_entropy(out, y)\n",
    "    pred = out.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "    acc = pred.eq(y.view_as(pred)).sum().item() / len(y)\n",
    "\n",
    "    return loss, acc\n",
    "\n",
    "\n",
    "if inner_mu > 0:\n",
    "    #inner_opt = hg.Momentum(train_loss, inner_lr, inner_mu, data_or_iter=train_iterator)\n",
    "    inner_opt = hg.HeavyBall(train_loss, inner_lr, inner_mu, data_or_iter=train_iterator)\n",
    "else:\n",
    "    inner_opt = hg.GradientDescent(train_loss, inner_lr, data_or_iter=train_iterator)\n",
    "\n",
    "inner_opt_cg = hg.GradientDescent(train_loss, 1., data_or_iter=train_iterator)\n",
    "\n",
    "\n",
    "params_history = []\n",
    "total_time = 0\n",
    "running_time = []\n",
    "for o_step in range(n_steps):\n",
    "    start_time = time.time()\n",
    "\n",
    "    inner_losses = []\n",
    "    if params_history:\n",
    "        params_history = [params_history[-1]]\n",
    "    else:\n",
    "        params_history = [inner_opt.get_opt_params(parameters)]\n",
    "    for t in range(T):\n",
    "        params_history.append(inner_opt(params_history[-1], hparams, create_graph=False))\n",
    "        inner_losses.append(inner_opt.curr_loss)\n",
    "\n",
    "#         if t % train_log_interval == 0 or t == T-1:\n",
    "#             print('t={} loss: {}'.format(t, inner_losses[-1]))\n",
    "\n",
    "    final_params = params_history[-1]\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    hg.reverse(params_history[-K-1:], hparams, [inner_opt]*K, val_loss)\n",
    "    #hg.fixed_point(final_params, hparams, K, inner_opt, val_loss, stochastic=False, tol=tol)\n",
    "    #hg.CG(final_params[:len(parameters)], hparams, K, inner_opt_cg, val_loss, stochastic=False, tol=tol)\n",
    "    outer_opt.step()\n",
    "\n",
    "    for p, new_p in zip(parameters, final_params[:len(parameters)]):\n",
    "        if warm_start:\n",
    "            p.data = new_p\n",
    "        else:\n",
    "            p.data = torch.zeros_like(p)\n",
    "\n",
    "    iter_time = time.time() - start_time\n",
    "    total_time += iter_time\n",
    "    running_time.append(total_time)\n",
    "    if o_step % val_log_interval == 0 or o_step == n_steps-1:\n",
    "        test_loss, test_acc = eval(final_params[:len(parameters)], x_test, y_test)\n",
    "        test_losses.append(test_loss)\n",
    "        test_accs.append(test_acc)\n",
    "        print('o_step={} ({:.2e}s) Val loss: {:.4e}, Val Acc: {:.2f}%'.format(o_step, iter_time, val_losses[-1],\n",
    "                                                                              100*val_accs[-1]))\n",
    "        print('          Test loss: {:.4e}, Test Acc: {:.2f}%'.format(test_loss, 100*test_acc))\n",
    "        print('          l2_hp norm: {:.4e}'.format(torch.norm(hparams[0])))\n",
    "        if len(hparams) == 2:\n",
    "            print('          l1_hp : ', torch.norm(hparams[1]))\n",
    "\n",
    "print('HPO ended in {:.2e} seconds\\n'.format(total_time))\n",
    "\n",
    "plt.title('val_accuracy')\n",
    "plt.plot(running_time, val_accs)\n",
    "plt.show()\n",
    "\n",
    "plt.title('test_accuracy')\n",
    "plt.plot(test_accs)\n",
    "plt.show()\n",
    "\n",
    "val_rv = val_accs\n",
    "run_rv = running_time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 45,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 1080x576 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "s = 20\n",
    "plt.plot(run_zoj[:s], val_zoj[:s], '-', label='zoj')\n",
    "plt.plot(run_hozog[:s], val_hozog[:s], '-', label='hozog')\n",
    "plt.plot(run_cg[:s], val_cg[:s], '-', label='cg')\n",
    "#plt.plot(run_fp[:s], val_fp[:s], '-', label='fp')\n",
    "plt.plot(run_rv[:s], val_rv[:s], 'r-', label='rv')\n",
    "plt.xlabel('Running Time')\n",
    "plt.ylabel('outer loss')\n",
    "plt.legend(loc='lower right')\n",
    "plt.gcf().set_size_inches(15, 8)\n",
    "plt.legend()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3.7-2019.10 [python/3.7-2019.10 cuda/9.2.88]",
   "language": "python",
   "name": "sys_python37_2019_10"
  },
  "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.7.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
