{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "%matplotlib inline\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import hypergrad as hg\n",
    "import numpy as np\n",
    "import torch.nn.functional as F\n",
    "#from sklearn.datasets import make_spd_matrix as spd\n",
    "import time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.random.manual_seed(0)\n",
    "np.random.seed(0)\n",
    "torch.backends.cudnn.deterministic = True\n",
    "torch.backends.cudnn.benchmark = False"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "p, d, n = 64, 128, 500\n",
    "m = 0.1 \n",
    "\n",
    "H_true = torch.randn((p, d))\n",
    "w_true = torch.randn(d)\n",
    "Xg = torch.randn((n, p))\n",
    "yg = Xg @ H_true @ w_true + m * torch.randn(n)\n",
    "Xf = torch.randn((n, p))\n",
    "yf = Xf @ H_true @ w_true + m * torch.randn(n)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "b = 0.1\n",
    "\n",
    "def inner_func(params, hparams):\n",
    "    H = hparams[0]\n",
    "    w = params[0]\n",
    "    \n",
    "    g =  (0.5 * (torch.norm(Xg @ H @ w - yg))**2) / n + 0.5 * b * (torch.norm(w))**2 \n",
    "    return g #.squeeze()\n",
    "\n",
    "def outer_func(params, hparams):\n",
    "    \n",
    "    H = hparams[0]\n",
    "    w = params[0]\n",
    "    \n",
    "    f = (0.5 * (torch.norm(Xf @ H @ w - yf))**2) / n \n",
    "    \n",
    "    return f #.squeeze()\n",
    "\n",
    "alpha = .001\n",
    "p0 = [torch.randn(d)]\n",
    "hp0 = [torch.randn((p, d))]\n",
    "\n",
    "def map_func(params, hparams):\n",
    "    \n",
    "    g = inner_func(params, hparams)\n",
    "    #inner_losses.append(g.item())\n",
    "    #print(torch.norm(torch.autograd.grad(g, params, create_graph=True)[0]))\n",
    "    \n",
    "    return [params[0] - alpha * torch.autograd.grad(g, params, create_graph=True)[0]]\n",
    "\n",
    "\n",
    "def regressor(params, Z, y):\n",
    "    w = params[0]\n",
    "    loss = (torch.norm(Z @ w - y))**2\n",
    "    return loss\n",
    "\n",
    "def inner_solver(hparams, steps=100, params0=None, optim=None): \n",
    "    \n",
    "    H = hparams[0] \n",
    "    Zg = Xg @ H\n",
    "    \n",
    "    # params = [torch.randn(d).requires_grad_(True)]\n",
    "    params = [p.requires_grad_(True) for p in p0]\n",
    "\n",
    "    for _ in range(steps):\n",
    "        \n",
    "        loss = 0.5 * regressor(params, Zg, yg) / n + 0.5 * b * (torch.norm(params[0]))**2 \n",
    "        params = [params[0] - alpha * torch.autograd.grad(loss, params, create_graph=True)[0]]\n",
    "\n",
    "    return params\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# ESJ"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "outer step=0 (6.85e-02s)(3.91e-02, 2.93e-02) | val loss=508.6379699707031 | hypergrad norm = 3.846e+02\n",
      "outer step=10 (2.64e-02s)(1.26e-02, 1.37e-02) | val loss=5.298474311828613 | hypergrad norm = 4.095e+01\n",
      "outer step=20 (2.67e-02s)(1.27e-02, 1.39e-02) | val loss=1.8429468870162964 | hypergrad norm = 2.248e+01\n",
      "outer step=30 (2.67e-02s)(1.27e-02, 1.38e-02) | val loss=0.8186774253845215 | hypergrad norm = 1.557e+01\n",
      "outer step=40 (2.66e-02s)(1.26e-02, 1.38e-02) | val loss=0.40757960081100464 | hypergrad norm = 1.075e+01\n",
      "outer step=50 (2.65e-02s)(1.26e-02, 1.38e-02) | val loss=0.16796354949474335 | hypergrad norm = 7.255e+00\n",
      "outer step=60 (2.81e-02s)(1.30e-02, 1.49e-02) | val loss=0.08678991347551346 | hypergrad norm = 4.863e+00\n",
      "outer step=70 (2.89e-02s)(1.33e-02, 1.54e-02) | val loss=0.06349124014377594 | hypergrad norm = 4.168e+00\n",
      "outer step=80 (2.81e-02s)(1.42e-02, 1.38e-02) | val loss=0.029139790683984756 | hypergrad norm = 2.747e+00\n",
      "outer step=90 (2.63e-02s)(1.26e-02, 1.36e-02) | val loss=0.0184906255453825 | hypergrad norm = 2.019e+00\n",
      "outer step=100 (2.65e-02s)(1.27e-02, 1.37e-02) | val loss=0.011450412683188915 | hypergrad norm = 1.365e+00\n",
      "outer step=110 (2.65e-02s)(1.26e-02, 1.38e-02) | val loss=0.008765975944697857 | hypergrad norm = 1.072e+00\n",
      "outer step=120 (2.67e-02s)(1.26e-02, 1.39e-02) | val loss=0.005947915371507406 | hypergrad norm = 5.349e-01\n",
      "outer step=130 (2.66e-02s)(1.26e-02, 1.38e-02) | val loss=0.005882582627236843 | hypergrad norm = 5.140e-01\n",
      "outer step=140 (2.66e-02s)(1.26e-02, 1.38e-02) | val loss=0.005854490678757429 | hypergrad norm = 5.353e-01\n",
      "outer step=149 (2.64e-02s)(1.25e-02, 1.38e-02) | val loss=0.005207022652029991 | hypergrad norm = 2.832e-01\n",
      "total time = 4.086423635482788\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "K = 150\n",
    "eval_interval = 10\n",
    "T = 20\n",
    "mu = .01\n",
    "beta = 0.05\n",
    "\n",
    "#hparams = [torch.randn((p, d)).requires_grad_(True)]\n",
    "hparams = [hp.clone() for hp in hp0]\n",
    "hparams = [hp.requires_grad_(True) for hp in hparams]\n",
    "\n",
    "outer_opt = torch.optim.Adam(lr=beta, params=hparams)\n",
    "\n",
    "total_time, val_losses, running_time = 0,  [], []\n",
    "\n",
    "for k in range(K):\n",
    "    \n",
    "    step_start_time = time.time() \n",
    "    inner_losses = []\n",
    "    params = inner_solver(hparams, steps=T)\n",
    "    t1 = time.time() - step_start_time # inner loop time\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    _, cost = hg.hgvzoj(params, hparams, outer_func, inner_solver, mu=mu, T=T, p=1, set_grad=True)\n",
    "    t2 = time.time() - step_start_time - t1 # hypergrad estimation time \n",
    "    val_losses.append(cost.item())\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",
    "    if k % eval_interval == 0 or k == K - 1:\n",
    "        print('outer step={} ({:.2e}s)({:.2e}, {:.2e}) | val loss={} | hypergrad norm = {:.3e}'.format(k, step_time, t1, t2, val_losses[-1], torch.norm(hparams[0].grad)))\n",
    "        \n",
    "print('total time = {}'.format(total_time))\n",
    "\n",
    "plt.title('validation loss')\n",
    "plt.xlabel('outer steps')\n",
    "#plt.plot(val_losses)\n",
    "plt.plot(running_time, val_losses)\n",
    "plt.show()\n",
    "\n",
    "val_zoj = val_losses\n",
    "run_zoj = running_time\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# HOZOG"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "outer step=0 (7.23e-02s)(4.03e-02, 3.18e-02) | val loss=508.6379699707031 | hypergrad norm = 5.972e+00\n",
      "outer step=10 (3.07e-02s)(1.46e-02, 1.60e-02) | val loss=262.81451416015625 | hypergrad norm = 2.106e-01\n",
      "outer step=20 (3.07e-02s)(1.47e-02, 1.59e-02) | val loss=157.8533172607422 | hypergrad norm = 1.245e+00\n",
      "outer step=30 (3.07e-02s)(1.47e-02, 1.59e-02) | val loss=97.42735290527344 | hypergrad norm = 9.865e-01\n",
      "outer step=40 (3.06e-02s)(1.46e-02, 1.58e-02) | val loss=54.96770477294922 | hypergrad norm = 1.041e-01\n",
      "outer step=50 (3.07e-02s)(1.46e-02, 1.59e-02) | val loss=42.21251678466797 | hypergrad norm = 4.353e-01\n",
      "outer step=60 (3.05e-02s)(1.46e-02, 1.58e-02) | val loss=36.51298904418945 | hypergrad norm = 2.037e-01\n",
      "outer step=70 (3.05e-02s)(1.46e-02, 1.58e-02) | val loss=30.039560317993164 | hypergrad norm = 4.330e-02\n",
      "outer step=80 (3.07e-02s)(1.47e-02, 1.59e-02) | val loss=22.680360794067383 | hypergrad norm = 1.141e-01\n",
      "outer step=90 (3.06e-02s)(1.49e-02, 1.56e-02) | val loss=18.543184280395508 | hypergrad norm = 5.133e-01\n",
      "outer step=100 (3.06e-02s)(1.50e-02, 1.55e-02) | val loss=13.889760971069336 | hypergrad norm = 1.962e-01\n",
      "outer step=110 (3.06e-02s)(1.49e-02, 1.55e-02) | val loss=11.552711486816406 | hypergrad norm = 3.052e-03\n",
      "outer step=120 (3.08e-02s)(1.51e-02, 1.56e-02) | val loss=10.157872200012207 | hypergrad norm = 2.357e-01\n",
      "outer step=130 (3.06e-02s)(1.50e-02, 1.55e-02) | val loss=9.026032447814941 | hypergrad norm = 1.042e-01\n",
      "outer step=140 (3.09e-02s)(1.51e-02, 1.57e-02) | val loss=8.223917961120605 | hypergrad norm = 5.951e-02\n",
      "outer step=149 (3.09e-02s)(1.51e-02, 1.56e-02) | val loss=7.5318779945373535 | hypergrad norm = 4.225e-02\n",
      "total time = 4.7006916999816895\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "K = 150\n",
    "eval_interval = 10\n",
    "T = 20\n",
    "mu = 0.01\n",
    "beta = 0.1 # works better than 0.05\n",
    "\n",
    "#hparams = [torch.randn((p, d)).requires_grad_(True)]\n",
    "hparams = [hp.clone() for hp in hp0]\n",
    "hparams = [hp.requires_grad_(True) for hp in hparams]\n",
    "\n",
    "outer_opt = torch.optim.Adam(lr=beta, params=hparams)\n",
    "\n",
    "total_time, val_losses, running_time = 0,  [], []\n",
    "\n",
    "for k in range(K):\n",
    "    \n",
    "    step_start_time = time.time() \n",
    "    inner_losses = []\n",
    "    params = inner_solver(hparams, steps=T)\n",
    "    t1 = time.time() - step_start_time # inner loop time\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    _, cost = hg.hozog(params, hparams, outer_func, inner_solver, mu=mu, T=T, p=1, params0=None, set_grad=True)\n",
    "    t2 = time.time() - step_start_time - t1 # hypergrad estimation time \n",
    "    val_losses.append(cost.item())\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",
    "    if k % eval_interval == 0 or k == K - 1:\n",
    "        print('outer step={} ({:.2e}s)({:.2e}, {:.2e}) | val loss={} | hypergrad norm = {:.3e}'.format(k, step_time, t1, t2, val_losses[-1], torch.norm(hparams[0].grad)))\n",
    "\n",
    "\n",
    "print('total time = {}'.format(total_time))\n",
    "\n",
    "plt.title('validation loss')\n",
    "plt.xlabel('outer steps')\n",
    "#plt.plot(val_losses)\n",
    "plt.plot(running_time, val_losses)\n",
    "plt.show()\n",
    "\n",
    "val_hozog = val_losses\n",
    "run_hozog = running_time\n",
    "    "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# AID-FP"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "outer step=0 (9.98e-02s)(5.17e-02, 4.80e-02) | val loss=508.6379699707031 | hypergrad norm = 2.435e+02\n",
      "outer step=10 (3.70e-02s)(1.66e-02, 2.03e-02) | val loss=10.422000885009766 | hypergrad norm = 1.657e+01\n",
      "outer step=20 (3.69e-02s)(1.66e-02, 2.02e-02) | val loss=3.3422176837921143 | hypergrad norm = 9.267e+00\n",
      "outer step=30 (3.71e-02s)(1.68e-02, 2.02e-02) | val loss=1.3675638437271118 | hypergrad norm = 4.755e+00\n",
      "outer step=40 (3.69e-02s)(1.66e-02, 2.01e-02) | val loss=0.7099811434745789 | hypergrad norm = 2.909e+00\n",
      "outer step=50 (3.69e-02s)(1.65e-02, 2.03e-02) | val loss=0.3342474699020386 | hypergrad norm = 1.416e+00\n",
      "outer step=60 (3.70e-02s)(1.66e-02, 2.03e-02) | val loss=0.22853603959083557 | hypergrad norm = 1.001e+00\n",
      "outer step=70 (3.69e-02s)(1.66e-02, 2.02e-02) | val loss=0.15343041718006134 | hypergrad norm = 6.573e-01\n",
      "outer step=80 (3.69e-02s)(1.66e-02, 2.02e-02) | val loss=0.11572324484586716 | hypergrad norm = 3.779e-01\n",
      "outer step=90 (3.70e-02s)(1.66e-02, 2.03e-02) | val loss=0.09151680022478104 | hypergrad norm = 1.995e-01\n",
      "outer step=100 (2.53e-02s)(1.16e-02, 1.35e-02) | val loss=0.07587730884552002 | hypergrad norm = 1.548e-01\n",
      "outer step=110 (2.53e-02s)(1.16e-02, 1.35e-02) | val loss=0.06420482695102692 | hypergrad norm = 1.034e-01\n",
      "outer step=120 (2.53e-02s)(1.17e-02, 1.35e-02) | val loss=0.05535510554909706 | hypergrad norm = 6.279e-02\n",
      "outer step=130 (2.88e-02s)(1.32e-02, 1.54e-02) | val loss=0.04843415692448616 | hypergrad norm = 4.388e-02\n",
      "outer step=140 (2.54e-02s)(1.16e-02, 1.36e-02) | val loss=0.04286954179406166 | hypergrad norm = 3.631e-02\n",
      "outer step=149 (2.66e-02s)(1.18e-02, 1.47e-02) | val loss=0.03871908038854599 | hypergrad norm = 3.103e-02\n",
      "total time = 5.0319504737854\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "K = 150\n",
    "eval_interval = 10\n",
    "T = 20\n",
    "beta = 0.05\n",
    "\n",
    "#hparams = [torch.randn((p, d)).requires_grad_(True)]\n",
    "hparams = [hp.clone() for hp in hp0]\n",
    "hparams = [hp.requires_grad_(True) for hp in hparams]\n",
    "\n",
    "outer_opt = torch.optim.Adam(lr=beta, params=hparams)\n",
    "\n",
    "total_time, val_losses, running_time = 0,  [], []\n",
    "\n",
    "for k in range(K):\n",
    "    \n",
    "    step_start_time = time.time() \n",
    "    inner_losses = []\n",
    "    params = inner_solver(hparams, steps=T)\n",
    "    t1 = time.time() - step_start_time # inner loop time\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    _, cost = hg.fixed_point(params, hparams, T, map_func, outer_func, set_grad=True)\n",
    "    t2 = time.time() - step_start_time - t1 # hypergrad estimation time \n",
    "    val_losses.append(cost.item())\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",
    "    if k % eval_interval == 0 or k == K - 1:\n",
    "        print('outer step={} ({:.2e}s)({:.2e}, {:.2e}) | val loss={} | hypergrad norm = {:.3e}'.format(k, step_time, t1, t2, val_losses[-1], torch.norm(hparams[0].grad)))\n",
    "        \n",
    "print('total time = {}'.format(total_time))\n",
    "\n",
    "plt.title('validation loss')\n",
    "plt.xlabel('outer steps')\n",
    "#plt.plot(val_losses)\n",
    "plt.plot(running_time, val_losses)\n",
    "plt.show()\n",
    "\n",
    "val_fp = val_losses\n",
    "run_fp = running_time\n",
    "    \n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# AID-CG"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "outer step=0 (1.06e-01s)(6.31e-02, 4.28e-02) | val loss=508.6379699707031 | hypergrad norm = 2.183e+02\n",
      "outer step=10 (4.33e-02s)(1.75e-02, 2.57e-02) | val loss=7.023938179016113 | hypergrad norm = 2.232e+00\n",
      "outer step=20 (3.79e-02s)(1.28e-02, 2.49e-02) | val loss=2.2614846229553223 | hypergrad norm = 6.525e-01\n",
      "outer step=30 (4.35e-02s)(1.77e-02, 2.57e-02) | val loss=0.9717137217521667 | hypergrad norm = 2.975e-01\n",
      "outer step=40 (2.86e-02s)(1.31e-02, 1.54e-02) | val loss=0.4653817415237427 | hypergrad norm = 1.323e-01\n",
      "outer step=50 (2.57e-02s)(1.17e-02, 1.39e-02) | val loss=0.24654822051525116 | hypergrad norm = 7.641e-02\n",
      "outer step=60 (2.56e-02s)(1.18e-02, 1.37e-02) | val loss=0.1509014368057251 | hypergrad norm = 4.090e-02\n",
      "outer step=70 (2.63e-02s)(1.17e-02, 1.45e-02) | val loss=0.11597567796707153 | hypergrad norm = 2.923e-02\n",
      "outer step=80 (2.89e-02s)(1.32e-02, 1.56e-02) | val loss=0.09113471955060959 | hypergrad norm = 1.911e-02\n",
      "outer step=90 (2.61e-02s)(1.18e-02, 1.42e-02) | val loss=0.07688280194997787 | hypergrad norm = 1.586e-02\n",
      "outer step=100 (2.91e-02s)(1.32e-02, 1.58e-02) | val loss=0.06535600125789642 | hypergrad norm = 1.211e-02\n",
      "outer step=110 (2.58e-02s)(1.17e-02, 1.39e-02) | val loss=0.057658348232507706 | hypergrad norm = 9.949e-03\n",
      "outer step=120 (2.60e-02s)(1.18e-02, 1.41e-02) | val loss=0.05160854011774063 | hypergrad norm = 8.543e-03\n",
      "outer step=130 (2.58e-02s)(1.17e-02, 1.39e-02) | val loss=0.0468650758266449 | hypergrad norm = 7.588e-03\n",
      "outer step=140 (2.58e-02s)(1.17e-02, 1.39e-02) | val loss=0.04297155141830444 | hypergrad norm = 6.894e-03\n",
      "outer step=149 (2.60e-02s)(1.18e-02, 1.40e-02) | val loss=0.03987754508852959 | hypergrad norm = 6.314e-03\n",
      "total time = 4.500315427780151\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "K = 150\n",
    "eval_interval = 10\n",
    "T = 20\n",
    "beta = 0.05\n",
    "\n",
    "#hparams = [torch.randn((p, d)).requires_grad_(True)]\n",
    "hparams = [hp.clone() for hp in hp0]\n",
    "hparams = [hp.requires_grad_(True) for hp in hparams]\n",
    "\n",
    "outer_opt = torch.optim.Adam(lr=beta, params=hparams)\n",
    "\n",
    "total_time, val_losses, running_time = 0,  [], []\n",
    "\n",
    "for k in range(K):\n",
    "    \n",
    "    step_start_time = time.time() \n",
    "    inner_losses = []\n",
    "    params = inner_solver(hparams, steps=T)\n",
    "    t1 = time.time() - step_start_time # inner loop time\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    _, cost = hg.CG(params, hparams, T, map_func, outer_func, set_grad=True)\n",
    "    t2 = time.time() - step_start_time - t1 # hypergrad estimation time \n",
    "    val_losses.append(cost.item())\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",
    "    if k % eval_interval == 0 or k == K - 1:\n",
    "        print('outer step={} ({:.2e}s)({:.2e}, {:.2e}) | val loss={} | hypergrad norm = {:.3e}'.format(k, step_time, t1, t2, val_losses[-1], torch.norm(hparams[0].grad)))\n",
    "\n",
    "\n",
    "print('total time = {}'.format(total_time))\n",
    "\n",
    "plt.title('validation loss')\n",
    "plt.xlabel('outer steps')\n",
    "#plt.plot(val_losses)\n",
    "plt.plot(running_time, val_losses)\n",
    "plt.show()\n",
    "\n",
    "val_cg = val_losses\n",
    "run_cg = running_time\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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5s2bx5QpUwAICwtjwYIFtGvXrs7P8ne8SUlJnoRzy5YtnHbaabXep2fPnvTs2bNK29SpUw8rpsagokEtVJf4xq1UW5Ze89YoflnHWZNW7WD4LBj5JrRyJ7iOUpj7N+3ZKSIiItLIzjjjDM/xN998Q1lZWb2uLysrq7JNyZlnnllr340bNzJ69GjP+TPPPHPQhK4mp59+OjabK3XasmWLZ0uX+li9erXnuHq83udr1qyp972DmRLOFqpTo1Sq9S4cVMs6Tl9tjVJX3QfDmGUQ446tNA/eGAb7tjduHCIiIiIt2IknnkiHDh0AyM3N5Spg518AACAASURBVL333qvX9YsWLSI3NxeAjh07csIJJ9TYLycnh8suu4yCAlfdkuuuu46bb7653vHGxcVV2YbljTfeqNf1X3/9NZs3bwYgPDycs88+u8rr559fuczrrbfeqncCHsyUcLZQ3oWD/JVwhiZ6rePMqHmEMy0/jUYX1xmufhvCW7vO8zNdxYQKsw9+nYiIiIj4hM1mY+zYsZ7z//znP5SXl9fp2vLych555BHP+d///neMMQf0syyLUaNG8dtvvwHQq1cvnn/++cOO+Y477vAcT548ucqazEOZNGmS53jUqFEHrHEdOHCgZ2uWzMxMXnjhhcOOM9go4WyhujT2XpzpNW+NkpYXgIQToMPxrum1IWGu86zf4M0RUOqf/xYiIiIiUtXtt99OR3eNjx9++IH77ruvTtfdf//9/PjjjwAkJiZy22231djvkUce8YycxsfH88477xAREXHY8Y4cOZLjjz8ecO2beeONN+JwOA553TPPPMPixYsBiI6O5h//+McBfWw2Gw8//LDn/N577202U2uVcLZQ3lNqt2f5aWsU7ym1mZme4+SYAKzhrMkRZ8NfXwDc34ilfQ0LrgdH3b5dExEREZHD16ZNG958803P2sgpU6Zw44031jpymJuby80338zkyZMBV5I2e/bsGivifvzxx/zrX/+q0q+he1OGhoby1ltvERXl+nf0/Pnz+etf/0qG10w+byUlJfzrX//irrvu8rRNnz6dI488ssb+w4cP5/bbbwegqKiIAQMGMHXqVIqLiw8a17Zt20hPTz9on0BSldoWyntK7Z/7inA6LWy2A6ciNIQ9yXsNZ+UfxKToJAwGC4vdhbspdZQSVjHS2NiOvwzyd7sq2AL8ugQ+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pFZJnuPMgkzKneU+f3ajSzkVbl4JR5xT2fbbUnihv6uokIiIiIhIC6GEUzDG0Cnev+s4Q2uZUhsRGkHbyLYAOCwHmQWZB1zbJEW3hasXwlnjKtv2b3cVE/rhzcDFJSIiIiLSiJRwCgBdErzXcfq+Um3VrVEyqrzmPa22SRcOqi4kFP4yCa54A8JiXG3lxbDoVvjgbigvCWR0IiIiIodt9OjRGGMwxmCz2di6tW7V+SdNmuS5btKkSYfs4/1js9lo3bo1nTp14uSTT+bqq69m8uTJbNy40YfvrHaWZbFkyRLuuusuevfuTXJyMhEREcTExNClSxcGDx7Mo48+yh9//FHnezqdTj788EPGjRtHamoqnTp1IjIyksjISBITE+nbty+33nor8+fPp7CwaW67p4RTgKqVandkFfn8/gdLOJNjmsHWKAdz3MVw02fQrntl2zczYdZFkNMM36+IiIg0awUFBcyfP99zblkWr7zyit+fa1kWeXl5pKWlsX79embPns348eM54YQTOOOMM1i8eLHfnr1kyRJOPvlkBg0axP/+9z++/fZb0tPTKSkpIT8/nx07drB48WIeeOABjj76aIYMGcLPP/980Hu+9dZb9OjRgyFDhjB16lS+/vpr0tLSKC4upri4mMzMTNauXcuMGTMYMWIE7du3Z/z48ezevdtv79MfQgMdgAQH70q12/08wlmekYFlWRhjgKqVatPymnjhoNq0PQbGfALvjYWNC11tO7+B/3cODJ9Vdb2niIiISBBbsGABeXl5VdpeffVVz8ikL11xxRWMHDnSc15SUsK+ffv4888/Wbt2LV988QWlpaWsXr2awYMHc+ONNzJt2jTsdrvPYvjPf/7DxIkTPYU14+PjGThwIKmpqbRv3x6Hw0FmZiZfffUVy5cvJy8vjw8//JD8/HxWrFhxwP3Ky8u56667mDZtmqetXbt2nHfeeaSmptK2bVvCwsLYu3cvv//+OytWrODHH3+koKCAyZMn43A4mDp1qs/en78p4RQAusR7Var1wxpOW2wsJioKq7AQZ2EhztxcQmJjgWZWqfZgwlvBsFmQ0geWTgDLAYV74bVLXVNvz7jDVelWREREJIjNmjULALvdzogRI5g9ezbbt2/n008/5bzzzvPps7p3787QoUNrfT0jI4MnnniC//3vf1iWxYsvvojT6eSll2rcRbHennzySSZMmACAzWbjgQce4L777qNVq1Y19i8oKOC5557j8ccfr/Wed955J9OnTwcgPDycRx99lNtuu42IiIhar/nll1+YMmUKM2fObMC7CQxNqRXA/3txGmOwd+zoOfeeVltlL87mOKXWmzFw+u1w7XsQ3c7VZjlh2UR4axSU5B38ehEREZEA2rJlCytXrgRg4MCB3H333Z7XKhLRxpSYmMiUKVOYP3++Z3R15syZzJkzp8H3Xrt2Lf/85z8B179l58yZw8MPP1xrsgkQHR3N/fffz/fff0+fPn0OeH3+/PmeZDMiIoJPPvmEu++++6DJJkC3bt2YMWMG69at48QTT2zAu2p8SjgFgKS4CEJtrj+ku/NKKCp1+PwZ9loq1abEeBUNymvmCWeFrme5tk5JSa1s2/QevHguUQV/Bi4uERERkYN45ZVXPFNLR40axSmnnMLxxx8PwDvvvENOTk5A4rr88su57777POf//ve/cTqdDbrnxIkTcThc/yYeO3YsI0aMqPO1Xbp04cknn6zS5nQ6mThxouf88ccf58wzz6xXTL169WL06NH1uibQlHAKAKEhNpLbRHrO/THK6b0XZ1lmZcLZIaoDocY1uzurOIuict8XLQpKrZPgug8h9abKtr2/csp398DGRYGLS0RERKQGlmXx6quvAhAXF8fFF18MwDXXXANAUVER8+bNC1h89957L5GRrn/Pbtq0idWrVx/2vTZt2sTSpUsB19ThBx54oMHxLVmyhM2bNwPQoUMHbrnllgbfsylQwike/p5WG1qtcFCFEFsIHaMrp9u2mFFOgNAwuOhJuOwFCHX9BRnqKIb517rWeTrKAxygiIiIiMsnn3zCjh07ABg+fDjh4eEAXH311dhsrrQiENNqKyQkJHDhhRd6zj///PPDvteyZcs8xxdccAEdOnRoUGzV7zlixAjCwsIafM+mQAmneFSpVJvlj0q1SZ5j7ym10AK2RjmUk6+AMcugTdfKtq+egdeHQv6egIUlIiIiUsE7mRw1apTnODk5mQEDBgCwZs0aNm3a1OixVTjttNM8x19//fVh32fVqlWe49NPP71BMfnznk2BEk7x6OK1F+ef/phSm1hz0SBoQZVqD6bjiXDTCvYmeC0w3/aFa+uUPw//L0wRERGRhsrJyWHhQtfWbkccccQBaw+9E9DG2JOzNl27dvUc79lz+F/a79xZOQBy9NFHNyQkj/T0dM/xkUce6ZN7NgXaFkU8qu7F6Y+E02sNZ/WE07twUEsc4awQ2YafTvg/+tu+gc8eBSzIS4dZg2DQY9B7tLZOERERqebEV5tW1U5f2nDthkZ5zpw5cygqctXZuPrqqw/Yb/Pyyy/ntttuo6CggNdff51HH32UkJCQRonNW5s2bTzHWVlZh30f72vj4uIaFNPh3HPMmDEH3QJl69atVZLrYKYRTvHo7Oe9OEO9tkUp370bq7xyfWKVrVFa0hrOmhgb9LsP/rYAItx/GTnL4MPxsOhWKM4NbHwiIiLS4nhPp60oEuQtOjqayy67DHDtjfnRRx81WmzevCvTVk+KJTA0wikenb2m1KbtK8LhtAix+e4Pqi08nJC2bXHs3QsOB+V79nhGPb0TzhY7pba6Y/4CN38O866BzPWuth/nwNaVMHgydBsY2PhERESkRfj5559Zt24dAH379uWYY46psd+oUaN44403AFeCOmTIkEaLscL+/fs9x/Hx8VVeW7p0KYWFtQ+qDB061HOckJBQ4z0bIiEhwTNV91D3HDt27AH//R588EE2btzok1gakxJO8WgVHkpCdBhZBaWUOpxk5haTHBd56AvrwZ6Y6Eo4cU2rrSnh3Jm/E8uy9K0UuIoIjV7qGt38YbarLXcnzLkCThgGgx6H6LYBDVFERCTQGmta6eHKy8sjJiYm0GEctkONblY477zzSE5OZufOnbz//vvs3buXtm0b998p27Zt8xy3a9euyms33XQT27dvr/Xaiv1FAZKSKotd/vHHHz6JLSkpyZNwbtmypUqBo+p69uxJz549q7RNnTrVJ3E0Nk2plSq8Rzn9UqnWa1qtd6Xa+Ih4It3bghSUFZBTEphNg4OSPRIunQbDXoYor7+0f1oAz/WB9W+B11+QIiIiIr5SXl7uGbUEuP322zHG1PgTEhLiSajKysqYPXt2o8e7du1az3Fqauph38e7KNJXX33VoJhquueaNWt8cs+mQAmnVOFdOMgvlWqTvAsHVVbqMsYcMMopXoyBEy6HsV/DyVdWthdlwzs3wuzhsP/PwMUnIiIizdJHH31EZmbmYV3b2HtyZmVlsXTpUs95v379qry+bds2LMuq9cfbBRdc4DletmwZu3btanB8559/vuf4rbfeoqysrMH3bAo0pVaq6FJlL04/FA7yqlRbnlH1L6+UVin8vv93wLWO8/i2x/v8+U1eVDxcNsM1nfaDuyDHnWT+vgym94W/THJVsrXpuyQRERFpOO+k8dprr61TZdQ333yT3377jR9//JHvv/+eXr16+THCSk888YSnkm6PHj3o27fvYd/ruOOO4/zzz2fZsmWUlpby6KOP8r///a9B8Q0cOJDu3buzefNmMjMzeeGFF7j99tsbdM+mQAmnVNE5obJSrX+2RqmcD199a5TkGK/CQXkqHHRQx/wFblsNn/wb1r0AWFCaD4vvgZ/ehoufgXbHBjpKERERacL27t3LBx98AEBMTAzPP/88kZGHru/Rpk0b7rrrLsCVsDZGwvn222/z5JNPes4nTJiArYFfwD/00EN8+umnOBwOnn32Wc466yyGDx9ep2u3b9/Oc889VyUmm83Gww8/zIgRIwC49957OfXUUxuUGDcFGgaRKvw+pTbRaw1n9YRTU2rrJzwGLnoCbvgY2nollztWw4wzYeVT4GgZUzVERETE99544w3PtM/LL7+8TskmwJVXXkloqGtc680336S0tNRvMWZmZjJ+/HiGDx/umRY7evRoRo4c2eB7n3766fznP/8BXAWFRo4cyaRJkygoqL3OSWFhIU888QS9evXi66+/PuD14cOHe0Y1i4qKGDBgAFOnTqW4uPigsWzbto309PSD9glWGuGUKrok+HdKrT3Rew2nEk6f6Hwa3PKlK8H8cjI4y8FRCp/+GzYugkufhaTGmcoiIiIizUddq9NW1759ey644AIWL15MVlYW7733HsOGDTusGDZv3syiRYs856Wlpezfv58dO3awZs0avvjiiyoJ7ZgxY5g+ffphPasm999/P8XFxTz88MM4nU4eeughnnvuOQYOHEhqairt2rXD6XSSmZnJV199xfLly8nNPfie6VOnTsWyLKZPn05xcTHjxo3jkUce4fzzzyc1NZW2bdsSGRlJXl4eW7ZsYdWqVaxcuZJy9x72bdq0ISoq6qDPCCZKOKWK9jHhhIfaKCl3klNURk5hGbFRdp/dPyQhAWO3Y5WV4czJwVlQgC3aNY03JSbF008JZz2FhsO5D0CPS+G9sZD+vat91wZ48Vw4fSz0/yeENZ2/nERERCRwvvvuO9avd+0DnpycTP/+/et1/TXXXMPixYsBV+J6uAnnvHnzmDdv3iH79e3blwkTJnDR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\n",
      "text/plain": [
       "<Figure size 1080x576 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "s, s2 = 20, 30 \n",
    "lw = 3\n",
    "\n",
    "plt.plot(run_zoj[:s2], val_zoj[:s2], '-', label='ESJ', linewidth=lw)\n",
    "plt.plot(run_hozog[:s2], val_hozog[:s2], '-', label='HOZOG', linewidth=lw)\n",
    "plt.plot(run_cg[:s], val_cg[:s], '-', label='AID-CG', linewidth=lw)\n",
    "plt.plot(run_fp[:s], val_fp[:s], '-', label='AID-FP', linewidth=lw)\n",
    "plt.xlabel('Running time', fontsize=24)\n",
    "plt.ylabel('Outer loss', fontsize=24)\n",
    "plt.xticks(fontsize=20)\n",
    "plt.yticks(fontsize=20)\n",
    "plt.grid()\n",
    "plt.legend(loc='upper right', fontsize=30)\n",
    "plt.gcf().set_size_inches(15, 8)\n",
    "#plt.savefig('hrl_T20.pdf', bbox_inches='tight')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Higher Dimension"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [],
   "source": [
    "p, d, n = 128, 256, 1000\n",
    "m = 0.1 \n",
    "\n",
    "H_true = torch.randn((p, d))\n",
    "w_true = torch.randn(d)\n",
    "Xg = torch.randn((n, p))\n",
    "yg = Xg @ H_true @ w_true + m * torch.randn(n)\n",
    "Xf = torch.randn((n, p))\n",
    "yf = Xf @ H_true @ w_true + m * torch.randn(n)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [],
   "source": [
    "b = 0.1\n",
    "\n",
    "def inner_func(params, hparams):\n",
    "    H = hparams[0]\n",
    "    w = params[0]\n",
    "    \n",
    "    g =  (0.5 * (torch.norm(Xg @ H @ w - yg))**2) / n + 0.5 * b * (torch.norm(w))**2 \n",
    "    return g #.squeeze()\n",
    "\n",
    "def outer_func(params, hparams):\n",
    "    \n",
    "    H = hparams[0]\n",
    "    w = params[0]\n",
    "    \n",
    "    f = (0.5 * (torch.norm(Xf @ H @ w - yf))**2) / n \n",
    "    \n",
    "    return f #.squeeze()\n",
    "\n",
    "alpha = .001\n",
    "p0 = [torch.randn(d)]\n",
    "hp0 = [torch.randn((p, d))]\n",
    "\n",
    "def map_func(params, hparams):\n",
    "    \n",
    "    g = inner_func(params, hparams)\n",
    "    #inner_losses.append(g.item())\n",
    "    #print(torch.norm(torch.autograd.grad(g, params, create_graph=True)[0]))\n",
    "    \n",
    "    return [params[0] - alpha * torch.autograd.grad(g, params, create_graph=True)[0]]\n",
    "\n",
    "\n",
    "def regressor(params, Z, y):\n",
    "    w = params[0]\n",
    "    loss = (torch.norm(Z @ w - y))**2\n",
    "    return loss\n",
    "\n",
    "def inner_solver(hparams, steps=100, params0=None, optim=None): \n",
    "    \n",
    "    H = hparams[0] \n",
    "    Zg = Xg @ H\n",
    "    \n",
    "    # params = [torch.randn(d).requires_grad_(True)]\n",
    "    params = [p.requires_grad_(True) for p in p0]\n",
    "\n",
    "    for _ in range(steps):\n",
    "        \n",
    "        loss = 0.5 * regressor(params, Zg, yg) / n + 0.5 * b * (torch.norm(params[0]))**2 \n",
    "        params = [params[0] - alpha * torch.autograd.grad(loss, params, create_graph=True)[0]]\n",
    "\n",
    "    return params\n",
    "\n",
    "\n",
    "# def inner_solver(hparams, steps=100, params0=None, optim=None):\n",
    "\n",
    "#     # params = [torch.randn(d).requires_grad_(True)]\n",
    "#     params = [p.requires_grad_(True) for p in p0]\n",
    "\n",
    "#     for _ in range(steps):\n",
    "#         params = map_func(params, hparams)\n",
    "\n",
    "#     return params\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# ESJ"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "outer step=0 (6.86e-02s)(3.77e-02, 3.04e-02) | val loss=1197.4111328125 | hypergrad norm = 8.699e+02\n",
      "outer step=10 (2.80e-02s)(1.30e-02, 1.48e-02) | val loss=8.839682579040527 | hypergrad norm = 6.224e+01\n",
      "outer step=20 (2.82e-02s)(1.30e-02, 1.49e-02) | val loss=2.408905029296875 | hypergrad norm = 3.425e+01\n",
      "outer step=30 (2.80e-02s)(1.30e-02, 1.48e-02) | val loss=0.8888125419616699 | hypergrad norm = 2.023e+01\n",
      "outer step=40 (2.80e-02s)(1.30e-02, 1.48e-02) | val loss=0.38051822781562805 | hypergrad norm = 1.301e+01\n",
      "outer step=50 (2.80e-02s)(1.29e-02, 1.48e-02) | val loss=0.2949046492576599 | hypergrad norm = 1.196e+01\n",
      "outer step=60 (2.79e-02s)(1.29e-02, 1.47e-02) | val loss=0.23724213242530823 | hypergrad norm = 1.098e+01\n",
      "outer step=70 (2.81e-02s)(1.30e-02, 1.49e-02) | val loss=0.4177219271659851 | hypergrad norm = 1.435e+01\n",
      "outer step=80 (2.80e-02s)(1.30e-02, 1.48e-02) | val loss=1.9105294942855835 | hypergrad norm = 3.294e+01\n",
      "outer step=90 (2.79e-02s)(1.29e-02, 1.48e-02) | val loss=0.6200764775276184 | hypergrad norm = 1.843e+01\n",
      "outer step=100 (2.81e-02s)(1.31e-02, 1.48e-02) | val loss=0.26972267031669617 | hypergrad norm = 1.180e+01\n",
      "outer step=110 (2.80e-02s)(1.29e-02, 1.48e-02) | val loss=0.13050037622451782 | hypergrad norm = 8.230e+00\n",
      "outer step=120 (2.79e-02s)(1.30e-02, 1.47e-02) | val loss=0.1024964302778244 | hypergrad norm = 7.155e+00\n",
      "outer step=130 (2.80e-02s)(1.30e-02, 1.48e-02) | val loss=0.07978766411542892 | hypergrad norm = 6.047e+00\n",
      "outer step=140 (2.81e-02s)(1.30e-02, 1.49e-02) | val loss=0.036162905395030975 | hypergrad norm = 4.045e+00\n",
      "outer step=149 (2.79e-02s)(1.29e-02, 1.48e-02) | val loss=0.3495720326900482 | hypergrad norm = 1.425e+01\n",
      "total time = 4.277534008026123\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "K = 150\n",
    "eval_interval = 10\n",
    "T = 20\n",
    "#K = 10 \n",
    "mu = .01\n",
    "beta = 0.05\n",
    "\n",
    "#hparams = [torch.randn((p, d)).requires_grad_(True)]\n",
    "hparams = [hp.clone() for hp in hp0]\n",
    "hparams = [hp.requires_grad_(True) for hp in hparams]\n",
    "\n",
    "outer_opt = torch.optim.Adam(lr=beta, params=hparams)\n",
    "\n",
    "total_time, val_losses, running_time = 0,  [], []\n",
    "\n",
    "for k in range(K):\n",
    "    \n",
    "    step_start_time = time.time() \n",
    "    inner_losses = []\n",
    "    params = inner_solver(hparams, steps=T)\n",
    "    t1 = time.time() - step_start_time # inner loop time\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    _, cost = hg.hgvzoj(params, hparams, outer_func, inner_solver, mu=mu, T=T, p=1, set_grad=True)\n",
    "    t2 = time.time() - step_start_time - t1 # hypergrad estimation time \n",
    "    val_losses.append(cost.item())\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",
    "    if k % eval_interval == 0 or k == K - 1:\n",
    "        print('outer step={} ({:.2e}s)({:.2e}, {:.2e}) | val loss={} | hypergrad norm = {:.3e}'.format(k, step_time, t1, t2, val_losses[-1], torch.norm(hparams[0].grad)))\n",
    "        \n",
    "print('total time = {}'.format(total_time))\n",
    "\n",
    "plt.title('validation loss')\n",
    "plt.xlabel('outer steps')\n",
    "#plt.plot(val_losses)\n",
    "plt.plot(running_time, val_losses)\n",
    "plt.show()\n",
    "\n",
    "val_zoj = val_losses\n",
    "run_zoj = running_time\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# HOZOG "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "outer step=0 (7.58e-02s)(4.00e-02, 3.53e-02) | val loss=1197.4111328125 | hypergrad norm = 7.935e-01\n",
      "outer step=10 (2.79e-02s)(1.29e-02, 1.47e-02) | val loss=523.5926513671875 | hypergrad norm = 4.828e+00\n",
      "outer step=20 (2.79e-02s)(1.30e-02, 1.47e-02) | val loss=185.0880584716797 | hypergrad norm = 1.591e+00\n",
      "outer step=30 (2.80e-02s)(1.30e-02, 1.47e-02) | val loss=99.13282775878906 | hypergrad norm = 1.877e-01\n",
      "outer step=40 (2.79e-02s)(1.30e-02, 1.47e-02) | val loss=63.260467529296875 | hypergrad norm = 3.010e-01\n",
      "outer step=50 (2.79e-02s)(1.30e-02, 1.47e-02) | val loss=48.292152404785156 | hypergrad norm = 5.878e-01\n",
      "outer step=60 (2.75e-02s)(1.30e-02, 1.43e-02) | val loss=38.41683578491211 | hypergrad norm = 5.302e-02\n",
      "outer step=70 (2.66e-02s)(1.25e-02, 1.39e-02) | val loss=33.170833587646484 | hypergrad norm = 2.548e-01\n",
      "outer step=80 (2.66e-02s)(1.25e-02, 1.39e-02) | val loss=29.22417640686035 | hypergrad norm = 1.431e-01\n",
      "outer step=90 (2.67e-02s)(1.25e-02, 1.39e-02) | val loss=23.805339813232422 | hypergrad norm = 1.781e-01\n",
      "outer step=100 (2.67e-02s)(1.25e-02, 1.39e-02) | val loss=19.665494918823242 | hypergrad norm = 6.542e-02\n",
      "outer step=110 (2.67e-02s)(1.26e-02, 1.39e-02) | val loss=16.744539260864258 | hypergrad norm = 1.080e-01\n",
      "outer step=120 (3.18e-02s)(1.44e-02, 1.72e-02) | val loss=14.831977844238281 | hypergrad norm = 1.195e-01\n",
      "outer step=130 (2.91e-02s)(1.32e-02, 1.56e-02) | val loss=13.956759452819824 | hypergrad norm = 7.296e-02\n",
      "outer step=140 (2.89e-02s)(1.31e-02, 1.55e-02) | val loss=12.396344184875488 | hypergrad norm = 1.173e-02\n",
      "outer step=149 (2.67e-02s)(1.26e-02, 1.39e-02) | val loss=11.22706413269043 | hypergrad norm = 1.114e-01\n",
      "total time = 4.219234466552734\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "K = 150\n",
    "eval_interval = 10\n",
    "T = 20\n",
    "#K = 10 \n",
    "mu = 0.01\n",
    "beta = 0.1\n",
    "\n",
    "#hparams = [torch.randn((p, d)).requires_grad_(True)]\n",
    "hparams = [hp.clone() for hp in hp0]\n",
    "hparams = [hp.requires_grad_(True) for hp in hparams]\n",
    "\n",
    "outer_opt = torch.optim.Adam(lr=beta, params=hparams)\n",
    "\n",
    "total_time, val_losses, running_time = 0,  [], []\n",
    "\n",
    "for k in range(K):\n",
    "    \n",
    "    step_start_time = time.time() \n",
    "    inner_losses = []\n",
    "    params = inner_solver(hparams, steps=T)\n",
    "    t1 = time.time() - step_start_time # inner loop time\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    _, cost = hg.hozog(params, hparams, outer_func, inner_solver, mu=mu, T=T, p=1, params0=None, set_grad=True)\n",
    "    t2 = time.time() - step_start_time - t1 # hypergrad estimation time \n",
    "    val_losses.append(cost.item())\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",
    "    if k % eval_interval == 0 or k == K - 1:\n",
    "        print('outer step={} ({:.2e}s)({:.2e}, {:.2e}) | val loss={} | hypergrad norm = {:.3e}'.format(k, step_time, t1, t2, val_losses[-1], torch.norm(hparams[0].grad)))\n",
    "\n",
    "\n",
    "print('total time = {}'.format(total_time))\n",
    "\n",
    "plt.title('validation loss')\n",
    "plt.xlabel('outer steps')\n",
    "#plt.plot(val_losses)\n",
    "plt.plot(running_time, val_losses)\n",
    "plt.show()\n",
    "\n",
    "val_hozog = val_losses\n",
    "run_hozog = running_time\n",
    "    "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# AID-FP"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "outer step=0 (7.67e-02s)(3.89e-02, 3.74e-02) | val loss=1197.4111328125 | hypergrad norm = 5.186e+02\n",
      "outer step=10 (2.89e-02s)(1.30e-02, 1.58e-02) | val loss=17.918315887451172 | hypergrad norm = 2.187e+01\n",
      "outer step=20 (2.88e-02s)(1.30e-02, 1.56e-02) | val loss=4.412731170654297 | hypergrad norm = 1.144e+01\n",
      "outer step=30 (2.87e-02s)(1.29e-02, 1.56e-02) | val loss=1.5219109058380127 | hypergrad norm = 5.968e+00\n",
      "outer step=40 (2.87e-02s)(1.29e-02, 1.56e-02) | val loss=0.9506080150604248 | hypergrad norm = 3.715e+00\n",
      "outer step=50 (2.88e-02s)(1.29e-02, 1.56e-02) | val loss=0.6880261898040771 | hypergrad norm = 2.244e+00\n",
      "outer step=60 (2.76e-02s)(1.30e-02, 1.45e-02) | val loss=0.24693500995635986 | hypergrad norm = 1.342e+00\n",
      "outer step=70 (2.70e-02s)(1.25e-02, 1.43e-02) | val loss=0.15687395632266998 | hypergrad norm = 7.552e-01\n",
      "outer step=80 (2.70e-02s)(1.24e-02, 1.44e-02) | val loss=0.12205974757671356 | hypergrad norm = 6.044e-01\n",
      "outer step=90 (2.69e-02s)(1.24e-02, 1.43e-02) | val loss=0.09525397419929504 | hypergrad norm = 3.698e-01\n",
      "outer step=100 (2.70e-02s)(1.25e-02, 1.44e-02) | val loss=0.07939820736646652 | hypergrad norm = 2.148e-01\n",
      "outer step=110 (2.69e-02s)(1.24e-02, 1.42e-02) | val loss=0.0700264647603035 | hypergrad norm = 1.433e-01\n",
      "outer step=120 (2.69e-02s)(1.24e-02, 1.43e-02) | val loss=0.06311522424221039 | hypergrad norm = 9.312e-02\n",
      "outer step=130 (2.69e-02s)(1.24e-02, 1.43e-02) | val loss=0.05764485523104668 | hypergrad norm = 6.151e-02\n",
      "outer step=140 (2.69e-02s)(1.24e-02, 1.43e-02) | val loss=0.05312753841280937 | hypergrad norm = 3.960e-02\n",
      "outer step=149 (2.71e-02s)(1.25e-02, 1.44e-02) | val loss=0.04969889298081398 | hypergrad norm = 2.904e-02\n",
      "total time = 4.2472639083862305\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "K = 150\n",
    "eval_interval = 10\n",
    "T = 20\n",
    "beta = 0.05\n",
    "\n",
    "#hparams = [torch.randn((p, d)).requires_grad_(True)]\n",
    "hparams = [hp.clone() for hp in hp0]\n",
    "hparams = [hp.requires_grad_(True) for hp in hparams]\n",
    "\n",
    "outer_opt = torch.optim.Adam(lr=beta, params=hparams)\n",
    "\n",
    "total_time, val_losses, running_time = 0,  [], []\n",
    "\n",
    "for k in range(K):\n",
    "    \n",
    "    step_start_time = time.time() \n",
    "    inner_losses = []\n",
    "    params = inner_solver(hparams, steps=T)\n",
    "    t1 = time.time() - step_start_time # inner loop time\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    _, cost = hg.fixed_point(params, hparams, T, map_func, outer_func, set_grad=True)\n",
    "    t2 = time.time() - step_start_time - t1 # hypergrad estimation time \n",
    "    val_losses.append(cost.item())\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",
    "    if k % eval_interval == 0 or k == K - 1:\n",
    "        print('outer step={} ({:.2e}s)({:.2e}, {:.2e}) | val loss={} | hypergrad norm = {:.3e}'.format(k, step_time, t1, t2, val_losses[-1], torch.norm(hparams[0].grad)))\n",
    "        \n",
    "print('total time = {}'.format(total_time))\n",
    "\n",
    "plt.title('validation loss')\n",
    "plt.xlabel('outer steps')\n",
    "#plt.plot(val_losses)\n",
    "plt.plot(running_time, val_losses)\n",
    "plt.show()\n",
    "\n",
    "val_fp = val_losses\n",
    "run_fp = running_time\n",
    "    "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# AID-CG"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "outer step=0 (7.73e-02s)(4.92e-02, 2.77e-02) | val loss=1197.4111328125 | hypergrad norm = 3.977e+02\n",
      "outer step=10 (2.89e-02s)(1.27e-02, 1.60e-02) | val loss=17.882030487060547 | hypergrad norm = 4.482e+00\n",
      "outer step=20 (2.87e-02s)(1.26e-02, 1.59e-02) | val loss=5.776135444641113 | hypergrad norm = 1.331e+00\n",
      "outer step=30 (2.92e-02s)(1.27e-02, 1.62e-02) | val loss=2.389629602432251 | hypergrad norm = 4.621e-01\n",
      "outer step=40 (2.93e-02s)(1.27e-02, 1.63e-02) | val loss=1.305540680885315 | hypergrad norm = 2.379e-01\n",
      "outer step=50 (2.76e-02s)(1.25e-02, 1.49e-02) | val loss=0.8788031935691833 | hypergrad norm = 1.592e-01\n",
      "outer step=60 (2.81e-02s)(1.30e-02, 1.49e-02) | val loss=0.6584704518318176 | hypergrad norm = 1.199e-01\n",
      "outer step=70 (2.69e-02s)(1.20e-02, 1.47e-02) | val loss=0.4949369430541992 | hypergrad norm = 8.541e-02\n",
      "outer step=80 (2.68e-02s)(1.20e-02, 1.47e-02) | val loss=0.3933676481246948 | hypergrad norm = 6.333e-02\n",
      "outer step=90 (2.77e-02s)(1.20e-02, 1.55e-02) | val loss=0.3318435847759247 | hypergrad norm = 5.278e-02\n",
      "outer step=100 (2.80e-02s)(1.25e-02, 1.53e-02) | val loss=0.2882140576839447 | hypergrad norm = 4.618e-02\n",
      "outer step=110 (2.69e-02s)(1.20e-02, 1.46e-02) | val loss=0.2518264651298523 | hypergrad norm = 3.996e-02\n",
      "outer step=120 (2.69e-02s)(1.20e-02, 1.47e-02) | val loss=0.2219531536102295 | hypergrad norm = 3.465e-02\n",
      "outer step=130 (2.67e-02s)(1.21e-02, 1.45e-02) | val loss=0.1983851194381714 | hypergrad norm = 3.076e-02\n",
      "outer step=140 (2.66e-02s)(1.19e-02, 1.45e-02) | val loss=0.17912206053733826 | hypergrad norm = 2.771e-02\n",
      "outer step=149 (2.67e-02s)(1.20e-02, 1.45e-02) | val loss=0.16424468159675598 | hypergrad norm = 2.527e-02\n",
      "total time = 4.248292446136475\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "K = 150\n",
    "eval_interval = 10\n",
    "T = 20\n",
    "beta = 0.01 # diverges with 0.05\n",
    "\n",
    "#hparams = [torch.randn((p, d)).requires_grad_(True)]\n",
    "hparams = [hp.clone() for hp in hp0]\n",
    "hparams = [hp.requires_grad_(True) for hp in hparams]\n",
    "\n",
    "outer_opt = torch.optim.Adam(lr=beta, params=hparams)\n",
    "\n",
    "total_time, val_losses, running_time = 0,  [], []\n",
    "\n",
    "for k in range(K):\n",
    "    \n",
    "    step_start_time = time.time() \n",
    "    inner_losses = []\n",
    "    params = inner_solver(hparams, steps=T)\n",
    "    t1 = time.time() - step_start_time # inner loop time\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    _, cost = hg.CG(params, hparams, T, map_func, outer_func, set_grad=True)\n",
    "    t2 = time.time() - step_start_time - t1 # hypergrad estimation time \n",
    "    val_losses.append(cost.item())\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",
    "    if k % eval_interval == 0 or k == K - 1:\n",
    "        print('outer step={} ({:.2e}s)({:.2e}, {:.2e}) | val loss={} | hypergrad norm = {:.3e}'.format(k, step_time, t1, t2, val_losses[-1], torch.norm(hparams[0].grad)))\n",
    "\n",
    "\n",
    "print('total time = {}'.format(total_time))\n",
    "\n",
    "plt.title('validation loss')\n",
    "plt.xlabel('outer steps')\n",
    "#plt.plot(val_losses)\n",
    "plt.plot(running_time, val_losses)\n",
    "plt.show()\n",
    "\n",
    "val_cg = val_losses\n",
    "run_cg = running_time\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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PWN6sirRnv65IXkDbRu1reGIxe80agHjl8CsiyE9EXKPw3uj4KZl0KKLr6MTERGRCvJONK+//voK7Sj7/vvvs2PHDjZv3symTZvo2bNnLUbo8cwzz7h3IO7WrRv9+/ev8r26du3KhRdeyJdffklubi5PPvkkL774YrXiGzp0KHFxcSQkJJCUlMTrr7/OXXfdVa171gdKUqVGxHqtS614GRpP4pGz/VesggL3ee+WvYskqX899a81FKn4tchT4LrFMPsSu0RNViq8MwJu/sJ+TkRERPza4cOH+fTTTwFo3Lgxr776KmFhYeX2a9asGRMnTgTsJLcuktQPP/yQZ5991n3+yCOP4HBUb6LpP//5T77++msKCgqYMWMG559/PqNHj65Q37179/Lyyy8XicnhcPDYY48xZoy9R8sDDzxA7969q5VM1wea7is1omMVytAERkcT6FqcbmVnk+u1KL13y6LrUuUkEhMH1y6E4Eb2eXqinager3pxbREREakb7733nntK6siRIyuUoAJcffXVBAba42fvv/8+ubm5tRZjUlIS999/P6NHj3ZP2b355psZO3Zste997rnn8vjjjwP2pkpjx45l6tSpZGSU/vtxZmYmzzzzDD179uSHH34o9vzo0aPdo6dZWVkMHjyY6dOnk52dXWYse/bsITGxfi6b00iq1IiitVIrVoYGIKRrHPmH7OQjOz4eIuz7nBl9JsGOYHKduexN28uhzEO0CG9R1q2kIWnbG8bOgTmj7fWpqb/Be1fCDZ/a61dFRETEL1V0V98TxcTEcNFFF7F06VJSUlL4+OOPGTVqVJViSEhIYMmSJe7z3Nxcjh49yr59+1i7di3ff/99kST4lltu4ZVXXqnSa5XkoYceIjs7m8ceewyn08k///lPXn75ZYYOHUq/fv1o0aIFTqeTpKQkVq9ezYoVK0hLSyvzntOnT8eyLF555RWys7OZNGkSTzzxBBdeeCH9+vUjOjqasLAw0tPT2bVrF6tWreK7774jPz8fgMjISMLDw8t8DX+iJFVqhHcZmj0pGTidFg5H+cWQQ+O6kvHd9wDkJCRAb3sENTggmO4tuvNj8o8AbDi4gaGxQ2shcvFbnQbBqDfhg/H2jr9JW+D9sTBunhJVERERP7Rx40a2bNkCQNu2bRk0aFCl+l933XUsXboUsJPdqiap8+fPZ/78+eW2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\n",
      "text/plain": [
       "<Figure size 1080x576 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "s, s2 = 25, 25\n",
    "lw = 3\n",
    "\n",
    "plt.plot(run_zoj[:s2], val_zoj[:s2], '-', label='ESJ', linewidth=lw)\n",
    "plt.plot(run_hozog[:s2], val_hozog[:s2], '-', label='HOZOG', linewidth=lw)\n",
    "plt.plot(run_cg[:s], val_cg[:s], '-', label='AID-CG', linewidth=lw)\n",
    "plt.plot(run_fp[:s], val_fp[:s], '-', label='AID-FP', linewidth=lw)\n",
    "plt.xlabel('Running time', fontsize=24)\n",
    "plt.ylabel('Outer loss', fontsize=24)\n",
    "plt.xticks(fontsize=20)\n",
    "plt.yticks(fontsize=20)\n",
    "plt.grid()\n",
    "plt.legend(loc='upper right', fontsize=30)\n",
    "plt.gcf().set_size_inches(15, 8)\n",
    "plt.savefig('hrl_T20_D256.pdf', bbox_inches='tight', transparent=True, dpi=600)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Different number T of inner loop steps"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "outer step=0 (3.28e-02s)(1.90e-02, 1.24e-02) | val loss=508.63787841796875 | hypergrad norm = 3.846e+02\n",
      "outer step=10 (1.26e-02s)(5.77e-03, 6.74e-03) | val loss=5.386643886566162 | hypergrad norm = 4.129e+01\n",
      "outer step=20 (1.26e-02s)(5.77e-03, 6.69e-03) | val loss=1.8146494626998901 | hypergrad norm = 2.238e+01\n",
      "outer step=30 (1.23e-02s)(5.62e-03, 6.60e-03) | val loss=0.7977696061134338 | hypergrad norm = 1.537e+01\n",
      "outer step=40 (1.29e-02s)(5.71e-03, 7.00e-03) | val loss=0.40675872564315796 | hypergrad norm = 1.075e+01\n",
      "outer step=50 (1.27e-02s)(5.70e-03, 6.87e-03) | val loss=0.14815568923950195 | hypergrad norm = 6.738e+00\n",
      "outer step=60 (1.25e-02s)(5.69e-03, 6.71e-03) | val loss=0.07873979210853577 | hypergrad norm = 4.573e+00\n",
      "outer step=70 (1.23e-02s)(5.60e-03, 6.57e-03) | val loss=0.057468317449092865 | hypergrad norm = 3.979e+00\n",
      "outer step=80 (1.24e-02s)(5.67e-03, 6.67e-03) | val loss=0.028070863336324692 | hypergrad norm = 2.688e+00\n",
      "outer step=90 (1.23e-02s)(5.62e-03, 6.62e-03) | val loss=0.01711208000779152 | hypergrad norm = 1.918e+00\n",
      "outer step=99 (1.40e-02s)(6.47e-03, 7.35e-03) | val loss=0.014298493042588234 | hypergrad norm = 1.653e+00\n",
      "total time = 1.2930059432983398\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "K = 100\n",
    "eval_interval = 10\n",
    "T = 20\n",
    "mu = .01\n",
    "beta = 0.05\n",
    "\n",
    "#hparams = [torch.randn((p, d)).requires_grad_(True)]\n",
    "hparams = [hp.clone() for hp in hp0]\n",
    "hparams = [hp.requires_grad_(True) for hp in hparams]\n",
    "\n",
    "outer_opt = torch.optim.Adam(lr=beta, params=hparams)\n",
    "\n",
    "total_time, val_losses, running_time = 0,  [], []\n",
    "\n",
    "for k in range(K):\n",
    "    \n",
    "    step_start_time = time.time() \n",
    "    inner_losses = []\n",
    "    params = inner_solver(hparams, steps=T)\n",
    "    t1 = time.time() - step_start_time # inner loop time\n",
    "\n",
    "    outer_opt.zero_grad()\n",
    "    _, cost = hg.hgvzoj(params, hparams, outer_func, inner_solver, mu=mu, T=T, p=1, set_grad=True)\n",
    "    t2 = time.time() - step_start_time - t1 # hypergrad estimation time \n",
    "    val_losses.append(cost.item())\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",
    "    if k % eval_interval == 0 or k == K - 1:\n",
    "        print('outer step={} ({:.2e}s)({:.2e}, {:.2e}) | val loss={} | hypergrad norm = {:.3e}'.format(k, step_time, t1, t2, val_losses[-1], torch.norm(hparams[0].grad)))\n",
    "        \n",
    "print('total time = {}'.format(total_time))\n",
    "\n",
    "plt.title('validation loss')\n",
    "plt.xlabel('outer steps')\n",
    "#plt.plot(val_losses)\n",
    "plt.plot(running_time, val_losses)\n",
    "plt.show()\n",
    "\n",
    "val_zoj20 = val_losses\n",
    "run_zoj20 = running_time\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.6.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
