{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "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",
    "import dill\n",
    "\n",
    "import torch.nn.functional as F \n",
    "%matplotlib inline"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "seed = 2023\n",
    "torch.random.manual_seed(seed)\n",
    "np.random.seed(seed)\n",
    "torch.backends.cudnn.deterministic = True\n",
    "torch.backends.cudnn.benchmark = False"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dataset 20newsgroup, train_samples=5657, val_samples=5657, test_samples=7532, n_features=99238, n_classes=20\n"
     ]
    }
   ],
   "source": [
    "# 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=('footers', 'quotes')\n",
    "                                     )\n",
    "x_test, y_test = fetch_20newsgroups_vectorized(subset='test', return_X_y=True,\n",
    "                                               remove=('footers', 'quotes') # 'headers'\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_lower, x_upper, x_test = xs\n",
    "y_lower, y_upper, 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",
    "lower_size = 50 # len(y_train)\n",
    "upper_size = 50 # len(y_val) "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Define problem"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_cl_loss(parameters, data):\n",
    "    \n",
    "    W = parameters[0]\n",
    "    b = parameters[1]\n",
    "    x, y = data  \n",
    "    out = x @ W + b \n",
    "    loss = F.cross_entropy(out, y) \n",
    "    return loss \n",
    "\n",
    "def reg_func(ys, xs, lamda):\n",
    "    \n",
    "    return 0.5 * lamda * (torch.norm(ys[0] - xs[0])**2 + torch.norm(ys[1] - xs[1])**2)\n",
    "    \n",
    "def inner_func(ys, xs, data, lamda): \n",
    "\n",
    "    g = get_cl_loss(ys, data) + reg_func(ys, xs, lamda=lamda)\n",
    "    return g \n",
    "\n",
    "def outer_func(ys, xs, outer_data, lower_data): \n",
    "    \n",
    "    f = 0.5 * get_cl_loss(xs, outer_data) + 0.5 * get_cl_loss(ys, lower_data) \n",
    "    return f \n",
    "\n",
    "\n",
    "################################################ \n",
    "\n",
    "def map_func(ys, xs, data, s, lamda): \n",
    "    \n",
    "    g = inner_func(ys, xs, data, lamda)\n",
    "    \n",
    "    grads = torch.autograd.grad(g, ys, create_graph=True)\n",
    "    \n",
    "    return [y - s * grad for (y, grad) in zip(ys, grads)]\n",
    "\n",
    "def inner_solver(ys, xs, data, s, steps, lamda): \n",
    "     \n",
    "    for _ in range(steps): \n",
    "        ys = [y.detach().requires_grad_(True) for y in ys]\n",
    "        ys = map_func(ys, xs, data, s, lamda)\n",
    "        \n",
    "    return [y.clone().detach() for y in ys]\n",
    "\n",
    "def update_tensor_grads(tensors, grads):\n",
    "    for l, g in zip(tensors, 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 getTimeAveHypergrad(grads_list, u):\n",
    "    W = len(grads_list)\n",
    "    weights = list(u[-W:])\n",
    "    TAH = [torch.zeros_like(p) for p in grads_list[0]]\n",
    "    \n",
    "    with torch.no_grad():\n",
    "        for grads, w in zip(grads_list, weights):\n",
    "            TAH = [av + w*g for av, g in zip(TAH, grads)]\n",
    "            \n",
    "        return [av / torch.sum(u[-W:]) for av in TAH]\n",
    "\n",
    "def eval_model(params, x, y): \n",
    "    with torch.no_grad(): \n",
    "        out = x @ params[0] \n",
    "        out += params[1] if len(params)==2 else 0. # add bias if using it \n",
    "        loss = F.cross_entropy(out, y).item()  \n",
    "\n",
    "        pred = out.argmax(dim=1, keepdim=True) # \n",
    "        acc = pred.eq(y.view_as(pred)).sum().item() / len(y)\n",
    "\n",
    "    return acc, loss \n",
    "\n",
    "p0 = [torch.randn(n_features, n_classes), torch.zeros(n_classes)]\n",
    "hp0 = [torch.randn(n_features, n_classes), torch.zeros(n_classes)]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Ours"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Performance on lower split before training: Accuracy=0.04189499734841789, Loss=3.512615442276001\n",
      "Performance on upper split before training: Accuracy=0.03676860526780979, Loss=3.53226375579834\n",
      "Round: 0, outer loss: 3.5138163566589355\n",
      "Round: 10, outer loss: 2.9770889282226562\n",
      "Round: 20, outer loss: 2.5704376697540283\n",
      "Round: 30, outer loss: 1.8362114429473877\n",
      "Round: 40, outer loss: 1.6301753520965576\n",
      "Round: 50, outer loss: 1.2821398973464966\n",
      "Round: 60, outer loss: 1.2371129989624023\n",
      "Round: 70, outer loss: 0.9519717693328857\n",
      "Round: 80, outer loss: 0.9639967679977417\n",
      "Round: 90, outer loss: 0.7088080644607544\n",
      "Round: 100, outer loss: 0.8735253810882568\n",
      "Round: 110, outer loss: 0.5299795866012573\n",
      "Round: 120, outer loss: 0.17895488440990448\n",
      "Round: 130, outer loss: 0.12130094319581985\n",
      "Round: 140, outer loss: 0.1551232635974884\n",
      "Round: 150, outer loss: 0.17779263854026794\n",
      "Performance on lower split after step 150: Accuracy=0.9333569029520947, Loss=0.2611349821090698\n",
      "Performance on upper split after step 150: Accuracy=0.99717164574863, Loss=0.03229682147502899\n",
      "Performance on lower split after step 151: Accuracy=0.9338872193742266, Loss=0.2586894929409027\n",
      "Performance on upper split after step 151: Accuracy=0.99717164574863, Loss=0.03144893795251846\n",
      "Performance on lower split after step 152: Accuracy=0.9344175357963584, Loss=0.2559218406677246\n",
      "Performance on upper split after step 152: Accuracy=0.99717164574863, Loss=0.030491910874843597\n",
      "Performance on lower split after step 153: Accuracy=0.935124624359201, Loss=0.2538340091705322\n",
      "Performance on upper split after step 153: Accuracy=0.9969948736079194, Loss=0.029791662469506264\n",
      "Performance on lower split after step 154: Accuracy=0.9361852572034647, Loss=0.25046345591545105\n",
      "Performance on upper split after step 154: Accuracy=0.9973484178893406, Loss=0.028856687247753143\n",
      "Performance on lower split after step 155: Accuracy=0.9372458900477285, Loss=0.24806340038776398\n",
      "Performance on upper split after step 155: Accuracy=0.9973484178893406, Loss=0.028202896937727928\n",
      "Performance on lower split after step 156: Accuracy=0.9377762064698604, Loss=0.2460901290178299\n",
      "Performance on upper split after step 156: Accuracy=0.9973484178893406, Loss=0.02765466272830963\n",
      "Performance on lower split after step 157: Accuracy=0.9383065228919922, Loss=0.24340012669563293\n",
      "Performance on upper split after step 157: Accuracy=0.9973484178893406, Loss=0.026906877756118774\n",
      "Performance on lower split after step 158: Accuracy=0.9386600671734134, Loss=0.24155452847480774\n",
      "Performance on upper split after step 158: Accuracy=0.9973484178893406, Loss=0.026390885934233665\n",
      "Performance on lower split after step 159: Accuracy=0.9386600671734134, Loss=0.23893864452838898\n",
      "Performance on upper split after step 159: Accuracy=0.9973484178893406, Loss=0.02573743462562561\n",
      "Round: 160, outer loss: 0.09432952105998993\n",
      "Performance on lower split after step 160: Accuracy=0.9398974721583878, Loss=0.23679226636886597\n",
      "Performance on upper split after step 160: Accuracy=0.9973484178893406, Loss=0.02521233633160591\n",
      "Performance on lower split after step 161: Accuracy=0.9404277885805197, Loss=0.23409155011177063\n",
      "Performance on upper split after step 161: Accuracy=0.9973484178893406, Loss=0.024545378983020782\n",
      "Performance on lower split after step 162: Accuracy=0.9411348771433622, Loss=0.23220492899417877\n",
      "Performance on upper split after step 162: Accuracy=0.9973484178893406, Loss=0.02404714934527874\n",
      "Performance on lower split after step 163: Accuracy=0.9411348771433622, Loss=0.2307453602552414\n",
      "Performance on upper split after step 163: Accuracy=0.99717164574863, Loss=0.02371644787490368\n",
      "Performance on lower split after step 164: Accuracy=0.9418419657062047, Loss=0.22904807329177856\n",
      "Performance on upper split after step 164: Accuracy=0.99717164574863, Loss=0.023268647491931915\n",
      "Performance on lower split after step 165: Accuracy=0.9423722821283366, Loss=0.2272576093673706\n",
      "Performance on upper split after step 165: Accuracy=0.99717164574863, Loss=0.022878019139170647\n",
      "Performance on lower split after step 166: Accuracy=0.9432561428318897, Loss=0.22540007531642914\n",
      "Performance on upper split after step 166: Accuracy=0.9969948736079194, Loss=0.022456659004092216\n",
      "Performance on lower split after step 167: Accuracy=0.9439632313947321, Loss=0.2230975329875946\n",
      "Performance on upper split after step 167: Accuracy=0.9969948736079194, Loss=0.022015616297721863\n",
      "Performance on lower split after step 168: Accuracy=0.9441400035354428, Loss=0.22149793803691864\n",
      "Performance on upper split after step 168: Accuracy=0.9973484178893406, Loss=0.021692311391234398\n",
      "Performance on lower split after step 169: Accuracy=0.9443167756761535, Loss=0.2196042388677597\n",
      "Performance on upper split after step 169: Accuracy=0.9973484178893406, Loss=0.02130434475839138\n",
      "Round: 170, outer loss: 0.12091253697872162\n",
      "Performance on lower split after step 170: Accuracy=0.944493547816864, Loss=0.2177208960056305\n",
      "Performance on upper split after step 170: Accuracy=0.9973484178893406, Loss=0.02098500356078148\n",
      "Performance on lower split after step 171: Accuracy=0.9455541806611278, Loss=0.2160554975271225\n",
      "Performance on upper split after step 171: Accuracy=0.9973484178893406, Loss=0.020613308995962143\n",
      "Performance on lower split after step 172: Accuracy=0.9457309528018384, Loss=0.21416059136390686\n",
      "Performance on upper split after step 172: Accuracy=0.9973484178893406, Loss=0.02026587910950184\n",
      "Performance on lower split after step 173: Accuracy=0.9460844970832597, Loss=0.21239440143108368\n",
      "Performance on upper split after step 173: Accuracy=0.9973484178893406, Loss=0.019882235676050186\n",
      "Performance on lower split after step 174: Accuracy=0.9464380413646809, Loss=0.21089951694011688\n",
      "Performance on upper split after step 174: Accuracy=0.9973484178893406, Loss=0.019593508914113045\n",
      "Performance on lower split after step 175: Accuracy=0.9471451299275234, Loss=0.20938800275325775\n",
      "Performance on upper split after step 175: Accuracy=0.9973484178893406, Loss=0.01925767958164215\n",
      "Performance on lower split after step 176: Accuracy=0.9474986742089446, Loss=0.207475483417511\n",
      "Performance on upper split after step 176: Accuracy=0.9975251900300512, Loss=0.018923319876194\n",
      "Performance on lower split after step 177: Accuracy=0.9480289906310766, Loss=0.20611059665679932\n",
      "Performance on upper split after step 177: Accuracy=0.9975251900300512, Loss=0.018636217340826988\n",
      "Performance on lower split after step 178: Accuracy=0.9482057627717871, Loss=0.20428690314292908\n",
      "Performance on upper split after step 178: Accuracy=0.9977019621707619, Loss=0.01833079755306244\n",
      "Performance on lower split after step 179: Accuracy=0.9490896234753403, Loss=0.2023405134677887\n",
      "Performance on upper split after step 179: Accuracy=0.9980555064521831, Loss=0.01798032969236374\n",
      "Round: 180, outer loss: 0.08204460144042969\n",
      "Performance on lower split after step 180: Accuracy=0.9503270284603147, Loss=0.20060130953788757\n",
      "Performance on upper split after step 180: Accuracy=0.9982322785928938, Loss=0.01769883930683136\n",
      "Performance on lower split after step 181: Accuracy=0.9505038006010252, Loss=0.19907882809638977\n",
      "Performance on upper split after step 181: Accuracy=0.9984090507336044, Loss=0.017415331676602364\n",
      "Performance on lower split after step 182: Accuracy=0.9505038006010252, Loss=0.19727541506290436\n",
      "Performance on upper split after step 182: Accuracy=0.9984090507336044, Loss=0.017140086740255356\n",
      "Performance on lower split after step 183: Accuracy=0.9512108891638678, Loss=0.1958395093679428\n",
      "Performance on upper split after step 183: Accuracy=0.9984090507336044, Loss=0.01686154305934906\n",
      "Performance on lower split after step 184: Accuracy=0.9513876613045784, Loss=0.19417138397693634\n",
      "Performance on upper split after step 184: Accuracy=0.998585822874315, Loss=0.01661728508770466\n",
      "Performance on lower split after step 185: Accuracy=0.9519179777267103, Loss=0.19259661436080933\n",
      "Performance on upper split after step 185: Accuracy=0.998585822874315, Loss=0.016361040994524956\n",
      "Performance on lower split after step 186: Accuracy=0.9517412055859996, Loss=0.19140875339508057\n",
      "Performance on upper split after step 186: Accuracy=0.998585822874315, Loss=0.01617570035159588\n",
      "Performance on lower split after step 187: Accuracy=0.9519179777267103, Loss=0.1900484412908554\n",
      "Performance on upper split after step 187: Accuracy=0.998585822874315, Loss=0.01593535766005516\n",
      "Performance on lower split after step 188: Accuracy=0.9533321548523953, Loss=0.1885439157485962\n",
      "Performance on upper split after step 188: Accuracy=0.998585822874315, Loss=0.015675408765673637\n",
      "Performance on lower split after step 189: Accuracy=0.9542160155559484, Loss=0.1871490627527237\n",
      "Performance on upper split after step 189: Accuracy=0.998585822874315, Loss=0.015453162603080273\n",
      "Round: 190, outer loss: 0.09034475684165955\n",
      "Performance on lower split after step 190: Accuracy=0.9540392434152377, Loss=0.18554700911045074\n",
      "Performance on upper split after step 190: Accuracy=0.998585822874315, Loss=0.015208302065730095\n",
      "Performance on lower split after step 191: Accuracy=0.9550998762595015, Loss=0.18423405289649963\n",
      "Performance on upper split after step 191: Accuracy=0.9984090507336044, Loss=0.014949134550988674\n",
      "Performance on lower split after step 192: Accuracy=0.9549231041187909, Loss=0.18260809779167175\n",
      "Performance on upper split after step 192: Accuracy=0.9984090507336044, Loss=0.0147312106564641\n",
      "Performance on lower split after step 193: Accuracy=0.9552766484002121, Loss=0.18140673637390137\n",
      "Performance on upper split after step 193: Accuracy=0.9984090507336044, Loss=0.014546583406627178\n",
      "Performance on lower split after step 194: Accuracy=0.9554534205409227, Loss=0.1801028698682785\n",
      "Performance on upper split after step 194: Accuracy=0.9984090507336044, Loss=0.014328527264297009\n",
      "Performance on lower split after step 195: Accuracy=0.9561605091037653, Loss=0.17900821566581726\n",
      "Performance on upper split after step 195: Accuracy=0.9984090507336044, Loss=0.014120977371931076\n",
      "Performance on lower split after step 196: Accuracy=0.9563372812444759, Loss=0.17763328552246094\n",
      "Performance on upper split after step 196: Accuracy=0.9984090507336044, Loss=0.013929260894656181\n",
      "Performance on lower split after step 197: Accuracy=0.9566908255258971, Loss=0.17618080973625183\n",
      "Performance on upper split after step 197: Accuracy=0.9984090507336044, Loss=0.013701703399419785\n",
      "Performance on lower split after step 198: Accuracy=0.9575746862294502, Loss=0.1746828854084015\n",
      "Performance on upper split after step 198: Accuracy=0.9984090507336044, Loss=0.01349836029112339\n",
      "Performance on lower split after step 199: Accuracy=0.9581050026515822, Loss=0.17310139536857605\n",
      "Performance on upper split after step 199: Accuracy=0.9984090507336044, Loss=0.013311789371073246\n",
      "Acc lower: 0.9466218843910201. Loss lower: 0.2121976235508919.\n",
      "Acc upper: 0.9977797419126756. Loss upper: 0.02042087148874998.\n"
     ]
    }
   ],
   "source": [
    "lower_iterator = CustomTensorIterator([x_lower, y_lower], batch_size=lower_size, shuffle=True, **kwargs)\n",
    "upper_iterator = CustomTensorIterator([x_upper, y_upper], batch_size=upper_size, shuffle=True, **kwargs)\n",
    "    \n",
    "xs = [hp.clone().detach().requires_grad_(True) for hp in hp0]\n",
    "ys = [p.clone().detach() for p in p0]\n",
    "\n",
    "T = 200\n",
    "N = 2 # 3 is good\n",
    "W = 4\n",
    "alpha = .1  \n",
    "beta = 1. \n",
    "lamda = 0.01 \n",
    "\n",
    "outer_opt = torch.optim.Adam(lr=alpha, params=xs) \n",
    "u = torch.ones(W) # latest coeffs at the end \n",
    "\n",
    "acc, loss = eval_model(ys, x_lower, y_lower)\n",
    "print(f'Performance on lower split before training: Accuracy={acc}, Loss={loss}')\n",
    "acc, loss = eval_model(ys, x_upper, y_upper)\n",
    "print(f'Performance on upper split before training: Accuracy={acc}, Loss={loss}')\n",
    "\n",
    "hg_buffer = []\n",
    "regret = 0. \n",
    "total_time = 0. \n",
    "regret_list = [] \n",
    "running_time = []\n",
    "sum_grad_norms = 0\n",
    "grad_norms_list = []\n",
    "\n",
    "avg_acc_upper = 0. \n",
    "avg_loss_upper = 0. \n",
    "avg_acc_lower = 0. \n",
    "avg_loss_lower = 0.\n",
    "for t in range(T): \n",
    "    step_start_time = time.time() \n",
    "\n",
    "    # get data defining costs f_t and g_t\n",
    "    upper_data = next(upper_iterator)\n",
    "    lower_data = next(lower_iterator)\n",
    "\n",
    "    ft = outer_func(ys, xs, upper_data, lower_data).item()\n",
    "    regret += ft \n",
    "    regret_list.append(regret)\n",
    "\n",
    "    zs = [y.clone().detach() for y in ys] \n",
    "    ys = inner_solver(zs, xs, lower_data, beta, steps=N, lamda=lamda) # obtain y_{t+1} data, s, steps\n",
    "    t1 = time.time() - step_start_time # inner loop time\n",
    "\n",
    "    grads, cost = hg.fixed_point(ys, xs, \n",
    "                                K=N, \n",
    "                                fp_map=lambda y, x: map_func(y, x, lower_data, beta, lamda), \n",
    "                                outer_loss=lambda y, x: outer_func(y, x, upper_data, lower_data), \n",
    "                                set_grad=False)\n",
    "\n",
    "    hg_buffer.append(grads)\n",
    "    del hg_buffer[:-W]\n",
    "    TAH = getTimeAveHypergrad(hg_buffer, u)\n",
    "    t2 = time.time() - step_start_time - t1 # time averaged hypergrad estimation time \n",
    "\n",
    "    norm = torch.norm(TAH[0]).item() \n",
    "    sum_grad_norms += norm \n",
    "    grad_norms_list.append(sum_grad_norms) \n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    update_tensor_grads(xs, TAH)\n",
    "    outer_opt.step()\n",
    "\n",
    "    step_time = time.time()-step_start_time\n",
    "    total_time +=step_time \n",
    "    running_time.append(total_time) \n",
    "\n",
    "\n",
    "    if t%10 == 0:\n",
    "        print(f'Round: {t}, outer loss: {ft}')\n",
    "\n",
    "    if t>= T - 50:\n",
    "        acc, loss = eval_model(ys, x_lower, y_lower)\n",
    "        print(f'Performance on lower split after step {t}: Accuracy={acc}, Loss={loss}')\n",
    "        avg_acc_lower += acc\n",
    "        avg_loss_lower += loss \n",
    "        \n",
    "        acc, loss = eval_model(ys, x_upper, y_upper)\n",
    "        print(f'Performance on upper split after step {t}: Accuracy={acc}, Loss={loss}')\n",
    "        avg_acc_upper += acc\n",
    "        avg_loss_upper += loss \n",
    "\n",
    "print(f'Acc lower: {avg_acc_lower/50}. Loss lower: {avg_loss_lower/50}.') \n",
    "print(f'Acc upper: {avg_acc_upper/50}. Loss upper: {avg_loss_upper/50}.')\n",
    "\n",
    "norms_ours4 = grad_norms_list\n",
    "regret_ours4 = regret_list \n",
    "rt_ours4 = running_time "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# OAGD"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Performance on lower split before training: Accuracy=0.04189499734841789, Loss=3.512615442276001\n",
      "Performance on upper split before training: Accuracy=0.03676860526780979, Loss=3.53226375579834\n",
      "Round: 0, outer loss: 3.4915809631347656\n",
      "Round: 10, outer loss: 2.814603567123413\n",
      "Round: 20, outer loss: 2.326474905014038\n",
      "Round: 30, outer loss: 1.8932240009307861\n",
      "Round: 40, outer loss: 1.3231102228164673\n",
      "Round: 50, outer loss: 1.0652058124542236\n",
      "Round: 60, outer loss: 1.1154513359069824\n",
      "Round: 70, outer loss: 0.9128819704055786\n",
      "Round: 80, outer loss: 0.7152917981147766\n",
      "Round: 90, outer loss: 0.5827782154083252\n",
      "Round: 100, outer loss: 0.4621267020702362\n",
      "Round: 110, outer loss: 0.8796855211257935\n",
      "Round: 120, outer loss: 0.10880033671855927\n",
      "Round: 130, outer loss: 0.17715805768966675\n",
      "Round: 140, outer loss: 0.07310432940721512\n",
      "Round: 150, outer loss: 0.09956064820289612\n",
      "Round: 160, outer loss: 0.16876201331615448\n",
      "Round: 170, outer loss: 0.04610592871904373\n",
      "Round: 180, outer loss: 0.13231058418750763\n",
      "Round: 190, outer loss: 0.19299958646297455\n",
      "Round: 200, outer loss: 0.052766334265470505\n",
      "Round: 210, outer loss: 0.08412772417068481\n",
      "Round: 220, outer loss: 0.1066710576415062\n",
      "Round: 230, outer loss: 0.03405176103115082\n",
      "Round: 240, outer loss: 0.05610708147287369\n",
      "Round: 250, outer loss: 0.04109315201640129\n",
      "Performance on lower split after step 250: Accuracy=0.9876259501502563, Loss=0.08007244765758514\n",
      "Performance on upper split after step 250: Accuracy=0.9998232278592893, Loss=0.005772873759269714\n",
      "Performance on lower split after step 251: Accuracy=0.987802722290967, Loss=0.07944166660308838\n",
      "Performance on upper split after step 251: Accuracy=0.9996464557185788, Loss=0.005736614111810923\n",
      "Performance on lower split after step 252: Accuracy=0.9881562665723882, Loss=0.07883480936288834\n",
      "Performance on upper split after step 252: Accuracy=0.9996464557185788, Loss=0.005696241743862629\n",
      "Performance on lower split after step 253: Accuracy=0.9881562665723882, Loss=0.07823261618614197\n",
      "Performance on upper split after step 253: Accuracy=0.9996464557185788, Loss=0.005679092835634947\n",
      "Performance on lower split after step 254: Accuracy=0.9885098108538094, Loss=0.07766889035701752\n",
      "Performance on upper split after step 254: Accuracy=0.9996464557185788, Loss=0.005657324101775885\n",
      "Performance on lower split after step 255: Accuracy=0.9885098108538094, Loss=0.07700178027153015\n",
      "Performance on upper split after step 255: Accuracy=0.9996464557185788, Loss=0.0056143393740057945\n",
      "Performance on lower split after step 256: Accuracy=0.9886865829945201, Loss=0.07654677331447601\n",
      "Performance on upper split after step 256: Accuracy=0.9996464557185788, Loss=0.005600202362984419\n",
      "Performance on lower split after step 257: Accuracy=0.9888633551352307, Loss=0.07581791281700134\n",
      "Performance on upper split after step 257: Accuracy=0.9996464557185788, Loss=0.005553499795496464\n",
      "Performance on lower split after step 258: Accuracy=0.9888633551352307, Loss=0.07532212138175964\n",
      "Performance on upper split after step 258: Accuracy=0.9996464557185788, Loss=0.005523392464965582\n",
      "Performance on lower split after step 259: Accuracy=0.9890401272759413, Loss=0.0747501403093338\n",
      "Performance on upper split after step 259: Accuracy=0.9996464557185788, Loss=0.005495557561516762\n",
      "Round: 260, outer loss: 0.03449031710624695\n",
      "Performance on lower split after step 260: Accuracy=0.9892168994166519, Loss=0.07412683218717575\n",
      "Performance on upper split after step 260: Accuracy=0.9996464557185788, Loss=0.005469502881169319\n",
      "Performance on lower split after step 261: Accuracy=0.9893936715573626, Loss=0.07346859574317932\n",
      "Performance on upper split after step 261: Accuracy=0.9996464557185788, Loss=0.005448658484965563\n",
      "Performance on lower split after step 262: Accuracy=0.9892168994166519, Loss=0.07301115244626999\n",
      "Performance on upper split after step 262: Accuracy=0.9996464557185788, Loss=0.005437082145363092\n",
      "Performance on lower split after step 263: Accuracy=0.9892168994166519, Loss=0.0724826380610466\n",
      "Performance on upper split after step 263: Accuracy=0.9996464557185788, Loss=0.005427120719105005\n",
      "Performance on lower split after step 264: Accuracy=0.9895704436980732, Loss=0.07184871286153793\n",
      "Performance on upper split after step 264: Accuracy=0.9996464557185788, Loss=0.005412878934293985\n",
      "Performance on lower split after step 265: Accuracy=0.9893936715573626, Loss=0.07131869345903397\n",
      "Performance on upper split after step 265: Accuracy=0.9996464557185788, Loss=0.00540367653593421\n",
      "Performance on lower split after step 266: Accuracy=0.9895704436980732, Loss=0.07086788862943649\n",
      "Performance on upper split after step 266: Accuracy=0.9996464557185788, Loss=0.005403989460319281\n",
      "Performance on lower split after step 267: Accuracy=0.9895704436980732, Loss=0.07043468952178955\n",
      "Performance on upper split after step 267: Accuracy=0.9996464557185788, Loss=0.005400428082793951\n",
      "Performance on lower split after step 268: Accuracy=0.9895704436980732, Loss=0.07002510875463486\n",
      "Performance on upper split after step 268: Accuracy=0.9996464557185788, Loss=0.005408048164099455\n",
      "Performance on lower split after step 269: Accuracy=0.9895704436980732, Loss=0.06958965957164764\n",
      "Performance on upper split after step 269: Accuracy=0.9996464557185788, Loss=0.005407247226685286\n",
      "Round: 270, outer loss: 0.044190507382154465\n",
      "Performance on lower split after step 270: Accuracy=0.9895704436980732, Loss=0.06920300424098969\n",
      "Performance on upper split after step 270: Accuracy=0.9994696835778681, Loss=0.005416416097432375\n",
      "Performance on lower split after step 271: Accuracy=0.9897472158387838, Loss=0.06878037005662918\n",
      "Performance on upper split after step 271: Accuracy=0.9994696835778681, Loss=0.005425603594630957\n",
      "Performance on lower split after step 272: Accuracy=0.9899239879794944, Loss=0.068321093916893\n",
      "Performance on upper split after step 272: Accuracy=0.9994696835778681, Loss=0.005426590796560049\n",
      "Performance on lower split after step 273: Accuracy=0.9901007601202051, Loss=0.0679149329662323\n",
      "Performance on upper split after step 273: Accuracy=0.9994696835778681, Loss=0.005436996929347515\n",
      "Performance on lower split after step 274: Accuracy=0.9899239879794944, Loss=0.06755106896162033\n",
      "Performance on upper split after step 274: Accuracy=0.9994696835778681, Loss=0.0054399315267801285\n",
      "Performance on lower split after step 275: Accuracy=0.9901007601202051, Loss=0.06709674000740051\n",
      "Performance on upper split after step 275: Accuracy=0.9994696835778681, Loss=0.005455453880131245\n",
      "Performance on lower split after step 276: Accuracy=0.9901007601202051, Loss=0.06672562658786774\n",
      "Performance on upper split after step 276: Accuracy=0.9994696835778681, Loss=0.005466004833579063\n",
      "Performance on lower split after step 277: Accuracy=0.9899239879794944, Loss=0.06624680012464523\n",
      "Performance on upper split after step 277: Accuracy=0.9994696835778681, Loss=0.00545278238132596\n",
      "Performance on lower split after step 278: Accuracy=0.9899239879794944, Loss=0.06582020968198776\n",
      "Performance on upper split after step 278: Accuracy=0.9994696835778681, Loss=0.00544839259237051\n",
      "Performance on lower split after step 279: Accuracy=0.9901007601202051, Loss=0.06545844674110413\n",
      "Performance on upper split after step 279: Accuracy=0.9994696835778681, Loss=0.005455222446471453\n",
      "Round: 280, outer loss: 0.02802991308271885\n",
      "Performance on lower split after step 280: Accuracy=0.9901007601202051, Loss=0.06505146622657776\n",
      "Performance on upper split after step 280: Accuracy=0.9994696835778681, Loss=0.005453301593661308\n",
      "Performance on lower split after step 281: Accuracy=0.9901007601202051, Loss=0.06474370509386063\n",
      "Performance on upper split after step 281: Accuracy=0.9994696835778681, Loss=0.005464290734380484\n",
      "Performance on lower split after step 282: Accuracy=0.9902775322609156, Loss=0.06437427550554276\n",
      "Performance on upper split after step 282: Accuracy=0.9994696835778681, Loss=0.005467760842293501\n",
      "Performance on lower split after step 283: Accuracy=0.9901007601202051, Loss=0.06403020024299622\n",
      "Performance on upper split after step 283: Accuracy=0.9994696835778681, Loss=0.005474510136991739\n",
      "Performance on lower split after step 284: Accuracy=0.9904543044016263, Loss=0.06368475407361984\n",
      "Performance on upper split after step 284: Accuracy=0.9994696835778681, Loss=0.005470657255500555\n",
      "Performance on lower split after step 285: Accuracy=0.990631076542337, Loss=0.06349492073059082\n",
      "Performance on upper split after step 285: Accuracy=0.9994696835778681, Loss=0.0055025857873260975\n",
      "Performance on lower split after step 286: Accuracy=0.9904543044016263, Loss=0.06315845251083374\n",
      "Performance on upper split after step 286: Accuracy=0.9994696835778681, Loss=0.005503115709871054\n",
      "Performance on lower split after step 287: Accuracy=0.9902775322609156, Loss=0.06278323382139206\n",
      "Performance on upper split after step 287: Accuracy=0.9994696835778681, Loss=0.005497151054441929\n",
      "Performance on lower split after step 288: Accuracy=0.9901007601202051, Loss=0.0625113844871521\n",
      "Performance on upper split after step 288: Accuracy=0.9994696835778681, Loss=0.005503467749804258\n",
      "Performance on lower split after step 289: Accuracy=0.9901007601202051, Loss=0.0622333362698555\n",
      "Performance on upper split after step 289: Accuracy=0.9994696835778681, Loss=0.00550569873303175\n",
      "Round: 290, outer loss: 0.028405020013451576\n",
      "Performance on lower split after step 290: Accuracy=0.9902775322609156, Loss=0.06192103400826454\n",
      "Performance on upper split after step 290: Accuracy=0.9994696835778681, Loss=0.005503315944224596\n",
      "Performance on lower split after step 291: Accuracy=0.9904543044016263, Loss=0.061543915420770645\n",
      "Performance on upper split after step 291: Accuracy=0.9994696835778681, Loss=0.0055080135352909565\n",
      "Performance on lower split after step 292: Accuracy=0.9908078486830475, Loss=0.06113462895154953\n",
      "Performance on upper split after step 292: Accuracy=0.9994696835778681, Loss=0.005517263896763325\n",
      "Performance on lower split after step 293: Accuracy=0.9908078486830475, Loss=0.06083274260163307\n",
      "Performance on upper split after step 293: Accuracy=0.9994696835778681, Loss=0.0055188206024467945\n",
      "Performance on lower split after step 294: Accuracy=0.9908078486830475, Loss=0.0605105459690094\n",
      "Performance on upper split after step 294: Accuracy=0.9994696835778681, Loss=0.005534119438380003\n",
      "Performance on lower split after step 295: Accuracy=0.9909846208237582, Loss=0.06021643429994583\n",
      "Performance on upper split after step 295: Accuracy=0.9994696835778681, Loss=0.005520728416740894\n",
      "Performance on lower split after step 296: Accuracy=0.9909846208237582, Loss=0.059920381754636765\n",
      "Performance on upper split after step 296: Accuracy=0.9994696835778681, Loss=0.005518226884305477\n",
      "Performance on lower split after step 297: Accuracy=0.9909846208237582, Loss=0.05964989960193634\n",
      "Performance on upper split after step 297: Accuracy=0.9994696835778681, Loss=0.005523134954273701\n",
      "Performance on lower split after step 298: Accuracy=0.9911613929644688, Loss=0.059370994567871094\n",
      "Performance on upper split after step 298: Accuracy=0.9994696835778681, Loss=0.00553516298532486\n",
      "Performance on lower split after step 299: Accuracy=0.9911613929644688, Loss=0.05905463546514511\n",
      "Performance on upper split after step 299: Accuracy=0.9994696835778681, Loss=0.005530982743948698\n",
      "Acc lower: 0.9897790348241123. Loss lower: 0.06808404728770256.\n",
      "Acc upper: 0.9995439278769677. Loss upper: 0.00550046949647367.\n"
     ]
    }
   ],
   "source": [
    "lower_iterator = CustomTensorIterator([x_lower, y_lower], batch_size=lower_size, shuffle=True, **kwargs)\n",
    "upper_iterator = CustomTensorIterator([x_upper, y_upper], batch_size=upper_size, shuffle=True, **kwargs)\n",
    "\n",
    "xs = [hp.clone().detach().requires_grad_(True) for hp in hp0]\n",
    "ys = [p.clone().detach() for p in p0]\n",
    "\n",
    "T = 300 \n",
    "N = 3\n",
    "W = 4\n",
    "alpha = .1 \n",
    "beta = 1. \n",
    "lamda = 0.01 \n",
    "\n",
    "outer_opt = torch.optim.Adam(lr=alpha, params=xs) \n",
    "buffer = []\n",
    "u = torch.ones(W) # latest coeffs at the end \n",
    "\n",
    "regret = 0. \n",
    "total_time = 0. \n",
    "regret_list = [] \n",
    "running_time = [] \n",
    "sum_grad_norms = 0\n",
    "grad_norms_list = []\n",
    "\n",
    "acc, loss = eval_model(ys, x_lower, y_lower)\n",
    "print(f'Performance on lower split before training: Accuracy={acc}, Loss={loss}')\n",
    "acc, loss = eval_model(ys, x_upper, y_upper)\n",
    "print(f'Performance on upper split before training: Accuracy={acc}, Loss={loss}')\n",
    "\n",
    "avg_acc_upper = 0. \n",
    "avg_loss_upper = 0. \n",
    "avg_acc_lower = 0. \n",
    "avg_loss_lower = 0. \n",
    "for t in range(T): \n",
    "    step_start_time = time.time() \n",
    "    \n",
    "    # get data defining costs f_t and g_t\n",
    "    upper_data = next(upper_iterator)\n",
    "    lower_data = next(lower_iterator)\n",
    "    buffer.append((upper_data, lower_data))\n",
    "    del buffer[:-W]\n",
    "    \n",
    "    ft = outer_func(ys, xs, upper_data, lower_data).item()\n",
    "    regret += ft \n",
    "    regret_list.append(regret)\n",
    "    \n",
    "    zs = [y.clone().detach() for y in ys] \n",
    "    ys = inner_solver(zs, xs, lower_data, beta, steps=N, lamda=lamda) # obtain y_{t+1}\n",
    "    t1 = time.time() - step_start_time # inner loop time\n",
    "    \n",
    "    #outer_opt.zero_grad()\n",
    "    TAH = [torch.zeros_like(hp) for hp in hp0]\n",
    "    grads_list = []\n",
    "    for i, (f_data, g_data) in enumerate(buffer): # buffer[-W:]\n",
    "        grads, cost = hg.fixed_point(ys, xs, \n",
    "                            K=N, \n",
    "                            fp_map=lambda y, x: map_func(y, x, g_data, beta, lamda),\n",
    "                            outer_loss=lambda y, x: outer_func(y, x, f_data, g_data), \n",
    "                            set_grad=False)\n",
    "        grads_list.append(grads)\n",
    "        with torch.no_grad():\n",
    "            for av, g in zip(TAH, grads):\n",
    "                av.copy_(av + u[i]*g)\n",
    "                            \n",
    "    TAH = [av / torch.sum(u[-len(buffer):]) for av in TAH]\n",
    "    t2 = time.time() - step_start_time - t1 # time averaged hypergrad estimation time \n",
    "    \n",
    "    norm = torch.norm(TAH[0]).item() \n",
    "    sum_grad_norms += norm \n",
    "    grad_norms_list.append(sum_grad_norms) \n",
    "    \n",
    "    outer_opt.zero_grad() \n",
    "    update_tensor_grads(xs, TAH)\n",
    "    outer_opt.step()\n",
    "    \n",
    "    step_time = time.time()-step_start_time\n",
    "    total_time +=step_time \n",
    "    running_time.append(total_time)\n",
    "    \n",
    "    #print(f'Round: {t}, len buffer: {len(buffer)}, outer loss: {ft}')\n",
    "    if t%10 == 0:\n",
    "        print(f'Round: {t}, outer loss: {ft}')\n",
    "\n",
    "    if t>= T - 50:\n",
    "        acc, loss = eval_model(ys, x_lower, y_lower)\n",
    "        print(f'Performance on lower split after step {t}: Accuracy={acc}, Loss={loss}')\n",
    "        avg_acc_lower += acc\n",
    "        avg_loss_lower += loss \n",
    "        \n",
    "        acc, loss = eval_model(ys, x_upper, y_upper)\n",
    "        print(f'Performance on upper split after step {t}: Accuracy={acc}, Loss={loss}')\n",
    "        avg_acc_upper += acc\n",
    "        avg_loss_upper += loss \n",
    "\n",
    "print(f'Acc lower: {avg_acc_lower/50}. Loss lower: {avg_loss_lower/50}.') \n",
    "print(f'Acc upper: {avg_acc_upper/50}. Loss upper: {avg_loss_upper/50}.')\n",
    "\n",
    "norms_oagd4 = grad_norms_list   \n",
    "regret_oagd4 = regret_list \n",
    "rt_oagd4 = running_time "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 864x504 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "sz=35\n",
    "plt.plot(regret_ours4, '-', label='SOBOW', linewidth=3, color='tab:blue') \n",
    "plt.plot(regret_oagd4, '-', label='OAGD', linewidth=3, color=\"firebrick\") \n",
    "\n",
    "# plt.yscale('log')\n",
    "# plt.xscale('log')\n",
    "\n",
    "plt.rc('font', size=sz)\n",
    "plt.ticklabel_format(style='sci', axis='both', scilimits=(0,0), useMathText=True)\n",
    "plt.xlabel('Rounds T', fontsize=sz) \n",
    "plt.ylabel('Regret', fontsize=sz) \n",
    "plt.xticks(fontsize=sz)\n",
    "plt.yticks(fontsize=sz)\n",
    "#plt.xlim(0,3000)\n",
    "#plt.ylim(0,2000000)\n",
    "plt.grid()\n",
    "plt.legend(loc='upper left' , fontsize=20) \n",
    "plt.gcf().set_size_inches(12, 7)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.6.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
