{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "from itertools import repeat\n",
    "import random \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\n",
    "\n",
    "b = 0.001\n",
    "def inner_func(ys, xs, data):\n",
    "    H = xs[0]\n",
    "    w = ys[0]\n",
    "    X, y = data  \n",
    "    \n",
    "    g =  0.5 * (torch.norm(X @ H @ w - y))**2 + 0.5 * b * torch.norm(w)**2 \n",
    "    return g #.squeeze()\n",
    "\n",
    "def outer_func(ys, xs, data):\n",
    "    \n",
    "    H = xs[0]\n",
    "    w = ys[0]\n",
    "    X, y = data\n",
    "    \n",
    "    f = 0.5 * (torch.norm(X @ H @ w - y))**2 \n",
    "    \n",
    "    return f \n",
    "\n",
    "def map_func(ys, xs, data, s): \n",
    "    \n",
    "    g = inner_func(ys, xs, data)\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): \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)\n",
    "        \n",
    "    return [y.clone().detach() for y in ys]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "def getTimeAveHypergrad(grads_list, u):\n",
    "    \n",
    "    W = len(grads_list)\n",
    "    weights = list(u[-W:])\n",
    "    TAH = [torch.zeros_like(p) for p in grads_list[0]]\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",
    "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",
    "kwargs = {} \n",
    "\n",
    "def update_tensor_grads(hparams, grads):\n",
    "    for l, g in zip(hparams, grads):\n",
    "        if l.grad is None:\n",
    "            l.grad = torch.zeros_like(l)\n",
    "        if g is not None:\n",
    "            l.grad += g "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "shape of Xf: torch.Size([3000, 64])\n",
      "shape of yf: torch.Size([3000])\n"
     ]
    }
   ],
   "source": [
    "# p: data dim\n",
    "# d: representation dim\n",
    "# n: number of data points \n",
    "p, d, n = 64, 128, 3000\n",
    "m = 0.1 \n",
    "s = 3 # number of comparators\n",
    "SC = n//s # samples per comparator \n",
    "\n",
    "Xg = torch.randn((n, p))\n",
    "yg = torch.zeros(n)\n",
    "Xf = torch.randn((n, p))\n",
    "yf = torch.zeros(n)\n",
    "\n",
    "# here the dynamic comparator abrutly changes after SC steps \n",
    "comparators = []\n",
    "for i in range(s):\n",
    "    H_true = torch.randn((p, d))\n",
    "    w_true = torch.randn(d)\n",
    "    comparators.append((H_true, w_true))\n",
    "    yf[i*SC:(i+1)*SC] = Xf[i*SC:(i+1)*SC,:] @ H_true @ w_true + m * torch.randn(SC) \n",
    "    yg [i*SC:(i+1)*SC] = Xg[i*SC:(i+1)*SC,:] @ H_true @ w_true + m * torch.randn(SC) \n",
    "\n",
    "\n",
    "print(f\"shape of Xf: {Xf.shape}\")\n",
    "print(f\"shape of yf: {yf.shape}\")\n",
    "\n",
    "# anoter random model \n",
    "Hr = torch.randn((p, d))\n",
    "wr = torch.randn(d)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# ours"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Round: 0, outer loss: 3671.49292, inner loop time: 0.00135, TAH time: 0.00226, step time: 0.00411\n",
      "Round: 10, outer loss: 1612.47559, inner loop time: 0.00068, TAH time: 0.00221, step time: 0.00320\n",
      "Round: 20, outer loss: 466.70203, inner loop time: 0.00040, TAH time: 0.00153, step time: 0.00209\n",
      "Round: 30, outer loss: 131.43460, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 40, outer loss: 2260.95044, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 50, outer loss: 58.15809, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00174\n",
      "Round: 60, outer loss: 2149.68799, inner loop time: 0.00032, TAH time: 0.00133, step time: 0.00185\n",
      "Round: 70, outer loss: 10.66097, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 80, outer loss: 164.61685, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 90, outer loss: 147.50554, inner loop time: 0.00032, TAH time: 0.00133, step time: 0.00179\n",
      "Round: 100, outer loss: 198.51717, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00175\n",
      "Round: 110, outer loss: 425.76459, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00179\n",
      "Round: 120, outer loss: 622.51117, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00175\n",
      "Round: 130, outer loss: 730.17957, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 140, outer loss: 996.96869, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00177\n",
      "Round: 150, outer loss: 87.18211, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 160, outer loss: 817.81567, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 170, outer loss: 900.84778, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 180, outer loss: 1475.58032, inner loop time: 0.00032, TAH time: 0.00134, step time: 0.00180\n",
      "Round: 190, outer loss: 376.61871, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 200, outer loss: 6.08168, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00179\n",
      "Round: 210, outer loss: 127.35378, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00177\n",
      "Round: 220, outer loss: 409.53436, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 230, outer loss: 15.19113, inner loop time: 0.00033, TAH time: 0.00133, step time: 0.00181\n",
      "Round: 240, outer loss: 312.13266, inner loop time: 0.00034, TAH time: 0.00132, step time: 0.00180\n",
      "Round: 250, outer loss: 900.80554, inner loop time: 0.00031, TAH time: 0.00133, step time: 0.00179\n",
      "Round: 260, outer loss: 50.51894, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 270, outer loss: 84.75191, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 280, outer loss: 48.53439, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 290, outer loss: 740.47937, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00175\n",
      "Round: 300, outer loss: 37.79375, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 310, outer loss: 354.53683, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 320, outer loss: 358.59958, inner loop time: 0.00031, TAH time: 0.00129, step time: 0.00174\n",
      "Round: 330, outer loss: 0.21044, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 340, outer loss: 13.41729, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 350, outer loss: 411.78821, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 360, outer loss: 7.27770, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 370, outer loss: 119.69074, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00175\n",
      "Round: 380, outer loss: 55.38490, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 390, outer loss: 46.04538, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 400, outer loss: 25.85685, inner loop time: 0.00032, TAH time: 0.00134, step time: 0.00180\n",
      "Round: 410, outer loss: 169.19209, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 420, outer loss: 196.17931, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 430, outer loss: 76.37766, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 440, outer loss: 28.67743, inner loop time: 0.00032, TAH time: 0.00133, step time: 0.00178\n",
      "Round: 450, outer loss: 132.45705, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 460, outer loss: 8.95805, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 470, outer loss: 3.88099, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 480, outer loss: 3.06429, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 490, outer loss: 85.88490, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00175\n",
      "Round: 500, outer loss: 55.73804, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 510, outer loss: 16.54520, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 520, outer loss: 18.84038, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 530, outer loss: 8.18931, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 540, outer loss: 4.41392, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 550, outer loss: 76.34698, inner loop time: 0.00032, TAH time: 0.00133, step time: 0.00179\n",
      "Round: 560, outer loss: 45.04034, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 570, outer loss: 15.85681, inner loop time: 0.00033, TAH time: 0.00131, step time: 0.00178\n",
      "Round: 580, outer loss: 0.80490, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 590, outer loss: 2.35389, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 600, outer loss: 14.15017, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 610, outer loss: 19.86007, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 620, outer loss: 16.81109, inner loop time: 0.00032, TAH time: 0.00130, step time: 0.00176\n",
      "Round: 630, outer loss: 3.89622, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 640, outer loss: 2.90206, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00178\n",
      "Round: 650, outer loss: 5.66926, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 660, outer loss: 0.40807, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00175\n",
      "Round: 670, outer loss: 50.68674, inner loop time: 0.00032, TAH time: 0.00130, step time: 0.00176\n",
      "Round: 680, outer loss: 16.89513, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 690, outer loss: 30.66447, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 700, outer loss: 1.12686, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00175\n",
      "Round: 710, outer loss: 49.36619, inner loop time: 0.00034, TAH time: 0.00142, step time: 0.00191\n",
      "Round: 720, outer loss: 0.00460, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 730, outer loss: 2.64543, inner loop time: 0.00031, TAH time: 0.00129, step time: 0.00175\n",
      "Round: 740, outer loss: 6.68177, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 750, outer loss: 1.17443, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 760, outer loss: 9.43694, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 770, outer loss: 0.08673, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00177\n",
      "Round: 780, outer loss: 9.60622, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 790, outer loss: 10.96077, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00177\n",
      "Round: 800, outer loss: 0.06883, inner loop time: 0.00032, TAH time: 0.00130, step time: 0.00176\n",
      "Round: 810, outer loss: 1.36500, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 820, outer loss: 2.62643, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 830, outer loss: 5.27052, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 840, outer loss: 0.44793, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 850, outer loss: 2.92707, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00178\n",
      "Round: 860, outer loss: 1.49885, inner loop time: 0.00033, TAH time: 0.00133, step time: 0.00180\n",
      "Round: 870, outer loss: 2.93351, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 880, outer loss: 0.03755, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 890, outer loss: 1.10261, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 900, outer loss: 1.91303, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 910, outer loss: 6.53800, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 920, outer loss: 0.52817, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 930, outer loss: 0.69336, inner loop time: 0.00032, TAH time: 0.00130, step time: 0.00176\n",
      "Round: 940, outer loss: 3.24794, inner loop time: 0.00031, TAH time: 0.00133, step time: 0.00178\n",
      "Round: 950, outer loss: 1.12672, inner loop time: 0.00033, TAH time: 0.00131, step time: 0.00178\n",
      "Round: 960, outer loss: 2.33862, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 970, outer loss: 0.08346, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 980, outer loss: 5.54897, inner loop time: 0.00032, TAH time: 0.00133, step time: 0.00179\n",
      "Round: 990, outer loss: 0.04160, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1000, outer loss: 822.08948, inner loop time: 0.00032, TAH time: 0.00130, step time: 0.00175\n",
      "Round: 1010, outer loss: 776.49274, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 1020, outer loss: 9.50733, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1030, outer loss: 12829.06152, inner loop time: 0.00032, TAH time: 0.00130, step time: 0.00177\n",
      "Round: 1040, outer loss: 26.86646, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 1050, outer loss: 3620.04858, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1060, outer loss: 1001.30261, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1070, outer loss: 6.90375, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00178\n",
      "Round: 1080, outer loss: 5893.73486, inner loop time: 0.00032, TAH time: 0.00130, step time: 0.00176\n",
      "Round: 1090, outer loss: 15587.37891, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1100, outer loss: 1251.05725, inner loop time: 0.00032, TAH time: 0.00130, step time: 0.00176\n",
      "Round: 1110, outer loss: 48.99973, inner loop time: 0.00031, TAH time: 0.00129, step time: 0.00174\n",
      "Round: 1120, outer loss: 336.51953, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1130, outer loss: 455.99652, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1140, outer loss: 490.09204, inner loop time: 0.00031, TAH time: 0.00129, step time: 0.00174\n",
      "Round: 1150, outer loss: 11.66660, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 1160, outer loss: 1128.26270, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00177\n",
      "Round: 1170, outer loss: 6302.23389, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1180, outer loss: 1536.09070, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 1190, outer loss: 10.87018, inner loop time: 0.00037, TAH time: 0.00135, step time: 0.00186\n",
      "Round: 1200, outer loss: 0.86774, inner loop time: 0.00031, TAH time: 0.00132, step time: 0.00177\n",
      "Round: 1210, outer loss: 27.99012, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00176\n",
      "Round: 1220, outer loss: 48.57186, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00177\n",
      "Round: 1230, outer loss: 252.53174, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1240, outer loss: 666.00635, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1250, outer loss: 2384.66162, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 1260, outer loss: 422.09274, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 1270, outer loss: 123.74290, inner loop time: 0.00032, TAH time: 0.00130, step time: 0.00177\n",
      "Round: 1280, outer loss: 298.86139, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1290, outer loss: 359.48596, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1300, outer loss: 23.94556, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 1310, outer loss: 67.96304, inner loop time: 0.00040, TAH time: 0.00142, step time: 0.00199\n",
      "Round: 1320, outer loss: 16.83245, inner loop time: 0.00032, TAH time: 0.00133, step time: 0.00179\n",
      "Round: 1330, outer loss: 8.07336, inner loop time: 0.00032, TAH time: 0.00135, step time: 0.00183\n",
      "Round: 1340, outer loss: 429.84753, inner loop time: 0.00032, TAH time: 0.00133, step time: 0.00178\n",
      "Round: 1350, outer loss: 10.25433, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1360, outer loss: 24.58750, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00175\n",
      "Round: 1370, outer loss: 3.62633, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00178\n",
      "Round: 1380, outer loss: 43.53087, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1390, outer loss: 187.21846, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1400, outer loss: 190.77599, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1410, outer loss: 58.41533, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1420, outer loss: 2.89717, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1430, outer loss: 256.58301, inner loop time: 0.00034, TAH time: 0.00133, step time: 0.00181\n",
      "Round: 1440, outer loss: 286.22101, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 1450, outer loss: 5.15903, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1460, outer loss: 140.11459, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00176\n",
      "Round: 1470, outer loss: 243.90779, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1480, outer loss: 0.21386, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00178\n",
      "Round: 1490, outer loss: 24.85579, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1500, outer loss: 4.63362, inner loop time: 0.00032, TAH time: 0.00133, step time: 0.00179\n",
      "Round: 1510, outer loss: 181.67941, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1520, outer loss: 11.54002, inner loop time: 0.00032, TAH time: 0.00139, step time: 0.00185\n",
      "Round: 1530, outer loss: 6.86752, inner loop time: 0.00031, TAH time: 0.00129, step time: 0.00174\n",
      "Round: 1540, outer loss: 0.46378, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00176\n",
      "Round: 1550, outer loss: 84.92543, inner loop time: 0.00033, TAH time: 0.00131, step time: 0.00178\n",
      "Round: 1560, outer loss: 9.69324, inner loop time: 0.00033, TAH time: 0.00132, step time: 0.00179\n",
      "Round: 1570, outer loss: 6.71673, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1580, outer loss: 21.40023, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 1590, outer loss: 14.84200, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 1600, outer loss: 6.94588, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1610, outer loss: 12.59409, inner loop time: 0.00032, TAH time: 0.00130, step time: 0.00177\n",
      "Round: 1620, outer loss: 0.25788, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 1630, outer loss: 55.12532, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 1640, outer loss: 7.44924, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00177\n",
      "Round: 1650, outer loss: 71.18320, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1660, outer loss: 0.73851, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1670, outer loss: 66.22941, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1680, outer loss: 12.84092, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00178\n",
      "Round: 1690, outer loss: 70.07053, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1700, outer loss: 2.19260, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1710, outer loss: 0.67234, inner loop time: 0.00032, TAH time: 0.00130, step time: 0.00176\n",
      "Round: 1720, outer loss: 6.74509, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 1730, outer loss: 2.58044, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 1740, outer loss: 1.10236, inner loop time: 0.00032, TAH time: 0.00130, step time: 0.00176\n",
      "Round: 1750, outer loss: 3.32888, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 1760, outer loss: 0.17344, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 1770, outer loss: 7.31694, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00175\n",
      "Round: 1780, outer loss: 0.00500, inner loop time: 0.00032, TAH time: 0.00130, step time: 0.00175\n",
      "Round: 1790, outer loss: 0.00410, inner loop time: 0.00036, TAH time: 0.00145, step time: 0.00198\n",
      "Round: 1800, outer loss: 18.35959, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00175\n",
      "Round: 1810, outer loss: 0.86711, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00175\n",
      "Round: 1820, outer loss: 1.31979, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 1830, outer loss: 0.22259, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1840, outer loss: 0.16513, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1850, outer loss: 0.76716, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1860, outer loss: 0.05465, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 1870, outer loss: 0.10707, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 1880, outer loss: 0.07003, inner loop time: 0.00034, TAH time: 0.00131, step time: 0.00179\n",
      "Round: 1890, outer loss: 0.02762, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1900, outer loss: 0.89224, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1910, outer loss: 0.26285, inner loop time: 0.00037, TAH time: 0.00134, step time: 0.00187\n",
      "Round: 1920, outer loss: 3.40310, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 1930, outer loss: 0.75526, inner loop time: 0.00033, TAH time: 0.00131, step time: 0.00178\n",
      "Round: 1940, outer loss: 0.00062, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1950, outer loss: 0.08612, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1960, outer loss: 0.08263, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00176\n",
      "Round: 1970, outer loss: 0.50663, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1980, outer loss: 0.42408, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 1990, outer loss: 0.00406, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2000, outer loss: 1033.50281, inner loop time: 0.00032, TAH time: 0.00129, step time: 0.00174\n",
      "Round: 2010, outer loss: 69.71739, inner loop time: 0.00032, TAH time: 0.00130, step time: 0.00176\n",
      "Round: 2020, outer loss: 12914.50781, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2030, outer loss: 2051.25562, inner loop time: 0.00034, TAH time: 0.00131, step time: 0.00180\n",
      "Round: 2040, outer loss: 934.71149, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2050, outer loss: 536.79010, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 2060, outer loss: 4111.56738, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2070, outer loss: 450.04922, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2080, outer loss: 4616.85547, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2090, outer loss: 7.39073, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00176\n",
      "Round: 2100, outer loss: 878.76147, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2110, outer loss: 333.58670, inner loop time: 0.00031, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 2120, outer loss: 4404.45410, inner loop time: 0.00032, TAH time: 0.00133, step time: 0.00179\n",
      "Round: 2130, outer loss: 1062.44141, inner loop time: 0.00032, TAH time: 0.00130, step time: 0.00176\n",
      "Round: 2140, outer loss: 1984.94897, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2150, outer loss: 2.29226, inner loop time: 0.00038, TAH time: 0.00132, step time: 0.00185\n",
      "Round: 2160, outer loss: 5161.89307, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2170, outer loss: 2992.11133, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2180, outer loss: 1273.98596, inner loop time: 0.00032, TAH time: 0.00133, step time: 0.00179\n",
      "Round: 2190, outer loss: 36.87085, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2200, outer loss: 714.61493, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2210, outer loss: 95.35064, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2220, outer loss: 353.11545, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2230, outer loss: 3.57459, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2240, outer loss: 1825.54297, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2250, outer loss: 6.50741, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00176\n",
      "Round: 2260, outer loss: 382.55515, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2270, outer loss: 137.27367, inner loop time: 0.00033, TAH time: 0.00130, step time: 0.00177\n",
      "Round: 2280, outer loss: 1761.60645, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2290, outer loss: 53.41958, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 2300, outer loss: 1.44531, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00177\n",
      "Round: 2310, outer loss: 182.50502, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2320, outer loss: 225.50832, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2330, outer loss: 403.96460, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2340, outer loss: 283.71527, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2350, outer loss: 196.61858, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2360, outer loss: 9.65582, inner loop time: 0.00031, TAH time: 0.00129, step time: 0.00174\n",
      "Round: 2370, outer loss: 1.09815, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2380, outer loss: 166.09274, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00177\n",
      "Round: 2390, outer loss: 144.92241, inner loop time: 0.00035, TAH time: 0.00134, step time: 0.00183\n",
      "Round: 2400, outer loss: 3.61621, inner loop time: 0.00032, TAH time: 0.00130, step time: 0.00176\n",
      "Round: 2410, outer loss: 257.25714, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 2420, outer loss: 17.87333, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2430, outer loss: 22.85781, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2440, outer loss: 191.20868, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2450, outer loss: 1.24447, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2460, outer loss: 142.66206, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2470, outer loss: 55.16252, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2480, outer loss: 95.82666, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2490, outer loss: 43.19828, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2500, outer loss: 8.48960, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2510, outer loss: 3.76764, inner loop time: 0.00032, TAH time: 0.00134, step time: 0.00180\n",
      "Round: 2520, outer loss: 163.47392, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2530, outer loss: 16.95090, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2540, outer loss: 4.16020, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2550, outer loss: 25.87572, inner loop time: 0.00031, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 2560, outer loss: 99.96277, inner loop time: 0.00033, TAH time: 0.00131, step time: 0.00178\n",
      "Round: 2570, outer loss: 3.45636, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2580, outer loss: 89.91138, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2590, outer loss: 56.62479, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2600, outer loss: 4.98789, inner loop time: 0.00032, TAH time: 0.00130, step time: 0.00177\n",
      "Round: 2610, outer loss: 5.79887, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2620, outer loss: 19.87930, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2630, outer loss: 3.89603, inner loop time: 0.00034, TAH time: 0.00131, step time: 0.00179\n",
      "Round: 2640, outer loss: 4.56750, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2650, outer loss: 25.38392, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2660, outer loss: 0.30899, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2670, outer loss: 30.45806, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 2680, outer loss: 36.66970, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2690, outer loss: 10.46600, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2700, outer loss: 0.33933, inner loop time: 0.00032, TAH time: 0.00130, step time: 0.00176\n",
      "Round: 2710, outer loss: 1.19028, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2720, outer loss: 0.27697, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00175\n",
      "Round: 2730, outer loss: 22.92344, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 2740, outer loss: 7.51276, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2750, outer loss: 2.29128, inner loop time: 0.00032, TAH time: 0.00134, step time: 0.00181\n",
      "Round: 2760, outer loss: 0.41357, inner loop time: 0.00040, TAH time: 0.00139, step time: 0.00196\n",
      "Round: 2770, outer loss: 6.23888, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 2780, outer loss: 2.51132, inner loop time: 0.00032, TAH time: 0.00133, step time: 0.00179\n",
      "Round: 2790, outer loss: 0.26121, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2800, outer loss: 4.07267, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2810, outer loss: 8.46414, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00177\n",
      "Round: 2820, outer loss: 4.77456, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00175\n",
      "Round: 2830, outer loss: 6.53287, inner loop time: 0.00031, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2840, outer loss: 13.14803, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00178\n",
      "Round: 2850, outer loss: 0.37972, inner loop time: 0.00032, TAH time: 0.00132, step time: 0.00177\n",
      "Round: 2860, outer loss: 3.53988, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2870, outer loss: 0.17911, inner loop time: 0.00035, TAH time: 0.00143, step time: 0.00194\n",
      "Round: 2880, outer loss: 25.56170, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2890, outer loss: 1.98617, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00175\n",
      "Round: 2900, outer loss: 1.72346, inner loop time: 0.00031, TAH time: 0.00132, step time: 0.00177\n",
      "Round: 2910, outer loss: 11.53546, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2920, outer loss: 3.61075, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2930, outer loss: 2.44357, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2940, outer loss: 0.00007, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00176\n",
      "Round: 2950, outer loss: 0.97979, inner loop time: 0.00034, TAH time: 0.00135, step time: 0.00183\n",
      "Round: 2960, outer loss: 0.14169, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2970, outer loss: 2.59414, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 2980, outer loss: 0.28869, inner loop time: 0.00032, TAH time: 0.00130, step time: 0.00176\n",
      "Round: 2990, outer loss: 0.28619, inner loop time: 0.00037, TAH time: 0.00133, step time: 0.00185\n"
     ]
    }
   ],
   "source": [
    "g_iterator = CustomTensorIterator([Xg, yg], batch_size=1, shuffle=False, **kwargs)\n",
    "f_iterator = CustomTensorIterator([Xf, yf], batch_size=1, shuffle=False, **kwargs)\n",
    "\n",
    "xs = [Hr.clone().detach().requires_grad_(True)]\n",
    "ys = [wr.clone().detach()]\n",
    "\n",
    "T = n \n",
    "N = 1\n",
    "W = 20 # 10 or 50\n",
    "alpha = 1e-4\n",
    "beta = 1e-4\n",
    "\n",
    "outer_opt = torch.optim.Adam(lr=alpha, params=xs) \n",
    "u = torch.ones(W) # latest coeffs at the end \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",
    "for t in range(T): \n",
    "    step_start_time = time.time() \n",
    "    \n",
    "    # get data defining costs f_t and g_t\n",
    "    f_data = next(f_iterator)\n",
    "    g_data = next(g_iterator)\n",
    "    \n",
    "    ft = outer_func(ys, xs, f_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, g_data, beta, steps=N) # obtain y_{t+1}\n",
    "    t1 = time.time() - step_start_time # inner loop time\n",
    "    \n",
    "    grads, cost = hg.fixed_point(ys, xs, \n",
    "                        K=2*N, \n",
    "                        fp_map=lambda y, x: map_func(y, x, g_data, beta), \n",
    "                        outer_loss=lambda y, x: outer_func(y, x, f_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:6.3f}, inner loop time: {t1:6.5f}, TAH time: {t2:6.5f}, step time: {step_time:6.5f}')\n",
    "\n",
    "\n",
    "norms_ours_fp20 = grad_norms_list\n",
    "regret_ours_fp20 = regret_list \n",
    "rt_ours_fp20 = running_time "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "ours with oagd regret evaluation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Round: 0, outer loss: 3671.493, inner loop time: 0.00143, TAH time: 0.00239, step time: 0.00423\n",
      "Round: 10, outer loss: 1612.476, inner loop time: 0.00034, TAH time: 0.00118, step time: 0.00166\n",
      "Round: 20, outer loss: 465.760, inner loop time: 0.00032, TAH time: 0.00115, step time: 0.00161\n",
      "Round: 30, outer loss: 134.857, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00158\n",
      "Round: 40, outer loss: 2285.686, inner loop time: 0.00033, TAH time: 0.00113, step time: 0.00164\n",
      "Round: 50, outer loss: 59.162, inner loop time: 0.00031, TAH time: 0.00112, step time: 0.00156\n",
      "Round: 60, outer loss: 2125.285, inner loop time: 0.00040, TAH time: 0.00124, step time: 0.00182\n",
      "Round: 70, outer loss: 10.845, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00158\n",
      "Round: 80, outer loss: 164.043, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00163\n",
      "Round: 90, outer loss: 133.938, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00163\n",
      "Round: 100, outer loss: 207.005, inner loop time: 0.00034, TAH time: 0.00117, step time: 0.00170\n",
      "Round: 110, outer loss: 440.087, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00162\n",
      "Round: 120, outer loss: 652.113, inner loop time: 0.00033, TAH time: 0.00115, step time: 0.00162\n",
      "Round: 130, outer loss: 757.612, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00163\n",
      "Round: 140, outer loss: 1003.405, inner loop time: 0.00033, TAH time: 0.00124, step time: 0.00174\n",
      "Round: 150, outer loss: 84.931, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00158\n",
      "Round: 160, outer loss: 832.410, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00166\n",
      "Round: 170, outer loss: 898.463, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00162\n",
      "Round: 180, outer loss: 1506.593, inner loop time: 0.00034, TAH time: 0.00117, step time: 0.00167\n",
      "Round: 190, outer loss: 395.069, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00159\n",
      "Round: 200, outer loss:  5.434, inner loop time: 0.00033, TAH time: 0.00113, step time: 0.00165\n",
      "Round: 210, outer loss: 124.624, inner loop time: 0.00033, TAH time: 0.00115, step time: 0.00161\n",
      "Round: 220, outer loss: 442.586, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00164\n",
      "Round: 230, outer loss: 17.711, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00158\n",
      "Round: 240, outer loss: 321.846, inner loop time: 0.00034, TAH time: 0.00118, step time: 0.00170\n",
      "Round: 250, outer loss: 941.444, inner loop time: 0.00033, TAH time: 0.00119, step time: 0.00165\n",
      "Round: 260, outer loss: 52.400, inner loop time: 0.00033, TAH time: 0.00114, step time: 0.00164\n",
      "Round: 270, outer loss: 83.425, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00156\n",
      "Round: 280, outer loss: 44.834, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00161\n",
      "Round: 290, outer loss: 785.762, inner loop time: 0.00031, TAH time: 0.00112, step time: 0.00156\n",
      "Round: 300, outer loss: 37.892, inner loop time: 0.00032, TAH time: 0.00144, step time: 0.00189\n",
      "Round: 310, outer loss: 388.374, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00163\n",
      "Round: 320, outer loss: 358.213, inner loop time: 0.00032, TAH time: 0.00115, step time: 0.00162\n",
      "Round: 330, outer loss:  0.341, inner loop time: 0.00034, TAH time: 0.00118, step time: 0.00165\n",
      "Round: 340, outer loss: 10.477, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00165\n",
      "Round: 350, outer loss: 447.414, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00162\n",
      "Round: 360, outer loss:  6.551, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00165\n",
      "Round: 370, outer loss: 151.587, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00159\n",
      "Round: 380, outer loss: 50.367, inner loop time: 0.00033, TAH time: 0.00113, step time: 0.00161\n",
      "Round: 390, outer loss: 52.186, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00158\n",
      "Round: 400, outer loss: 30.775, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00161\n",
      "Round: 410, outer loss: 203.969, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00157\n",
      "Round: 420, outer loss: 228.881, inner loop time: 0.00034, TAH time: 0.00113, step time: 0.00162\n",
      "Round: 430, outer loss: 91.526, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00157\n",
      "Round: 440, outer loss: 36.370, inner loop time: 0.00034, TAH time: 0.00118, step time: 0.00166\n",
      "Round: 450, outer loss: 155.753, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00159\n",
      "Round: 460, outer loss: 10.000, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00166\n",
      "Round: 470, outer loss:  3.410, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00158\n",
      "Round: 480, outer loss:  2.859, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00166\n",
      "Round: 490, outer loss: 96.996, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00159\n",
      "Round: 500, outer loss: 63.373, inner loop time: 0.00034, TAH time: 0.00117, step time: 0.00164\n",
      "Round: 510, outer loss: 24.189, inner loop time: 0.00032, TAH time: 0.00116, step time: 0.00161\n",
      "Round: 520, outer loss: 24.529, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00161\n",
      "Round: 530, outer loss:  9.427, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00157\n",
      "Round: 540, outer loss:  5.957, inner loop time: 0.00033, TAH time: 0.00113, step time: 0.00163\n",
      "Round: 550, outer loss: 96.954, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00162\n",
      "Round: 560, outer loss: 55.826, inner loop time: 0.00033, TAH time: 0.00114, step time: 0.00165\n",
      "Round: 570, outer loss: 20.276, inner loop time: 0.00032, TAH time: 0.00116, step time: 0.00161\n",
      "Round: 580, outer loss:  0.493, inner loop time: 0.00034, TAH time: 0.00117, step time: 0.00169\n",
      "Round: 590, outer loss:  1.961, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00158\n",
      "Round: 600, outer loss: 17.441, inner loop time: 0.00034, TAH time: 0.00117, step time: 0.00166\n",
      "Round: 610, outer loss: 25.789, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00162\n",
      "Round: 620, outer loss: 21.307, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00166\n",
      "Round: 630, outer loss:  3.591, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00163\n",
      "Round: 640, outer loss:  4.947, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00164\n",
      "Round: 650, outer loss:  7.139, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00163\n",
      "Round: 660, outer loss:  0.209, inner loop time: 0.00034, TAH time: 0.00117, step time: 0.00166\n",
      "Round: 670, outer loss: 78.547, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00163\n",
      "Round: 680, outer loss: 24.771, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00167\n",
      "Round: 690, outer loss: 42.421, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00163\n",
      "Round: 700, outer loss:  1.503, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00162\n",
      "Round: 710, outer loss: 62.744, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00162\n",
      "Round: 720, outer loss:  0.009, inner loop time: 0.00034, TAH time: 0.00117, step time: 0.00166\n",
      "Round: 730, outer loss:  6.310, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00159\n",
      "Round: 740, outer loss:  8.961, inner loop time: 0.00033, TAH time: 0.00114, step time: 0.00164\n",
      "Round: 750, outer loss:  2.515, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00161\n",
      "Round: 760, outer loss: 11.185, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00160\n",
      "Round: 770, outer loss:  0.068, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00163\n",
      "Round: 780, outer loss: 14.078, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00167\n",
      "Round: 790, outer loss: 17.897, inner loop time: 0.00033, TAH time: 0.00115, step time: 0.00161\n",
      "Round: 800, outer loss:  0.416, inner loop time: 0.00033, TAH time: 0.00113, step time: 0.00162\n",
      "Round: 810, outer loss:  1.624, inner loop time: 0.00031, TAH time: 0.00112, step time: 0.00156\n",
      "Round: 820, outer loss:  4.066, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00160\n",
      "Round: 830, outer loss:  8.102, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00159\n",
      "Round: 840, outer loss:  0.596, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00175\n",
      "Round: 850, outer loss:  5.260, inner loop time: 0.00031, TAH time: 0.00111, step time: 0.00155\n",
      "Round: 860, outer loss:  2.829, inner loop time: 0.00034, TAH time: 0.00114, step time: 0.00161\n",
      "Round: 870, outer loss:  5.145, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00158\n",
      "Round: 880, outer loss:  0.029, inner loop time: 0.00031, TAH time: 0.00130, step time: 0.00179\n",
      "Round: 890, outer loss:  1.735, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00159\n",
      "Round: 900, outer loss:  3.208, inner loop time: 0.00031, TAH time: 0.00135, step time: 0.00189\n",
      "Round: 910, outer loss: 12.286, inner loop time: 0.00031, TAH time: 0.00115, step time: 0.00159\n",
      "Round: 920, outer loss:  1.558, inner loop time: 0.00032, TAH time: 0.00160, step time: 0.00207\n",
      "Round: 930, outer loss:  1.365, inner loop time: 0.00033, TAH time: 0.00118, step time: 0.00164\n",
      "Round: 940, outer loss:  5.906, inner loop time: 0.00036, TAH time: 0.00115, step time: 0.00168\n",
      "Round: 950, outer loss:  1.666, inner loop time: 0.00033, TAH time: 0.00119, step time: 0.00165\n",
      "Round: 960, outer loss:  4.214, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00169\n",
      "Round: 970, outer loss:  0.031, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00164\n",
      "Round: 980, outer loss:  9.391, inner loop time: 0.00033, TAH time: 0.00115, step time: 0.00165\n",
      "Round: 990, outer loss:  0.107, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00158\n",
      "Round: 1000, outer loss: 805.700, inner loop time: 0.00035, TAH time: 0.00117, step time: 0.00165\n",
      "Round: 1010, outer loss: 781.968, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00158\n",
      "Round: 1020, outer loss: 12.156, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00161\n",
      "Round: 1030, outer loss: 12974.946, inner loop time: 0.00033, TAH time: 0.00113, step time: 0.00158\n",
      "Round: 1040, outer loss: 31.745, inner loop time: 0.00032, TAH time: 0.00141, step time: 0.00190\n",
      "Round: 1050, outer loss: 3715.973, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00159\n",
      "Round: 1060, outer loss: 1090.051, inner loop time: 0.00033, TAH time: 0.00118, step time: 0.00167\n",
      "Round: 1070, outer loss:  8.306, inner loop time: 0.00032, TAH time: 0.00116, step time: 0.00161\n",
      "Round: 1080, outer loss: 6131.688, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00167\n",
      "Round: 1090, outer loss: 16538.947, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00158\n",
      "Round: 1100, outer loss: 1389.712, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00162\n",
      "Round: 1110, outer loss: 20.218, inner loop time: 0.00031, TAH time: 0.00112, step time: 0.00156\n",
      "Round: 1120, outer loss: 397.352, inner loop time: 0.00032, TAH time: 0.00131, step time: 0.00177\n",
      "Round: 1130, outer loss: 554.002, inner loop time: 0.00031, TAH time: 0.00112, step time: 0.00155\n",
      "Round: 1140, outer loss: 500.407, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00166\n",
      "Round: 1150, outer loss: 29.765, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00159\n",
      "Round: 1160, outer loss: 1226.218, inner loop time: 0.00034, TAH time: 0.00117, step time: 0.00165\n",
      "Round: 1170, outer loss: 7102.070, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00162\n",
      "Round: 1180, outer loss: 1725.158, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00163\n",
      "Round: 1190, outer loss:  3.840, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00163\n",
      "Round: 1200, outer loss:  4.387, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00161\n",
      "Round: 1210, outer loss: 43.960, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00157\n",
      "Round: 1220, outer loss: 40.394, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00161\n",
      "Round: 1230, outer loss: 191.464, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00162\n",
      "Round: 1240, outer loss: 670.773, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00164\n",
      "Round: 1250, outer loss: 2853.781, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00158\n",
      "Round: 1260, outer loss: 395.125, inner loop time: 0.00034, TAH time: 0.00118, step time: 0.00170\n",
      "Round: 1270, outer loss: 138.875, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00158\n",
      "Round: 1280, outer loss: 346.850, inner loop time: 0.00032, TAH time: 0.00123, step time: 0.00168\n",
      "Round: 1290, outer loss: 409.526, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00163\n",
      "Round: 1300, outer loss: 46.232, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00163\n",
      "Round: 1310, outer loss: 71.931, inner loop time: 0.00033, TAH time: 0.00118, step time: 0.00171\n",
      "Round: 1320, outer loss: 13.465, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00164\n",
      "Round: 1330, outer loss:  9.828, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00157\n",
      "Round: 1340, outer loss: 480.539, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00167\n",
      "Round: 1350, outer loss: 16.655, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00158\n",
      "Round: 1360, outer loss: 47.244, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00159\n",
      "Round: 1370, outer loss:  6.158, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00162\n",
      "Round: 1380, outer loss: 54.738, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00166\n",
      "Round: 1390, outer loss: 247.296, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00162\n",
      "Round: 1400, outer loss: 235.806, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00160\n",
      "Round: 1410, outer loss: 59.928, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00156\n",
      "Round: 1420, outer loss:  0.754, inner loop time: 0.00034, TAH time: 0.00118, step time: 0.00164\n",
      "Round: 1430, outer loss: 352.787, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00162\n",
      "Round: 1440, outer loss: 326.181, inner loop time: 0.00033, TAH time: 0.00112, step time: 0.00162\n",
      "Round: 1450, outer loss:  3.710, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00157\n",
      "Round: 1460, outer loss: 173.313, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00166\n",
      "Round: 1470, outer loss: 364.575, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00162\n",
      "Round: 1480, outer loss:  0.245, inner loop time: 0.00033, TAH time: 0.00115, step time: 0.00163\n",
      "Round: 1490, outer loss: 46.898, inner loop time: 0.00033, TAH time: 0.00115, step time: 0.00161\n",
      "Round: 1500, outer loss:  8.775, inner loop time: 0.00033, TAH time: 0.00113, step time: 0.00162\n",
      "Round: 1510, outer loss: 260.550, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00162\n",
      "Round: 1520, outer loss: 16.921, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00160\n",
      "Round: 1530, outer loss:  5.113, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00163\n",
      "Round: 1540, outer loss:  2.037, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00166\n",
      "Round: 1550, outer loss: 131.133, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00162\n",
      "Round: 1560, outer loss: 12.355, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00165\n",
      "Round: 1570, outer loss:  9.235, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00158\n",
      "Round: 1580, outer loss: 27.615, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00165\n",
      "Round: 1590, outer loss: 29.923, inner loop time: 0.00031, TAH time: 0.00112, step time: 0.00156\n",
      "Round: 1600, outer loss: 13.318, inner loop time: 0.00032, TAH time: 0.00130, step time: 0.00176\n",
      "Round: 1610, outer loss: 26.344, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00157\n",
      "Round: 1620, outer loss:  0.756, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00163\n",
      "Round: 1630, outer loss: 90.555, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00163\n",
      "Round: 1640, outer loss:  8.414, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00161\n",
      "Round: 1650, outer loss: 135.645, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00162\n",
      "Round: 1660, outer loss:  1.266, inner loop time: 0.00033, TAH time: 0.00121, step time: 0.00173\n",
      "Round: 1670, outer loss: 121.914, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00159\n",
      "Round: 1680, outer loss: 20.538, inner loop time: 0.00034, TAH time: 0.00117, step time: 0.00168\n",
      "Round: 1690, outer loss: 138.126, inner loop time: 0.00032, TAH time: 0.00111, step time: 0.00155\n",
      "Round: 1700, outer loss:  3.374, inner loop time: 0.00033, TAH time: 0.00114, step time: 0.00163\n",
      "Round: 1710, outer loss:  0.320, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00158\n",
      "Round: 1720, outer loss: 15.535, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00161\n",
      "Round: 1730, outer loss:  3.972, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00158\n",
      "Round: 1740, outer loss:  1.304, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00162\n",
      "Round: 1750, outer loss:  6.643, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00157\n",
      "Round: 1760, outer loss:  0.753, inner loop time: 0.00032, TAH time: 0.00144, step time: 0.00194\n",
      "Round: 1770, outer loss: 15.806, inner loop time: 0.00031, TAH time: 0.00111, step time: 0.00156\n",
      "Round: 1780, outer loss:  0.045, inner loop time: 0.00036, TAH time: 0.00120, step time: 0.00171\n",
      "Round: 1790, outer loss:  0.307, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00163\n",
      "Round: 1800, outer loss: 36.430, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00165\n",
      "Round: 1810, outer loss:  2.335, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00162\n",
      "Round: 1820, outer loss:  3.411, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00161\n",
      "Round: 1830, outer loss:  0.002, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00162\n",
      "Round: 1840, outer loss:  0.200, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00163\n",
      "Round: 1850, outer loss:  1.442, inner loop time: 0.00033, TAH time: 0.00118, step time: 0.00165\n",
      "Round: 1860, outer loss:  0.006, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00162\n",
      "Round: 1870, outer loss:  0.259, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00164\n",
      "Round: 1880, outer loss:  0.328, inner loop time: 0.00033, TAH time: 0.00114, step time: 0.00163\n",
      "Round: 1890, outer loss:  0.014, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00158\n",
      "Round: 1900, outer loss:  1.828, inner loop time: 0.00033, TAH time: 0.00118, step time: 0.00168\n",
      "Round: 1910, outer loss:  0.430, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00162\n",
      "Round: 1920, outer loss: 11.623, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00168\n",
      "Round: 1930, outer loss:  2.978, inner loop time: 0.00031, TAH time: 0.00112, step time: 0.00156\n",
      "Round: 1940, outer loss:  0.030, inner loop time: 0.00034, TAH time: 0.00118, step time: 0.00168\n",
      "Round: 1950, outer loss:  0.234, inner loop time: 0.00035, TAH time: 0.00116, step time: 0.00164\n",
      "Round: 1960, outer loss:  0.604, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00164\n",
      "Round: 1970, outer loss:  0.830, inner loop time: 0.00032, TAH time: 0.00116, step time: 0.00161\n",
      "Round: 1980, outer loss:  1.331, inner loop time: 0.00034, TAH time: 0.00117, step time: 0.00167\n",
      "Round: 1990, outer loss:  0.120, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00161\n",
      "Round: 2000, outer loss: 1077.187, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00163\n",
      "Round: 2010, outer loss: 77.064, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00156\n",
      "Round: 2020, outer loss: 13031.604, inner loop time: 0.00033, TAH time: 0.00113, step time: 0.00162\n",
      "Round: 2030, outer loss: 2038.740, inner loop time: 0.00031, TAH time: 0.00112, step time: 0.00156\n",
      "Round: 2040, outer loss: 899.833, inner loop time: 0.00032, TAH time: 0.00140, step time: 0.00189\n",
      "Round: 2050, outer loss: 601.022, inner loop time: 0.00031, TAH time: 0.00112, step time: 0.00156\n",
      "Round: 2060, outer loss: 4190.410, inner loop time: 0.00033, TAH time: 0.00118, step time: 0.00171\n",
      "Round: 2070, outer loss: 496.778, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00161\n",
      "Round: 2080, outer loss: 4664.851, inner loop time: 0.00034, TAH time: 0.00117, step time: 0.00169\n",
      "Round: 2090, outer loss:  1.653, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00159\n",
      "Round: 2100, outer loss: 835.831, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00168\n",
      "Round: 2110, outer loss: 408.720, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00161\n",
      "Round: 2120, outer loss: 4815.423, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00163\n",
      "Round: 2130, outer loss: 1223.214, inner loop time: 0.00031, TAH time: 0.00113, step time: 0.00158\n",
      "Round: 2140, outer loss: 2232.788, inner loop time: 0.00031, TAH time: 0.00133, step time: 0.00178\n",
      "Round: 2150, outer loss:  0.636, inner loop time: 0.00031, TAH time: 0.00111, step time: 0.00156\n",
      "Round: 2160, outer loss: 5544.560, inner loop time: 0.00033, TAH time: 0.00114, step time: 0.00163\n",
      "Round: 2170, outer loss: 3312.511, inner loop time: 0.00034, TAH time: 0.00118, step time: 0.00166\n",
      "Round: 2180, outer loss: 1509.919, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00159\n",
      "Round: 2190, outer loss: 14.087, inner loop time: 0.00031, TAH time: 0.00112, step time: 0.00156\n",
      "Round: 2200, outer loss: 919.307, inner loop time: 0.00032, TAH time: 0.00133, step time: 0.00183\n",
      "Round: 2210, outer loss: 87.231, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00158\n",
      "Round: 2220, outer loss: 391.810, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00168\n",
      "Round: 2230, outer loss: 16.210, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00158\n",
      "Round: 2240, outer loss: 2158.026, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00162\n",
      "Round: 2250, outer loss: 10.944, inner loop time: 0.00031, TAH time: 0.00111, step time: 0.00155\n",
      "Round: 2260, outer loss: 435.220, inner loop time: 0.00032, TAH time: 0.00142, step time: 0.00188\n",
      "Round: 2270, outer loss: 149.692, inner loop time: 0.00031, TAH time: 0.00111, step time: 0.00155\n",
      "Round: 2280, outer loss: 2084.001, inner loop time: 0.00049, TAH time: 0.00134, step time: 0.00199\n",
      "Round: 2290, outer loss: 63.918, inner loop time: 0.00033, TAH time: 0.00120, step time: 0.00165\n",
      "Round: 2300, outer loss:  1.832, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00164\n",
      "Round: 2310, outer loss: 189.635, inner loop time: 0.00031, TAH time: 0.00111, step time: 0.00155\n",
      "Round: 2320, outer loss: 229.990, inner loop time: 0.00032, TAH time: 0.00147, step time: 0.00202\n",
      "Round: 2330, outer loss: 421.021, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00158\n",
      "Round: 2340, outer loss: 315.995, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00161\n",
      "Round: 2350, outer loss: 224.527, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00156\n",
      "Round: 2360, outer loss: 18.157, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00164\n",
      "Round: 2370, outer loss:  1.805, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00156\n",
      "Round: 2380, outer loss: 222.702, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00169\n",
      "Round: 2390, outer loss: 168.192, inner loop time: 0.00038, TAH time: 0.00123, step time: 0.00179\n",
      "Round: 2400, outer loss:  4.866, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00167\n",
      "Round: 2410, outer loss: 342.375, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00158\n",
      "Round: 2420, outer loss: 17.503, inner loop time: 0.00035, TAH time: 0.00121, step time: 0.00174\n",
      "Round: 2430, outer loss: 32.561, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00161\n",
      "Round: 2440, outer loss: 230.974, inner loop time: 0.00035, TAH time: 0.00118, step time: 0.00172\n",
      "Round: 2450, outer loss:  1.861, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00158\n",
      "Round: 2460, outer loss: 206.170, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00168\n",
      "Round: 2470, outer loss: 71.764, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00158\n",
      "Round: 2480, outer loss: 140.022, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00162\n",
      "Round: 2490, outer loss: 56.588, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00159\n",
      "Round: 2500, outer loss: 11.695, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00170\n",
      "Round: 2510, outer loss:  8.261, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00156\n",
      "Round: 2520, outer loss: 223.783, inner loop time: 0.00034, TAH time: 0.00117, step time: 0.00169\n",
      "Round: 2530, outer loss: 21.167, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00162\n",
      "Round: 2540, outer loss:  9.071, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00168\n",
      "Round: 2550, outer loss: 31.443, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00162\n",
      "Round: 2560, outer loss: 165.999, inner loop time: 0.00032, TAH time: 0.00115, step time: 0.00166\n",
      "Round: 2570, outer loss:  3.074, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00157\n",
      "Round: 2580, outer loss: 146.692, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00168\n",
      "Round: 2590, outer loss: 93.314, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00159\n",
      "Round: 2600, outer loss:  7.978, inner loop time: 0.00031, TAH time: 0.00135, step time: 0.00181\n",
      "Round: 2610, outer loss:  6.977, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00158\n",
      "Round: 2620, outer loss: 46.030, inner loop time: 0.00034, TAH time: 0.00118, step time: 0.00169\n",
      "Round: 2630, outer loss:  3.575, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00161\n",
      "Round: 2640, outer loss: 10.648, inner loop time: 0.00033, TAH time: 0.00114, step time: 0.00165\n",
      "Round: 2650, outer loss: 41.338, inner loop time: 0.00033, TAH time: 0.00118, step time: 0.00164\n",
      "Round: 2660, outer loss:  0.524, inner loop time: 0.00034, TAH time: 0.00117, step time: 0.00168\n",
      "Round: 2670, outer loss: 52.092, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00159\n",
      "Round: 2680, outer loss: 63.340, inner loop time: 0.00034, TAH time: 0.00117, step time: 0.00169\n",
      "Round: 2690, outer loss: 21.153, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00159\n",
      "Round: 2700, outer loss:  0.543, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00161\n",
      "Round: 2710, outer loss:  3.627, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00163\n",
      "Round: 2720, outer loss:  0.397, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00163\n",
      "Round: 2730, outer loss: 45.923, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00163\n",
      "Round: 2740, outer loss: 16.133, inner loop time: 0.00033, TAH time: 0.00112, step time: 0.00162\n",
      "Round: 2750, outer loss:  4.674, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00162\n",
      "Round: 2760, outer loss:  0.675, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00164\n",
      "Round: 2770, outer loss: 13.244, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00163\n",
      "Round: 2780, outer loss:  3.696, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00169\n",
      "Round: 2790, outer loss:  0.013, inner loop time: 0.00031, TAH time: 0.00112, step time: 0.00156\n",
      "Round: 2800, outer loss:  9.028, inner loop time: 0.00032, TAH time: 0.00138, step time: 0.00188\n",
      "Round: 2810, outer loss: 15.736, inner loop time: 0.00034, TAH time: 0.00116, step time: 0.00163\n",
      "Round: 2820, outer loss: 10.730, inner loop time: 0.00037, TAH time: 0.00126, step time: 0.00178\n",
      "Round: 2830, outer loss: 13.671, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00156\n",
      "Round: 2840, outer loss: 26.804, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00169\n",
      "Round: 2850, outer loss:  1.613, inner loop time: 0.00033, TAH time: 0.00118, step time: 0.00164\n",
      "Round: 2860, outer loss: 10.118, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00168\n",
      "Round: 2870, outer loss:  0.687, inner loop time: 0.00033, TAH time: 0.00115, step time: 0.00161\n",
      "Round: 2880, outer loss: 56.447, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00168\n",
      "Round: 2890, outer loss:  4.660, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00159\n",
      "Round: 2900, outer loss:  2.649, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00164\n",
      "Round: 2910, outer loss: 25.444, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00163\n",
      "Round: 2920, outer loss:  7.854, inner loop time: 0.00032, TAH time: 0.00113, step time: 0.00165\n",
      "Round: 2930, outer loss:  3.105, inner loop time: 0.00032, TAH time: 0.00112, step time: 0.00157\n",
      "Round: 2940, outer loss:  0.010, inner loop time: 0.00033, TAH time: 0.00117, step time: 0.00171\n",
      "Round: 2950, outer loss:  1.546, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00163\n",
      "Round: 2960, outer loss:  0.627, inner loop time: 0.00033, TAH time: 0.00115, step time: 0.00164\n",
      "Round: 2970, outer loss:  5.507, inner loop time: 0.00032, TAH time: 0.00114, step time: 0.00159\n",
      "Round: 2980, outer loss:  1.429, inner loop time: 0.00034, TAH time: 0.00117, step time: 0.00170\n",
      "Round: 2990, outer loss:  0.719, inner loop time: 0.00033, TAH time: 0.00116, step time: 0.00162\n"
     ]
    }
   ],
   "source": [
    "g_iterator = CustomTensorIterator([Xg, yg], batch_size=1, shuffle=False, **kwargs)\n",
    "f_iterator = CustomTensorIterator([Xf, yf], batch_size=1, shuffle=False, **kwargs)\n",
    "\n",
    "xs = [Hr.clone().detach().requires_grad_(True)]\n",
    "ys = [wr.clone().detach()]\n",
    "\n",
    "T = n \n",
    "N = 1\n",
    "W = 10 # 10 or 50\n",
    "alpha = 1e-4\n",
    "beta = 1e-4\n",
    "\n",
    "outer_opt = torch.optim.Adam(lr=alpha, params=xs) \n",
    "u = torch.ones(W) # latest coeffs at the end \n",
    "\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",
    "buffer = []\n",
    "regret_dav = 0. \n",
    "regret_dav_list = [] \n",
    "\n",
    "for t in range(T): \n",
    "    step_start_time = time.time() \n",
    "    \n",
    "    # get data defining costs f_t and g_t\n",
    "    f_data = next(f_iterator)\n",
    "    g_data = next(g_iterator)\n",
    "    buffer.append((f_data, g_data))\n",
    "    del buffer[:-W]\n",
    "    \n",
    "    ft = outer_func(ys, xs, f_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, g_data, beta, steps=N) # obtain y_{t+1}\n",
    "    t1 = time.time() - step_start_time # inner loop time\n",
    "    \n",
    "    grads, cost = hg.fixed_point(ys, xs, \n",
    "                        K=2*N, \n",
    "                        fp_map=lambda y, x: map_func(y, x, g_data, beta), \n",
    "                        outer_loss=lambda y, x: outer_func(y, x, f_data), \n",
    "                        set_grad=False)\n",
    "    \n",
    "    hg_buffer.append(grads)\n",
    "    del hg_buffer[:-W]\n",
    "    TAH = getTimeAveHypergrad(hg_buffer, u) \n",
    "    \n",
    "    norm = torch.norm(TAH[0]).item() \n",
    "    sum_grad_norms += norm \n",
    "    grad_norms_list.append(sum_grad_norms) \n",
    "    t2 = time.time() - step_start_time - t1 # time averaged hypergrad estimation time\n",
    "    \n",
    "    \n",
    "    \n",
    "    TAH_dav = [torch.zeros_like(Hr)]\n",
    "    for i, (f_data, g_data) in enumerate(buffer): # buffer[-W:]\n",
    "        grads, cost = hg.fixed_point(ys, xs, \n",
    "                            K=2*N, \n",
    "                            fp_map=lambda y, x: map_func(y, x, g_data, beta), \n",
    "                            outer_loss=lambda y, x: outer_func(y, x, f_data), \n",
    "                            set_grad=False)\n",
    "        with torch.no_grad():\n",
    "            for av, g in zip(TAH_dav, grads):\n",
    "                av.copy_(av + u[i]*g)                        \n",
    "    TAH_dav = [av / torch.sum(u[-len(buffer):]) for av in TAH_dav]\n",
    "    norm = torch.norm(TAH_dav[0]).item() \n",
    "    regret_dav += norm \n",
    "    regret_dav_list.append(regret_dav) \n",
    "    \n",
    "    \n",
    "    \n",
    "    t3 = time.time() \n",
    "    outer_opt.zero_grad()\n",
    "    update_tensor_grads(xs, TAH)\n",
    "    outer_opt.step()\n",
    "    \n",
    "    step_time = time.time()- t3 + t1 + t2\n",
    "    total_time +=step_time \n",
    "    running_time.append(total_time) \n",
    "    \n",
    "    if t%10 == 0:\n",
    "        print(f'Round: {t}, outer loss: {ft:6.3f}, inner loop time: {t1:6.5f}, TAH time: {t2:6.5f}, step time: {step_time:6.5f}')\n",
    "\n",
    "\n",
    "norms_ours_fp20 = grad_norms_list \n",
    "regret_ours_fp20 = regret_list \n",
    "rt_ours_fp20 = running_time "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# oagd"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Round: 0, outer loss: 3671.492919921875, grad norm: 7840.84619140625\n",
      "Round: 10, outer loss: 1630.0474853515625, grad norm: 2000.83740234375\n",
      "Round: 20, outer loss: 490.4758605957031, grad norm: 2502.050537109375\n",
      "Round: 30, outer loss: 181.79635620117188, grad norm: 2582.586669921875\n",
      "Round: 40, outer loss: 2275.737060546875, grad norm: 2972.24755859375\n",
      "Round: 50, outer loss: 83.12561798095703, grad norm: 1991.160400390625\n",
      "Round: 60, outer loss: 2113.579345703125, grad norm: 2507.17822265625\n",
      "Round: 70, outer loss: 5.61162805557251, grad norm: 2504.203125\n",
      "Round: 80, outer loss: 119.84088134765625, grad norm: 2069.264404296875\n",
      "Round: 90, outer loss: 134.545166015625, grad norm: 967.091796875\n",
      "Round: 100, outer loss: 212.08677673339844, grad norm: 1051.2176513671875\n",
      "Round: 110, outer loss: 472.11883544921875, grad norm: 1875.7459716796875\n",
      "Round: 120, outer loss: 726.4761352539062, grad norm: 1651.4019775390625\n",
      "Round: 130, outer loss: 680.00390625, grad norm: 1576.7161865234375\n",
      "Round: 140, outer loss: 1090.5731201171875, grad norm: 1108.556884765625\n",
      "Round: 150, outer loss: 79.25093078613281, grad norm: 1434.0535888671875\n",
      "Round: 160, outer loss: 1065.748779296875, grad norm: 1273.3212890625\n",
      "Round: 170, outer loss: 828.514404296875, grad norm: 1296.876220703125\n",
      "Round: 180, outer loss: 1586.42578125, grad norm: 1051.0308837890625\n",
      "Round: 190, outer loss: 372.892578125, grad norm: 1599.618408203125\n",
      "Round: 200, outer loss: 10.483805656433105, grad norm: 841.3082275390625\n",
      "Round: 210, outer loss: 127.25821685791016, grad norm: 1219.7786865234375\n",
      "Round: 220, outer loss: 434.35089111328125, grad norm: 1014.6798095703125\n",
      "Round: 230, outer loss: 17.842327117919922, grad norm: 861.8641357421875\n",
      "Round: 240, outer loss: 288.3204345703125, grad norm: 904.3467407226562\n",
      "Round: 250, outer loss: 916.5994262695312, grad norm: 1045.66552734375\n",
      "Round: 260, outer loss: 56.83814239501953, grad norm: 802.1185302734375\n",
      "Round: 270, outer loss: 89.8577880859375, grad norm: 1217.1722412109375\n",
      "Round: 280, outer loss: 35.003318786621094, grad norm: 1277.281982421875\n",
      "Round: 290, outer loss: 665.9364624023438, grad norm: 876.2408447265625\n",
      "Round: 300, outer loss: 44.254234313964844, grad norm: 265.8956298828125\n",
      "Round: 310, outer loss: 359.8365783691406, grad norm: 739.2166748046875\n",
      "Round: 320, outer loss: 373.735107421875, grad norm: 834.66552734375\n",
      "Round: 330, outer loss: 0.710565447807312, grad norm: 502.4510803222656\n",
      "Round: 340, outer loss: 24.855546951293945, grad norm: 427.51171875\n",
      "Round: 350, outer loss: 457.85003662109375, grad norm: 683.7815551757812\n",
      "Round: 360, outer loss: 5.487490653991699, grad norm: 642.4226684570312\n",
      "Round: 370, outer loss: 141.66909790039062, grad norm: 719.2042236328125\n",
      "Round: 380, outer loss: 30.424442291259766, grad norm: 369.10284423828125\n",
      "Round: 390, outer loss: 43.61528015136719, grad norm: 480.8222961425781\n",
      "Round: 400, outer loss: 15.723472595214844, grad norm: 384.037109375\n",
      "Round: 410, outer loss: 221.2624969482422, grad norm: 447.3962707519531\n",
      "Round: 420, outer loss: 228.6680450439453, grad norm: 489.2508850097656\n",
      "Round: 430, outer loss: 102.28162384033203, grad norm: 556.1185913085938\n",
      "Round: 440, outer loss: 43.970279693603516, grad norm: 434.4980773925781\n",
      "Round: 450, outer loss: 149.55477905273438, grad norm: 456.3304138183594\n",
      "Round: 460, outer loss: 6.9302778244018555, grad norm: 465.0367126464844\n",
      "Round: 470, outer loss: 7.110723495483398, grad norm: 519.24169921875\n",
      "Round: 480, outer loss: 3.0639700889587402, grad norm: 316.3479919433594\n",
      "Round: 490, outer loss: 92.45513153076172, grad norm: 346.6475524902344\n",
      "Round: 500, outer loss: 55.469200134277344, grad norm: 287.8501281738281\n",
      "Round: 510, outer loss: 23.18657875061035, grad norm: 310.82464599609375\n",
      "Round: 520, outer loss: 22.006362915039062, grad norm: 419.980224609375\n",
      "Round: 530, outer loss: 5.935102462768555, grad norm: 220.04742431640625\n",
      "Round: 540, outer loss: 4.570864200592041, grad norm: 285.7051086425781\n",
      "Round: 550, outer loss: 72.08445739746094, grad norm: 305.88079833984375\n",
      "Round: 560, outer loss: 53.015567779541016, grad norm: 297.55316162109375\n",
      "Round: 570, outer loss: 21.136011123657227, grad norm: 335.72198486328125\n",
      "Round: 580, outer loss: 0.856810450553894, grad norm: 330.14208984375\n",
      "Round: 590, outer loss: 3.083695411682129, grad norm: 371.3921203613281\n",
      "Round: 600, outer loss: 11.20003890991211, grad norm: 195.42437744140625\n",
      "Round: 610, outer loss: 26.095643997192383, grad norm: 289.41571044921875\n",
      "Round: 620, outer loss: 21.263113021850586, grad norm: 244.22427368164062\n",
      "Round: 630, outer loss: 1.4920045137405396, grad norm: 317.84014892578125\n",
      "Round: 640, outer loss: 4.318902969360352, grad norm: 158.65174865722656\n",
      "Round: 650, outer loss: 4.505987167358398, grad norm: 111.40186309814453\n",
      "Round: 660, outer loss: 1.251013159751892, grad norm: 143.95343017578125\n",
      "Round: 670, outer loss: 60.057613372802734, grad norm: 226.82589721679688\n",
      "Round: 680, outer loss: 16.077987670898438, grad norm: 123.25325775146484\n",
      "Round: 690, outer loss: 42.31520080566406, grad norm: 178.04531860351562\n",
      "Round: 700, outer loss: 0.9230351448059082, grad norm: 172.28672790527344\n",
      "Round: 710, outer loss: 58.2197151184082, grad norm: 237.74427795410156\n",
      "Round: 720, outer loss: 0.18548767268657684, grad norm: 115.97095489501953\n",
      "Round: 730, outer loss: 6.262373924255371, grad norm: 130.8765106201172\n",
      "Round: 740, outer loss: 6.810381889343262, grad norm: 157.06455993652344\n",
      "Round: 750, outer loss: 1.542914628982544, grad norm: 163.505615234375\n",
      "Round: 760, outer loss: 9.360389709472656, grad norm: 149.92628479003906\n",
      "Round: 770, outer loss: 0.05265757814049721, grad norm: 124.29318237304688\n",
      "Round: 780, outer loss: 10.097578048706055, grad norm: 141.7802276611328\n",
      "Round: 790, outer loss: 15.418293952941895, grad norm: 105.4811782836914\n",
      "Round: 800, outer loss: 0.6324177384376526, grad norm: 154.02574157714844\n",
      "Round: 810, outer loss: 1.5488725900650024, grad norm: 73.92642211914062\n",
      "Round: 820, outer loss: 4.390538692474365, grad norm: 118.8342056274414\n",
      "Round: 830, outer loss: 7.016838073730469, grad norm: 178.96678161621094\n",
      "Round: 840, outer loss: 0.6629283428192139, grad norm: 89.08004760742188\n",
      "Round: 850, outer loss: 5.055473327636719, grad norm: 101.08140563964844\n",
      "Round: 860, outer loss: 1.926379919052124, grad norm: 86.63215637207031\n",
      "Round: 870, outer loss: 3.791506052017212, grad norm: 78.71101379394531\n",
      "Round: 880, outer loss: 0.01581577956676483, grad norm: 72.59512329101562\n",
      "Round: 890, outer loss: 1.2599908113479614, grad norm: 158.862060546875\n",
      "Round: 900, outer loss: 2.568464756011963, grad norm: 66.3719482421875\n",
      "Round: 910, outer loss: 10.76079273223877, grad norm: 94.42127227783203\n",
      "Round: 920, outer loss: 0.8715118169784546, grad norm: 61.406639099121094\n",
      "Round: 930, outer loss: 0.8061115145683289, grad norm: 86.10079956054688\n",
      "Round: 940, outer loss: 4.648118495941162, grad norm: 69.53147888183594\n",
      "Round: 950, outer loss: 1.1939811706542969, grad norm: 83.38134765625\n",
      "Round: 960, outer loss: 2.5607669353485107, grad norm: 64.74578857421875\n",
      "Round: 970, outer loss: 0.08768648654222488, grad norm: 57.72785568237305\n",
      "Round: 980, outer loss: 7.603354454040527, grad norm: 62.615821838378906\n",
      "Round: 990, outer loss: 0.08303872495889664, grad norm: 53.88966751098633\n",
      "Round: 1000, outer loss: 804.43603515625, grad norm: 380.10504150390625\n",
      "Round: 1010, outer loss: 677.168701171875, grad norm: 3033.044189453125\n",
      "Round: 1020, outer loss: 2.9795897006988525, grad norm: 3067.82568359375\n",
      "Round: 1030, outer loss: 13083.3974609375, grad norm: 5950.2861328125\n",
      "Round: 1040, outer loss: 29.258590698242188, grad norm: 1187.80029296875\n",
      "Round: 1050, outer loss: 3726.35888671875, grad norm: 4589.48681640625\n",
      "Round: 1060, outer loss: 1043.63525390625, grad norm: 3028.839599609375\n",
      "Round: 1070, outer loss: 17.3359375, grad norm: 3006.93017578125\n",
      "Round: 1080, outer loss: 6568.02880859375, grad norm: 3145.49658203125\n",
      "Round: 1090, outer loss: 16243.4384765625, grad norm: 3682.430908203125\n",
      "Round: 1100, outer loss: 1581.364013671875, grad norm: 2897.992431640625\n",
      "Round: 1110, outer loss: 19.87790870666504, grad norm: 762.7550659179688\n",
      "Round: 1120, outer loss: 475.6851501464844, grad norm: 2699.2431640625\n",
      "Round: 1130, outer loss: 610.5557250976562, grad norm: 2638.190185546875\n",
      "Round: 1140, outer loss: 600.5283813476562, grad norm: 2266.489013671875\n",
      "Round: 1150, outer loss: 0.07439851760864258, grad norm: 1830.549560546875\n",
      "Round: 1160, outer loss: 1166.908203125, grad norm: 1731.79052734375\n",
      "Round: 1170, outer loss: 6922.73291015625, grad norm: 2138.299072265625\n",
      "Round: 1180, outer loss: 1594.387451171875, grad norm: 1607.4781494140625\n",
      "Round: 1190, outer loss: 1.0899381637573242, grad norm: 646.805908203125\n",
      "Round: 1200, outer loss: 28.925640106201172, grad norm: 1574.353759765625\n",
      "Round: 1210, outer loss: 45.65845489501953, grad norm: 781.8336181640625\n",
      "Round: 1220, outer loss: 59.258785247802734, grad norm: 1154.327392578125\n",
      "Round: 1230, outer loss: 204.5530548095703, grad norm: 1748.5374755859375\n",
      "Round: 1240, outer loss: 658.23681640625, grad norm: 1854.296142578125\n",
      "Round: 1250, outer loss: 2668.3828125, grad norm: 1472.8463134765625\n",
      "Round: 1260, outer loss: 480.54351806640625, grad norm: 1216.798828125\n",
      "Round: 1270, outer loss: 90.08979797363281, grad norm: 1463.32373046875\n",
      "Round: 1280, outer loss: 239.46217346191406, grad norm: 1182.670654296875\n",
      "Round: 1290, outer loss: 438.52496337890625, grad norm: 1092.515625\n",
      "Round: 1300, outer loss: 37.10607147216797, grad norm: 547.2586669921875\n",
      "Round: 1310, outer loss: 57.18342971801758, grad norm: 739.3643188476562\n",
      "Round: 1320, outer loss: 9.107625961303711, grad norm: 1069.4884033203125\n",
      "Round: 1330, outer loss: 2.299985647201538, grad norm: 450.7214660644531\n",
      "Round: 1340, outer loss: 382.1669921875, grad norm: 1124.8551025390625\n",
      "Round: 1350, outer loss: 23.48518180847168, grad norm: 904.61572265625\n",
      "Round: 1360, outer loss: 43.86652755737305, grad norm: 957.4710083007812\n",
      "Round: 1370, outer loss: 4.127845764160156, grad norm: 542.0175170898438\n",
      "Round: 1380, outer loss: 49.95636749267578, grad norm: 678.9580078125\n",
      "Round: 1390, outer loss: 274.63726806640625, grad norm: 1021.1439819335938\n",
      "Round: 1400, outer loss: 212.04238891601562, grad norm: 419.08087158203125\n",
      "Round: 1410, outer loss: 63.88172149658203, grad norm: 828.2136840820312\n",
      "Round: 1420, outer loss: 0.21625426411628723, grad norm: 857.5028686523438\n",
      "Round: 1430, outer loss: 305.63543701171875, grad norm: 763.554443359375\n",
      "Round: 1440, outer loss: 271.66656494140625, grad norm: 719.5441284179688\n",
      "Round: 1450, outer loss: 3.1437008380889893, grad norm: 709.5386962890625\n",
      "Round: 1460, outer loss: 190.82427978515625, grad norm: 558.9964599609375\n",
      "Round: 1470, outer loss: 291.257080078125, grad norm: 531.203369140625\n",
      "Round: 1480, outer loss: 0.46899810433387756, grad norm: 706.0784912109375\n",
      "Round: 1490, outer loss: 37.928985595703125, grad norm: 424.46136474609375\n",
      "Round: 1500, outer loss: 6.415038585662842, grad norm: 504.92626953125\n",
      "Round: 1510, outer loss: 255.32962036132812, grad norm: 507.465576171875\n",
      "Round: 1520, outer loss: 14.331290245056152, grad norm: 703.1455078125\n",
      "Round: 1530, outer loss: 3.6044318675994873, grad norm: 287.03338623046875\n",
      "Round: 1540, outer loss: 2.4214370250701904, grad norm: 344.0482482910156\n",
      "Round: 1550, outer loss: 110.1860122680664, grad norm: 422.93780517578125\n",
      "Round: 1560, outer loss: 12.072355270385742, grad norm: 289.6629943847656\n",
      "Round: 1570, outer loss: 8.586134910583496, grad norm: 418.87762451171875\n",
      "Round: 1580, outer loss: 16.981430053710938, grad norm: 334.78973388671875\n",
      "Round: 1590, outer loss: 16.228548049926758, grad norm: 332.5879821777344\n",
      "Round: 1600, outer loss: 13.39773178100586, grad norm: 310.98272705078125\n",
      "Round: 1610, outer loss: 25.013002395629883, grad norm: 263.9021301269531\n",
      "Round: 1620, outer loss: 2.6601195335388184, grad norm: 284.4941101074219\n",
      "Round: 1630, outer loss: 86.61995697021484, grad norm: 305.52923583984375\n",
      "Round: 1640, outer loss: 9.531166076660156, grad norm: 151.04843139648438\n",
      "Round: 1650, outer loss: 119.76175689697266, grad norm: 366.1372375488281\n",
      "Round: 1660, outer loss: 1.5340334177017212, grad norm: 192.38272094726562\n",
      "Round: 1670, outer loss: 111.01667785644531, grad norm: 455.79962158203125\n",
      "Round: 1680, outer loss: 20.234460830688477, grad norm: 114.56674194335938\n",
      "Round: 1690, outer loss: 111.45575714111328, grad norm: 315.1564025878906\n",
      "Round: 1700, outer loss: 2.5511674880981445, grad norm: 130.26068115234375\n",
      "Round: 1710, outer loss: 0.7229553461074829, grad norm: 115.76864624023438\n",
      "Round: 1720, outer loss: 12.589557647705078, grad norm: 241.93701171875\n",
      "Round: 1730, outer loss: 4.367551326751709, grad norm: 317.0897521972656\n",
      "Round: 1740, outer loss: 0.6988598108291626, grad norm: 141.64419555664062\n",
      "Round: 1750, outer loss: 6.057681083679199, grad norm: 109.48025512695312\n",
      "Round: 1760, outer loss: 0.07144065946340561, grad norm: 114.96158599853516\n",
      "Round: 1770, outer loss: 14.269842147827148, grad norm: 162.59054565429688\n",
      "Round: 1780, outer loss: 0.11976500600576401, grad norm: 155.83062744140625\n",
      "Round: 1790, outer loss: 0.13400156795978546, grad norm: 108.85432434082031\n",
      "Round: 1800, outer loss: 21.496192932128906, grad norm: 164.51515197753906\n",
      "Round: 1810, outer loss: 1.0949820280075073, grad norm: 124.75685119628906\n",
      "Round: 1820, outer loss: 2.33709454536438, grad norm: 86.4377212524414\n",
      "Round: 1830, outer loss: 0.006515225395560265, grad norm: 122.6628189086914\n",
      "Round: 1840, outer loss: 0.18071161210536957, grad norm: 85.73355102539062\n",
      "Round: 1850, outer loss: 1.683554768562317, grad norm: 77.73070526123047\n",
      "Round: 1860, outer loss: 0.005226858891546726, grad norm: 111.34698486328125\n",
      "Round: 1870, outer loss: 0.014911150559782982, grad norm: 112.0660629272461\n",
      "Round: 1880, outer loss: 0.5298516750335693, grad norm: 105.49267578125\n",
      "Round: 1890, outer loss: 0.002340480452403426, grad norm: 105.70561218261719\n",
      "Round: 1900, outer loss: 1.9355857372283936, grad norm: 50.234493255615234\n",
      "Round: 1910, outer loss: 0.15578344464302063, grad norm: 55.21283721923828\n",
      "Round: 1920, outer loss: 9.134541511535645, grad norm: 101.69482421875\n",
      "Round: 1930, outer loss: 2.2404661178588867, grad norm: 50.11750030517578\n",
      "Round: 1940, outer loss: 0.013246304355561733, grad norm: 60.8394775390625\n",
      "Round: 1950, outer loss: 0.2299257218837738, grad norm: 82.26383972167969\n",
      "Round: 1960, outer loss: 0.6596909165382385, grad norm: 61.72764205932617\n",
      "Round: 1970, outer loss: 0.22902509570121765, grad norm: 54.81788635253906\n",
      "Round: 1980, outer loss: 1.276581883430481, grad norm: 76.59160614013672\n",
      "Round: 1990, outer loss: 0.0032812217250466347, grad norm: 55.6828498840332\n",
      "Round: 2000, outer loss: 1061.8524169921875, grad norm: 822.5469360351562\n",
      "Round: 2010, outer loss: 50.49775314331055, grad norm: 3921.3115234375\n",
      "Round: 2020, outer loss: 12984.9384765625, grad norm: 4667.02734375\n",
      "Round: 2030, outer loss: 2127.56982421875, grad norm: 4786.14208984375\n",
      "Round: 2040, outer loss: 977.8270874023438, grad norm: 3672.841064453125\n",
      "Round: 2050, outer loss: 894.0901489257812, grad norm: 2684.1689453125\n",
      "Round: 2060, outer loss: 4506.44921875, grad norm: 1835.4434814453125\n",
      "Round: 2070, outer loss: 366.39501953125, grad norm: 2917.872802734375\n",
      "Round: 2080, outer loss: 4900.34912109375, grad norm: 4414.71875\n",
      "Round: 2090, outer loss: 27.8670711517334, grad norm: 2047.9920654296875\n",
      "Round: 2100, outer loss: 1038.2572021484375, grad norm: 1709.804443359375\n",
      "Round: 2110, outer loss: 521.3319091796875, grad norm: 2358.033203125\n",
      "Round: 2120, outer loss: 4489.0458984375, grad norm: 2407.73828125\n",
      "Round: 2130, outer loss: 1098.461181640625, grad norm: 1605.6854248046875\n",
      "Round: 2140, outer loss: 2374.6318359375, grad norm: 2046.8238525390625\n",
      "Round: 2150, outer loss: 1.8644979000091553, grad norm: 1373.8275146484375\n",
      "Round: 2160, outer loss: 5671.28759765625, grad norm: 1573.7359619140625\n",
      "Round: 2170, outer loss: 3342.60595703125, grad norm: 2284.4033203125\n",
      "Round: 2180, outer loss: 1397.3023681640625, grad norm: 2209.851318359375\n",
      "Round: 2190, outer loss: 17.363441467285156, grad norm: 728.652099609375\n",
      "Round: 2200, outer loss: 768.8092651367188, grad norm: 1331.3126220703125\n",
      "Round: 2210, outer loss: 49.337005615234375, grad norm: 1708.9869384765625\n",
      "Round: 2220, outer loss: 350.1863098144531, grad norm: 834.0570678710938\n",
      "Round: 2230, outer loss: 35.98715591430664, grad norm: 1642.146728515625\n",
      "Round: 2240, outer loss: 1924.3643798828125, grad norm: 1058.55078125\n",
      "Round: 2250, outer loss: 10.259407997131348, grad norm: 1175.777587890625\n",
      "Round: 2260, outer loss: 397.4101257324219, grad norm: 840.383056640625\n",
      "Round: 2270, outer loss: 137.92642211914062, grad norm: 425.2940368652344\n",
      "Round: 2280, outer loss: 1883.5067138671875, grad norm: 1270.699462890625\n",
      "Round: 2290, outer loss: 58.1756591796875, grad norm: 660.327880859375\n",
      "Round: 2300, outer loss: 0.2177344411611557, grad norm: 930.1338500976562\n",
      "Round: 2310, outer loss: 164.96803283691406, grad norm: 622.0038452148438\n",
      "Round: 2320, outer loss: 236.38839721679688, grad norm: 742.8778076171875\n",
      "Round: 2330, outer loss: 420.95458984375, grad norm: 707.439453125\n",
      "Round: 2340, outer loss: 251.0138702392578, grad norm: 750.2783813476562\n",
      "Round: 2350, outer loss: 243.2803497314453, grad norm: 401.3804016113281\n",
      "Round: 2360, outer loss: 11.038137435913086, grad norm: 779.162841796875\n",
      "Round: 2370, outer loss: 0.706564724445343, grad norm: 520.5333862304688\n",
      "Round: 2380, outer loss: 185.6844024658203, grad norm: 550.98828125\n",
      "Round: 2390, outer loss: 144.2786865234375, grad norm: 455.0902099609375\n",
      "Round: 2400, outer loss: 0.9040729403495789, grad norm: 598.2813720703125\n",
      "Round: 2410, outer loss: 277.92999267578125, grad norm: 539.5907592773438\n",
      "Round: 2420, outer loss: 13.974503517150879, grad norm: 714.7051391601562\n",
      "Round: 2430, outer loss: 27.50800895690918, grad norm: 320.2635192871094\n",
      "Round: 2440, outer loss: 188.78561401367188, grad norm: 377.5055847167969\n",
      "Round: 2450, outer loss: 2.1358845233917236, grad norm: 295.6639099121094\n",
      "Round: 2460, outer loss: 163.2299041748047, grad norm: 347.50146484375\n",
      "Round: 2470, outer loss: 75.05814361572266, grad norm: 309.0194396972656\n",
      "Round: 2480, outer loss: 100.40007019042969, grad norm: 439.664794921875\n",
      "Round: 2490, outer loss: 48.634334564208984, grad norm: 269.3082580566406\n",
      "Round: 2500, outer loss: 8.236370086669922, grad norm: 402.33905029296875\n",
      "Round: 2510, outer loss: 7.03934907913208, grad norm: 366.9580993652344\n",
      "Round: 2520, outer loss: 197.4468536376953, grad norm: 415.1231994628906\n",
      "Round: 2530, outer loss: 23.412609100341797, grad norm: 277.2161560058594\n",
      "Round: 2540, outer loss: 6.120755195617676, grad norm: 414.3868713378906\n",
      "Round: 2550, outer loss: 26.557958602905273, grad norm: 195.09280395507812\n",
      "Round: 2560, outer loss: 124.59809112548828, grad norm: 290.0763854980469\n",
      "Round: 2570, outer loss: 1.9017350673675537, grad norm: 309.5164489746094\n",
      "Round: 2580, outer loss: 127.85649871826172, grad norm: 323.12615966796875\n",
      "Round: 2590, outer loss: 83.85803985595703, grad norm: 211.0781707763672\n",
      "Round: 2600, outer loss: 7.032623291015625, grad norm: 247.23184204101562\n",
      "Round: 2610, outer loss: 6.206526279449463, grad norm: 260.27410888671875\n",
      "Round: 2620, outer loss: 30.92105484008789, grad norm: 286.7816467285156\n",
      "Round: 2630, outer loss: 4.820025444030762, grad norm: 207.51243591308594\n",
      "Round: 2640, outer loss: 8.700106620788574, grad norm: 200.85874938964844\n",
      "Round: 2650, outer loss: 31.753019332885742, grad norm: 137.36495971679688\n",
      "Round: 2660, outer loss: 0.635826587677002, grad norm: 184.48529052734375\n",
      "Round: 2670, outer loss: 43.96973419189453, grad norm: 247.03248596191406\n",
      "Round: 2680, outer loss: 51.43633270263672, grad norm: 267.6781005859375\n",
      "Round: 2690, outer loss: 15.062750816345215, grad norm: 216.93655395507812\n",
      "Round: 2700, outer loss: 0.37129801511764526, grad norm: 235.71185302734375\n",
      "Round: 2710, outer loss: 2.536986827850342, grad norm: 182.6006622314453\n",
      "Round: 2720, outer loss: 0.0317370668053627, grad norm: 177.6239013671875\n",
      "Round: 2730, outer loss: 37.16912078857422, grad norm: 155.64382934570312\n",
      "Round: 2740, outer loss: 11.864018440246582, grad norm: 127.18424224853516\n",
      "Round: 2750, outer loss: 3.0122621059417725, grad norm: 120.76530456542969\n",
      "Round: 2760, outer loss: 0.6518012881278992, grad norm: 122.25959777832031\n",
      "Round: 2770, outer loss: 9.104776382446289, grad norm: 221.7166290283203\n",
      "Round: 2780, outer loss: 2.0681521892547607, grad norm: 126.5457992553711\n",
      "Round: 2790, outer loss: 0.03157844394445419, grad norm: 115.8636245727539\n",
      "Round: 2800, outer loss: 6.430568218231201, grad norm: 90.60125732421875\n",
      "Round: 2810, outer loss: 13.364019393920898, grad norm: 206.25355529785156\n",
      "Round: 2820, outer loss: 8.131579399108887, grad norm: 126.92019653320312\n",
      "Round: 2830, outer loss: 10.261170387268066, grad norm: 152.27061462402344\n",
      "Round: 2840, outer loss: 18.900609970092773, grad norm: 166.2947235107422\n",
      "Round: 2850, outer loss: 0.9164980053901672, grad norm: 79.13156127929688\n",
      "Round: 2860, outer loss: 6.983409404754639, grad norm: 155.42527770996094\n",
      "Round: 2870, outer loss: 0.7509229183197021, grad norm: 121.61809539794922\n",
      "Round: 2880, outer loss: 43.027191162109375, grad norm: 152.96295166015625\n",
      "Round: 2890, outer loss: 3.6557111740112305, grad norm: 119.85162353515625\n",
      "Round: 2900, outer loss: 1.8757227659225464, grad norm: 132.79949951171875\n",
      "Round: 2910, outer loss: 18.53292465209961, grad norm: 115.73218536376953\n",
      "Round: 2920, outer loss: 5.955082893371582, grad norm: 113.96492004394531\n",
      "Round: 2930, outer loss: 2.4312736988067627, grad norm: 92.51927947998047\n",
      "Round: 2940, outer loss: 0.0011270670220255852, grad norm: 70.33763122558594\n",
      "Round: 2950, outer loss: 1.4692457914352417, grad norm: 45.81996154785156\n",
      "Round: 2960, outer loss: 0.4543345272541046, grad norm: 33.68058776855469\n",
      "Round: 2970, outer loss: 4.6277055740356445, grad norm: 75.50334930419922\n",
      "Round: 2980, outer loss: 0.6568284034729004, grad norm: 98.84832763671875\n",
      "Round: 2990, outer loss: 0.3672787547111511, grad norm: 45.501529693603516\n"
     ]
    }
   ],
   "source": [
    "g_iterator = CustomTensorIterator([Xg, yg], batch_size=1, shuffle=False, **kwargs)\n",
    "f_iterator = CustomTensorIterator([Xf, yf], batch_size=1, shuffle=False, **kwargs)\n",
    "\n",
    "xs = [Hr.clone().detach().requires_grad_(True)]\n",
    "ys = [wr.clone().detach()]\n",
    "\n",
    "T = n \n",
    "N = 1\n",
    "W = 10 # 50 or 10\n",
    "alpha = 1e-4 \n",
    "beta = 1e-4 \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",
    "for t in range(T): \n",
    "    step_start_time = time.time() \n",
    "    \n",
    "    # get data defining costs f_t and g_t\n",
    "    f_data = next(f_iterator)\n",
    "    g_data = next(g_iterator)\n",
    "    buffer.append((f_data, g_data))\n",
    "    del buffer[:-W]\n",
    "    \n",
    "    ft = outer_func(ys, xs, f_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, g_data, beta, steps=N) # 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(Hr)]\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=2*N, \n",
    "                            fp_map=lambda y, x: map_func(y, x, g_data, beta), \n",
    "                            outer_loss=lambda y, x: outer_func(y, x, f_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}, grad norm: {norm}')\n",
    "    \n",
    "norms_oagd_fp20 = grad_norms_list\n",
    "regret_oagd_fp20 = regret_list \n",
    "rt_oagd_fp20 = running_time "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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3e7djg2DxezsbGRkZJj9b23o6NGjRumuXbvadU0isi6rOAt/XPsDR64eQVJOElLzUlGprd8T/2DvYHRq0Qkd1R3RsUVHRHtGonWeO7QZ11B86hJKLl5E0fnfkXv2LDIbcSU04eEBr+BgeAYHw6tVK3i2bAmv6o9ncLDuT33nv5l17OuCwQARWTRo0CB06dIFp06dAgAsX74cL7/8stEkUVusXbsWFy9eVNKPP/642XyGS3ROnDgRv/zyi031FxYW4v3338eCBQuUeswFA3PmzMFnn30GQPcW4qOPPsLixYttvo/GtmTJEqM195988skGuc7WrVtx//33KysBxcTEYPv27QgLC7O7rqFDhxql9+7di4ceeqjWMoY7XXt7e6Nfv352X5eIbsovz0dybjJO5p7EiZwTOJFzAldLrta5vtZ+rRGrjkX7oPaICmiLtuWBCMmRUF3ORlHiORSdS0bR2XW4kpWFKw68D0MqT0/4RkXBp3VrXQdfrVY6957BwUrnn517+zEYICKLhBB47rnnMG/ePABARkYGXn31Vfz73/+2uY6SkhI8++zNXRL9/f0xd+7cGvlu3Lhh1Pl/8MEH7WrrQw89pAQDqamp2LdvH+666y6jPP3790d8fDwSEhIAAO+88w6mTJli07yExrZz506jtzCRkZFW9zyoi927d2PSpEnKUq7t2rXDjh076rwpW0xMDLp3767Myfj666+tBgPffPONcjxy5Ej4+fnV6dpErqhSW4nTeadxPPs4juccx/Hs43Ue7qMSKkQHRaNrcFd0adEZnTQtEZpViarzl1H0+zkUnd2NovPnkV1cDEfvCqDy8oJP69bwad0avpGRN4/btYNf27bwCglpcmPtmwsGA0RUq1mzZuHzzz9XVvhZsmQJevXqhT/96U9Wy1ZUVGDKlCm4cOGCcu6tt94yOwbdcG8BT09PTJgwwa52duzYEbfddpsy5GTlypU1ggF9+wcMGICysjKUl5dj0qRJ2Lx5Mzp16mTX9RrS559/jvnz56OqSrfjppubGz7//PN67/Vgat++fRg/fryykk+bNm2wY8cOREVF1aveWbNm4cUXXwSgG370xx9/WNwrYu3atUaTyxsi4CFqTiq0FdiXsQ8Hrx7E8ezjSM5NRmll3ZbGbO3XGj1DeqKnujs6F7VAyJVylB4/jfzkZBQkb0BOURFyHNRu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\n",
      "text/plain": [
       "<Figure size 864x504 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "sz=40\n",
    "lw=4\n",
    "plt.plot(regret_dav_list, '-', label='SOBOW-20', linewidth=lw, color=\"tab:green\")\n",
    "plt.plot(norms_oagd_fp20, '-', label='OAGD-20', linewidth=lw, 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,3000000)\n",
    "plt.grid()\n",
    "plt.legend(loc=(0.1,0.7) , fontsize=35, ncol=1) \n",
    "plt.gcf().set_size_inches(12, 7)\n",
    "plt.show() "
   ]
  },
  {
   "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
}
