{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from copy import deepcopy\n",
    "\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "from sklearn.decomposition import PCA\n",
    "\n",
    "from utils import _build_dataset, train, plot_trajectory, hb_train, _get_traj_directions\n",
    "from models import DiagonalNet\n",
    "\n",
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "random_seed = 999\n",
    "n = 40\n",
    "d = 100\n",
    "n_informative = 5\n",
    "\n",
    "# build data\n",
    "X, y, w_star = _build_dataset(\n",
    "    n=n, d=d, n_informative=n_informative, random_seed=random_seed\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "reg coef 0.009444444444444441\n"
     ]
    }
   ],
   "source": [
    "# build model\n",
    "model_config = {\n",
    "    \"scale\": 0.01,\n",
    "    \"n_layer\": 2,\n",
    "    \"d\": d,\n",
    "}\n",
    "eta = 1e-3\n",
    "factor = 10\n",
    "eta_euler = eta / factor\n",
    "num_it = 10000\n",
    "mu = 0.7\n",
    "print(f\"reg coef {eta * (1 + mu) / (2 * (1 - mu)**2)}\")\n",
    "\n",
    "hbf_config = {\n",
    "    \"net\": DiagonalNet(**model_config),\n",
    "    \"eta\": eta,\n",
    "    \"num_it\": num_it * factor,\n",
    "    \"eta_euler\": eta_euler,\n",
    "    \"mu\": mu,\n",
    "    \"hbf\": True,\n",
    "}\n",
    "\n",
    "hb_config = {\n",
    "    \"net\": DiagonalNet(**model_config),\n",
    "    \"eta\": eta,\n",
    "    \"num_it\": num_it,\n",
    "    \"eta_euler\": eta_euler,\n",
    "    \"mu\": mu,\n",
    "}\n",
    "\n",
    "rgf_config = {\n",
    "    \"net\": DiagonalNet(**model_config),\n",
    "    \"eta\": eta,\n",
    "    \"num_it\": num_it * factor,\n",
    "    \"eta_euler\": eta_euler,\n",
    "    \"mu\": mu,\n",
    "}\n",
    "\n",
    "gd_config = {\n",
    "    \"net\": DiagonalNet(**model_config),\n",
    "    \"eta\": eta,\n",
    "    \"num_it\": num_it,\n",
    "    \"eta_euler\": eta_euler,\n",
    "    \"mu\": 0,\n",
    "}\n",
    "\n",
    "gf_config = {\n",
    "    \"net\": DiagonalNet(**model_config),\n",
    "    \"eta\": eta,\n",
    "    \"num_it\": num_it * factor,\n",
    "    \"eta_euler\": eta_euler,\n",
    "    \"mu\": 0,\n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "10"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "factor"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "hbf result\n",
      "loss: 1.698438048362732\n",
      "loss: 1.690220594406128\n",
      "loss: 1.6186163425445557\n",
      "loss: 1.2296675443649292\n",
      "loss: 0.8471442461013794\n",
      "loss: 0.6228559017181396\n",
      "loss: 0.44777950644493103\n",
      "loss: 0.4050222337245941\n",
      "loss: 0.3869345486164093\n",
      "loss: 0.3651495575904846\n",
      "loss: 0.3419426679611206\n",
      "loss: 0.3226388096809387\n",
      "loss: 0.3103238344192505\n",
      "loss: 0.3040173351764679\n",
      "loss: 0.3007383346557617\n",
      "loss: 0.29802942276000977\n",
      "loss: 0.29455190896987915\n",
      "loss: 0.2897433936595917\n",
      "loss: 0.2833712100982666\n",
      "loss: 0.27537161111831665\n",
      "loss: 0.26580575108528137\n",
      "loss: 0.254919171333313\n",
      "loss: 0.24311953783035278\n",
      "loss: 0.2309514880180359\n",
      "loss: 0.21902576088905334\n",
      "loss: 0.20784087479114532\n",
      "loss: 0.19780349731445312\n",
      "loss: 0.18907740712165833\n",
      "loss: 0.1816064417362213\n",
      "loss: 0.17530393600463867\n",
      "loss: 0.16991302371025085\n",
      "loss: 0.16526812314987183\n",
      "loss: 0.16113176941871643\n",
      "loss: 0.1573929637670517\n",
      "loss: 0.15387296676635742\n",
      "loss: 0.15051952004432678\n",
      "loss: 0.14727039635181427\n",
      "loss: 0.14407873153686523\n",
      "loss: 0.14092041552066803\n",
      "loss: 0.13778717815876007\n",
      "loss: 0.13468147814273834\n",
      "loss: 0.13161039352416992\n",
      "loss: 0.12858730554580688\n",
      "loss: 0.12562303245067596\n",
      "loss: 0.12273451685905457\n",
      "loss: 0.11995666474103928\n",
      "loss: 0.11728723347187042\n",
      "loss: 0.11473800987005234\n",
      "loss: 0.1123044341802597\n",
      "loss: 0.1099892258644104\n",
      "loss: 0.10779446363449097\n",
      "loss: 0.10572304576635361\n",
      "loss: 0.1037674993276596\n",
      "loss: 0.1019221842288971\n",
      "loss: 0.10017881542444229\n",
      "loss: 0.09852229803800583\n",
      "loss: 0.0969458743929863\n",
      "loss: 0.09545916318893433\n",
      "loss: 0.09405842423439026\n",
      "loss: 0.09272567927837372\n",
      "loss: 0.09145962446928024\n",
      "loss: 0.09026189893484116\n",
      "loss: 0.08911631256341934\n",
      "loss: 0.08802056312561035\n",
      "loss: 0.0869813933968544\n",
      "loss: 0.08598265796899796\n",
      "loss: 0.08502358198165894\n",
      "loss: 0.0841086208820343\n",
      "loss: 0.08322439342737198\n",
      "loss: 0.08237998187541962\n",
      "loss: 0.08156538754701614\n",
      "loss: 0.08077406138181686\n",
      "loss: 0.08001388609409332\n",
      "loss: 0.07927285879850388\n",
      "loss: 0.07854811102151871\n",
      "loss: 0.07786129415035248\n",
      "loss: 0.0772005245089531\n",
      "loss: 0.07655692100524902\n",
      "loss: 0.07594404369592667\n",
      "loss: 0.07534518837928772\n",
      "loss: 0.07476051151752472\n",
      "loss: 0.07420185208320618\n",
      "loss: 0.0736563429236412\n",
      "loss: 0.07312378287315369\n",
      "loss: 0.07260274887084961\n",
      "loss: 0.07210443168878555\n",
      "loss: 0.07161863148212433\n",
      "loss: 0.07113949954509735\n",
      "loss: 0.0706702321767807\n",
      "loss: 0.07022558897733688\n",
      "loss: 0.06979187577962875\n",
      "loss: 0.06936459243297577\n",
      "loss: 0.06894981861114502\n",
      "loss: 0.06854167580604553\n",
      "loss: 0.0681510865688324\n",
      "loss: 0.06776890158653259\n",
      "loss: 0.06739484518766403\n",
      "loss: 0.06703126430511475\n",
      "loss: 0.06667204946279526\n",
      "loss: 0.0663175955414772\n",
      "RGF result\n",
      "loss: 1.6983619928359985\n",
      "loss: 1.6887476444244385\n",
      "loss: 1.6000727415084839\n",
      "loss: 1.1687092781066895\n",
      "loss: 0.8275883197784424\n",
      "loss: 0.5990682244300842\n",
      "loss: 0.4407808780670166\n",
      "loss: 0.4036079943180084\n",
      "loss: 0.38523584604263306\n",
      "loss: 0.3630734384059906\n",
      "loss: 0.33989861607551575\n",
      "loss: 0.3210170567035675\n",
      "loss: 0.3091484606266022\n",
      "loss: 0.3030577301979065\n",
      "loss: 0.2997373044490814\n",
      "loss: 0.2968258857727051\n",
      "loss: 0.29305946826934814\n",
      "loss: 0.28789418935775757\n",
      "loss: 0.281110554933548\n",
      "loss: 0.27264904975891113\n",
      "loss: 0.2626342177391052\n",
      "loss: 0.2513488233089447\n",
      "loss: 0.23924727737903595\n",
      "loss: 0.22689999639987946\n",
      "loss: 0.21493689715862274\n",
      "loss: 0.2038484811782837\n",
      "loss: 0.19399064779281616\n",
      "loss: 0.18547235429286957\n",
      "loss: 0.17820879817008972\n",
      "loss: 0.17207682132720947\n",
      "loss: 0.16682976484298706\n",
      "loss: 0.16226431727409363\n",
      "loss: 0.15819111466407776\n",
      "loss: 0.1544693410396576\n",
      "loss: 0.1509374976158142\n",
      "loss: 0.1475714147090912\n",
      "loss: 0.14429329335689545\n",
      "loss: 0.14106537401676178\n",
      "loss: 0.13787075877189636\n",
      "loss: 0.13470621407032013\n",
      "loss: 0.13157525658607483\n",
      "loss: 0.12848863005638123\n",
      "loss: 0.12546001374721527\n",
      "loss: 0.12250187993049622\n",
      "loss: 0.1196400597691536\n",
      "loss: 0.11689677089452744\n",
      "loss: 0.11426972597837448\n",
      "loss: 0.11176714301109314\n",
      "loss: 0.10939326882362366\n",
      "loss: 0.10714371502399445\n",
      "loss: 0.10501803457736969\n",
      "loss: 0.10301299393177032\n",
      "loss: 0.10112256556749344\n",
      "loss: 0.09933973848819733\n",
      "loss: 0.09764876961708069\n",
      "loss: 0.09603969752788544\n",
      "loss: 0.09451227635145187\n",
      "loss: 0.09308235347270966\n",
      "loss: 0.0917234867811203\n",
      "loss: 0.09044646471738815\n",
      "loss: 0.08922949433326721\n",
      "loss: 0.08806721121072769\n",
      "loss: 0.08696181327104568\n",
      "loss: 0.08591019362211227\n",
      "loss: 0.08489958196878433\n",
      "loss: 0.08393379300832748\n",
      "loss: 0.08300958573818207\n",
      "loss: 0.08211669325828552\n",
      "loss: 0.08126629889011383\n",
      "loss: 0.08044591546058655\n",
      "loss: 0.07964938133955002\n",
      "loss: 0.0788847804069519\n",
      "loss: 0.07813809812068939\n",
      "loss: 0.07740917801856995\n",
      "loss: 0.07671520113945007\n",
      "loss: 0.07605085521936417\n",
      "loss: 0.07540418207645416\n",
      "loss: 0.07478699088096619\n",
      "loss: 0.07418522983789444\n",
      "loss: 0.07359800487756729\n",
      "loss: 0.07303506880998611\n",
      "loss: 0.07248695939779282\n",
      "loss: 0.07195239514112473\n",
      "loss: 0.07142766565084457\n",
      "loss: 0.0709274634718895\n",
      "loss: 0.0704394206404686\n",
      "loss: 0.06995788216590881\n",
      "loss: 0.06948722898960114\n",
      "loss: 0.06903821974992752\n",
      "loss: 0.06860252469778061\n",
      "loss: 0.06817283481359482\n",
      "loss: 0.0677572637796402\n",
      "loss: 0.06734715402126312\n",
      "loss: 0.06695235520601273\n",
      "loss: 0.06656862795352936\n",
      "loss: 0.06619521975517273\n",
      "loss: 0.06582996994256973\n",
      "loss: 0.06546923518180847\n",
      "loss: 0.06511320918798447\n",
      "loss: 0.06477061659097672\n",
      "HB result\n",
      "loss: 0.3654335141181946\n",
      "loss: 0.27578991651535034\n",
      "loss: 0.17575310170650482\n",
      "loss: 0.1380835920572281\n",
      "loss: 0.11028602719306946\n",
      "loss: 0.09297450631856918\n",
      "loss: 0.08255031704902649\n",
      "loss: 0.07551656663417816\n",
      "loss: 0.07038488239049911\n",
      "loss: 0.06645812094211578\n"
     ]
    }
   ],
   "source": [
    "print(\"hbf result\")\n",
    "hbf_history, hbf_net = train(X=X, y=y, w_star=w_star, return_loss=False, return_trajectory=True, **hbf_config)\n",
    "\n",
    "print(\"RGF result\")\n",
    "rgf_history, rgf_net = train(\n",
    "    X=X, y=y, w_star=w_star, return_loss=False, return_trajectory=True, **rgf_config\n",
    ")\n",
    "print(\"HB result\")\n",
    "hb_history, hb_net = hb_train(\n",
    "    X=X, y=y, w_star=w_star, return_loss=False, return_trajectory=True, **hb_config\n",
    ")\n",
    "# print(\"GD result\")\n",
    "# gd_history, gd_net = hb_train(\n",
    "#     X=X, y=y, w_star=w_star, return_loss=True, return_trajectory=False, **gd_config\n",
    "# )\n",
    "\n",
    "\n",
    "# print(\"GF result\")\n",
    "# gf_history, gf_net = train(\n",
    "#     X=X, y=y, w_star=w_star, return_loss=True, return_trajectory=False, **gf_config\n",
    "# )\n",
    "# # plot_trajectory(traj_list, name_list)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "reg coef 0.022500000000000013\n",
      "hbf result\n",
      "loss: 1.696801781654358\n",
      "loss: 1.6392894983291626\n",
      "loss: 1.0467209815979004\n",
      "loss: 0.6586979627609253\n",
      "loss: 0.4224713444709778\n",
      "loss: 0.38926827907562256\n",
      "loss: 0.3564014434814453\n",
      "loss: 0.32506510615348816\n",
      "loss: 0.30816352367401123\n",
      "loss: 0.3022236227989197\n",
      "loss: 0.29836612939834595\n",
      "loss: 0.2924029529094696\n",
      "loss: 0.28314560651779175\n",
      "loss: 0.2703869342803955\n",
      "loss: 0.2545883357524872\n",
      "loss: 0.23700006306171417\n",
      "loss: 0.21945513784885406\n",
      "loss: 0.20367839932441711\n",
      "loss: 0.19060082733631134\n",
      "loss: 0.18028351664543152\n",
      "loss: 0.17218874394893646\n",
      "loss: 0.16563555598258972\n",
      "loss: 0.16004076600074768\n",
      "loss: 0.15497061610221863\n",
      "loss: 0.15018115937709808\n",
      "loss: 0.1454886794090271\n",
      "loss: 0.1408628672361374\n",
      "loss: 0.13629035651683807\n",
      "loss: 0.13179990649223328\n",
      "loss: 0.12743672728538513\n",
      "loss: 0.12325259298086166\n",
      "loss: 0.11929012835025787\n",
      "loss: 0.11555986106395721\n",
      "loss: 0.11208643019199371\n",
      "loss: 0.10889048874378204\n",
      "loss: 0.10594221949577332\n",
      "loss: 0.10322681069374084\n",
      "loss: 0.10071539878845215\n",
      "loss: 0.0983937457203865\n",
      "loss: 0.0962616428732872\n",
      "loss: 0.09429127722978592\n",
      "loss: 0.0924609825015068\n",
      "loss: 0.09075678139925003\n",
      "loss: 0.08916084468364716\n",
      "loss: 0.08765114843845367\n",
      "loss: 0.08624397963285446\n",
      "loss: 0.08492472767829895\n",
      "loss: 0.08368118107318878\n",
      "loss: 0.08250442147254944\n",
      "loss: 0.0813881978392601\n",
      "loss: 0.08032729476690292\n",
      "loss: 0.07932533323764801\n",
      "loss: 0.07837168872356415\n",
      "loss: 0.0774618461728096\n",
      "loss: 0.07659539580345154\n",
      "loss: 0.07576817274093628\n",
      "loss: 0.07497214525938034\n",
      "loss: 0.07421048730611801\n",
      "loss: 0.07346929609775543\n",
      "loss: 0.07275968044996262\n",
      "loss: 0.07208357006311417\n",
      "loss: 0.07144071161746979\n",
      "loss: 0.07081484794616699\n",
      "loss: 0.07022146135568619\n",
      "loss: 0.06964264810085297\n",
      "loss: 0.06908836215734482\n",
      "loss: 0.06855456531047821\n",
      "loss: 0.06803470104932785\n",
      "loss: 0.06753681600093842\n",
      "loss: 0.06705347448587418\n",
      "loss: 0.06658367812633514\n",
      "loss: 0.0661269947886467\n",
      "loss: 0.06568945199251175\n",
      "loss: 0.06526405364274979\n",
      "loss: 0.06485281139612198\n",
      "loss: 0.06445100903511047\n",
      "loss: 0.06406769901514053\n",
      "loss: 0.06369215995073318\n",
      "loss: 0.06332966685295105\n",
      "loss: 0.06297275424003601\n",
      "loss: 0.0626271590590477\n",
      "loss: 0.06229556351900101\n",
      "loss: 0.061970386654138565\n",
      "loss: 0.06165727227926254\n",
      "loss: 0.061348844319581985\n",
      "loss: 0.06104513630270958\n",
      "loss: 0.060751158744096756\n",
      "loss: 0.060468435287475586\n",
      "loss: 0.06019126623868942\n",
      "loss: 0.05992304906249046\n",
      "loss: 0.05966213345527649\n",
      "loss: 0.059405017644166946\n",
      "loss: 0.05915115401148796\n",
      "loss: 0.058900658041238785\n",
      "loss: 0.05866193026304245\n",
      "loss: 0.058428697288036346\n",
      "loss: 0.05820010229945183\n",
      "loss: 0.057976484298706055\n",
      "loss: 0.05776231735944748\n",
      "loss: 0.05755170062184334\n",
      "RGF result\n",
      "loss: 1.6960160732269287\n",
      "loss: 1.6002507209777832\n",
      "loss: 0.9496427774429321\n",
      "loss: 0.5992650389671326\n",
      "loss: 0.4154219925403595\n",
      "loss: 0.3852492868900299\n",
      "loss: 0.3512866497039795\n",
      "loss: 0.3210289776325226\n",
      "loss: 0.3055776357650757\n",
      "loss: 0.2997508943080902\n",
      "loss: 0.2951142489910126\n",
      "loss: 0.2879185378551483\n",
      "loss: 0.2771124839782715\n",
      "loss: 0.2626732885837555\n",
      "loss: 0.24540668725967407\n",
      "loss: 0.2269495725631714\n",
      "loss: 0.20930764079093933\n",
      "loss: 0.1940392702817917\n",
      "loss: 0.18172596395015717\n",
      "loss: 0.17210406064987183\n",
      "loss: 0.16449905931949615\n",
      "loss: 0.15821778774261475\n",
      "loss: 0.1527070701122284\n",
      "loss: 0.14761069416999817\n",
      "loss: 0.14270491898059845\n",
      "loss: 0.13788266479969025\n",
      "loss: 0.13314516842365265\n",
      "loss: 0.1285022646188736\n",
      "loss: 0.12400110065937042\n",
      "loss: 0.11969300359487534\n",
      "loss: 0.11562524735927582\n",
      "loss: 0.1118120476603508\n",
      "loss: 0.1082761287689209\n",
      "loss: 0.10504072159528732\n",
      "loss: 0.10207296907901764\n",
      "loss: 0.09935633093118668\n",
      "loss: 0.0968514010310173\n",
      "loss: 0.09454499185085297\n",
      "loss: 0.09242960065603256\n",
      "loss: 0.09048239886760712\n",
      "loss: 0.08867751061916351\n",
      "loss: 0.08699727058410645\n",
      "loss: 0.08542285114526749\n",
      "loss: 0.08393491804599762\n",
      "loss: 0.08255583047866821\n",
      "loss: 0.08125923573970795\n",
      "loss: 0.08003537356853485\n",
      "loss: 0.07887942343950272\n",
      "loss: 0.07778266072273254\n",
      "loss: 0.07674168050289154\n",
      "loss: 0.07575409859418869\n",
      "loss: 0.07481725513935089\n",
      "loss: 0.07392318546772003\n",
      "loss: 0.07306798547506332\n",
      "loss: 0.07225333899259567\n",
      "loss: 0.07146825641393661\n",
      "loss: 0.07071080058813095\n",
      "loss: 0.06997790187597275\n",
      "loss: 0.06928221136331558\n",
      "loss: 0.0686170682311058\n",
      "loss: 0.06797900795936584\n",
      "loss: 0.06736474484205246\n",
      "loss: 0.06677526980638504\n",
      "loss: 0.06620414555072784\n",
      "loss: 0.06565739214420319\n",
      "loss: 0.06512876600027084\n",
      "loss: 0.06461372971534729\n",
      "loss: 0.06412095576524734\n",
      "loss: 0.06364455819129944\n",
      "loss: 0.06317780166864395\n",
      "loss: 0.06272555142641068\n",
      "loss: 0.06229399889707565\n",
      "loss: 0.061876311898231506\n",
      "loss: 0.06146560609340668\n",
      "loss: 0.06106896698474884\n",
      "loss: 0.06069127097725868\n",
      "loss: 0.06032394617795944\n",
      "loss: 0.05996239557862282\n",
      "loss: 0.05960632860660553\n",
      "loss: 0.05926491692662239\n",
      "loss: 0.058939091861248016\n",
      "loss: 0.05862182006239891\n",
      "loss: 0.05830957368016243\n",
      "loss: 0.05800177901983261\n",
      "loss: 0.05769868940114975\n",
      "loss: 0.05740823596715927\n",
      "loss: 0.057129308581352234\n",
      "loss: 0.05686154589056969\n",
      "loss: 0.05659770965576172\n",
      "loss: 0.056337784975767136\n",
      "loss: 0.056081097573041916\n",
      "loss: 0.05582817643880844\n",
      "loss: 0.0555778443813324\n",
      "loss: 0.05534083768725395\n",
      "loss: 0.05510890483856201\n",
      "loss: 0.05488687381148338\n",
      "loss: 0.05467086657881737\n",
      "loss: 0.05445871502161026\n",
      "loss: 0.054249390959739685\n",
      "loss: 0.05404238402843475\n",
      "HB result\n",
      "loss: 0.30237483978271484\n",
      "loss: 0.18072842061519623\n",
      "loss: 0.1275346875190735\n",
      "loss: 0.0963500440120697\n",
      "loss: 0.08142845332622528\n",
      "loss: 0.07277758419513702\n",
      "loss: 0.06705059856176376\n",
      "loss: 0.06296230107545853\n",
      "loss: 0.059893734753131866\n",
      "loss: 0.057501137256622314\n"
     ]
    }
   ],
   "source": [
    "mu = 0.8\n",
    "print(f\"reg coef {eta * (1 + mu) / (2 * (1 - mu)**2)}\")\n",
    "\n",
    "hbf_config = {\n",
    "    \"net\": DiagonalNet(**model_config),\n",
    "    \"eta\": eta,\n",
    "    \"num_it\": num_it * factor,\n",
    "    \"eta_euler\": eta_euler,\n",
    "    \"mu\": mu,\n",
    "    \"hbf\": True,\n",
    "}\n",
    "\n",
    "hb_config = {\n",
    "    \"net\": DiagonalNet(**model_config),\n",
    "    \"eta\": eta,\n",
    "    \"num_it\": num_it,\n",
    "    \"eta_euler\": eta_euler,\n",
    "    \"mu\": mu,\n",
    "}\n",
    "\n",
    "rgf_config = {\n",
    "    \"net\": DiagonalNet(**model_config),\n",
    "    \"eta\": eta,\n",
    "    \"num_it\": num_it * factor,\n",
    "    \"eta_euler\": eta_euler,\n",
    "    \"mu\": mu,\n",
    "}\n",
    "\n",
    "gd_config = {\n",
    "    \"net\": DiagonalNet(**model_config),\n",
    "    \"eta\": eta,\n",
    "    \"num_it\": num_it,\n",
    "    \"eta_euler\": eta_euler,\n",
    "    \"mu\": 0,\n",
    "}\n",
    "\n",
    "gf_config = {\n",
    "    \"net\": DiagonalNet(**model_config),\n",
    "    \"eta\": eta,\n",
    "    \"num_it\": num_it * factor,\n",
    "    \"eta_euler\": eta_euler,\n",
    "    \"mu\": 0,\n",
    "}\n",
    "\n",
    "print(\"hbf result\")\n",
    "hbf_history1, hbf_net1 = train(\n",
    "    X=X, y=y, w_star=w_star, return_loss=False, return_trajectory=True, **hbf_config\n",
    ")\n",
    "\n",
    "print(\"RGF result\")\n",
    "rgf_history1, rgf_net1 = train(\n",
    "    X=X, y=y, w_star=w_star, return_loss=False, return_trajectory=True, **rgf_config\n",
    ")\n",
    "print(\"HB result\")\n",
    "hb_history1, hb_net1 = hb_train(\n",
    "    X=X, y=y, w_star=w_star, return_loss=False, return_trajectory=True, **hb_config\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "reg coef 0.004999999999999999\n",
      "hbf result\n",
      "loss: 1.6988770961761475\n",
      "loss: 1.6962050199508667\n",
      "loss: 1.682157278060913\n",
      "loss: 1.6102204322814941\n",
      "loss: 1.3395087718963623\n",
      "loss: 0.9656050205230713\n",
      "loss: 0.7854816913604736\n",
      "loss: 0.6113219857215881\n",
      "loss: 0.4701389670372009\n",
      "loss: 0.4166480302810669\n",
      "loss: 0.3998384475708008\n",
      "loss: 0.3861301839351654\n",
      "loss: 0.36996695399284363\n",
      "loss: 0.35237064957618713\n",
      "loss: 0.3356262445449829\n",
      "loss: 0.32185712456703186\n",
      "loss: 0.312121719121933\n",
      "loss: 0.306091845035553\n",
      "loss: 0.30256590247154236\n",
      "loss: 0.30025163292884827\n",
      "loss: 0.298186331987381\n",
      "loss: 0.29578831791877747\n",
      "loss: 0.2927348017692566\n",
      "loss: 0.28885141015052795\n",
      "loss: 0.28408026695251465\n",
      "loss: 0.27838391065597534\n",
      "loss: 0.27175235748291016\n",
      "loss: 0.26427653431892395\n",
      "loss: 0.2560729682445526\n",
      "loss: 0.2472992092370987\n",
      "loss: 0.23818178474903107\n",
      "loss: 0.22899307310581207\n",
      "loss: 0.21997012197971344\n",
      "loss: 0.21132957935333252\n",
      "loss: 0.2032793164253235\n",
      "loss: 0.1959349513053894\n",
      "loss: 0.1893438845872879\n",
      "loss: 0.183492049574852\n",
      "loss: 0.1783057600259781\n",
      "loss: 0.17372044920921326\n",
      "loss: 0.16967475414276123\n",
      "loss: 0.16603542864322662\n",
      "loss: 0.16273163259029388\n",
      "loss: 0.1597135215997696\n",
      "loss: 0.15688790380954742\n",
      "loss: 0.15419261157512665\n",
      "loss: 0.15159274637699127\n",
      "loss: 0.14908641576766968\n",
      "loss: 0.1466350257396698\n",
      "loss: 0.1442166417837143\n",
      "loss: 0.14182020723819733\n",
      "loss: 0.13944032788276672\n",
      "loss: 0.13707615435123444\n",
      "loss: 0.13472969830036163\n",
      "loss: 0.1324014663696289\n",
      "loss: 0.13009962439537048\n",
      "loss: 0.1278306394815445\n",
      "loss: 0.12560026347637177\n",
      "loss: 0.12341520190238953\n",
      "loss: 0.1212807148694992\n",
      "loss: 0.11920252442359924\n",
      "loss: 0.11718635261058807\n",
      "loss: 0.11523550748825073\n",
      "loss: 0.11334553360939026\n",
      "loss: 0.11151833832263947\n",
      "loss: 0.10975542664527893\n",
      "loss: 0.10805580019950867\n",
      "loss: 0.10644251853227615\n",
      "loss: 0.1048983484506607\n",
      "loss: 0.10341992974281311\n",
      "loss: 0.10200441628694534\n",
      "loss: 0.10064636915922165\n",
      "loss: 0.09934508800506592\n",
      "loss: 0.098098985850811\n",
      "loss: 0.09691324084997177\n",
      "loss: 0.09577871859073639\n",
      "loss: 0.09469106048345566\n",
      "loss: 0.09364410489797592\n",
      "loss: 0.09263674169778824\n",
      "loss: 0.09166841208934784\n",
      "loss: 0.09073387086391449\n",
      "loss: 0.08983079344034195\n",
      "loss: 0.08895698189735413\n",
      "loss: 0.08812098205089569\n",
      "loss: 0.08731422573328018\n",
      "loss: 0.08652933686971664\n",
      "loss: 0.08576425164937973\n",
      "loss: 0.08501674234867096\n",
      "loss: 0.0842873603105545\n",
      "loss: 0.08358190953731537\n",
      "loss: 0.08291494846343994\n",
      "loss: 0.08226492255926132\n",
      "loss: 0.0816306620836258\n",
      "loss: 0.08101298660039902\n",
      "loss: 0.08042172342538834\n",
      "loss: 0.07984268665313721\n",
      "loss: 0.07927597314119339\n",
      "loss: 0.07872288674116135\n",
      "loss: 0.07818560302257538\n",
      "loss: 0.077666275203228\n",
      "RGF result\n",
      "loss: 1.6988600492477417\n",
      "loss: 1.6960110664367676\n",
      "loss: 1.6805607080459595\n",
      "loss: 1.5999830961227417\n",
      "loss: 1.3082010746002197\n",
      "loss: 0.949262797832489\n",
      "loss: 0.7744393348693848\n",
      "loss: 0.5989701151847839\n",
      "loss: 0.4645404517650604\n",
      "loss: 0.4153938591480255\n",
      "loss: 0.3991106152534485\n",
      "loss: 0.3852284252643585\n",
      "loss: 0.36890193819999695\n",
      "loss: 0.3512553572654724\n",
      "loss: 0.33459901809692383\n",
      "loss: 0.32100576162338257\n",
      "loss: 0.31146326661109924\n",
      "loss: 0.30555638670921326\n",
      "loss: 0.30207017064094543\n",
      "loss: 0.29972776770591736\n",
      "loss: 0.29758763313293457\n",
      "loss: 0.29508325457572937\n",
      "loss: 0.2919102609157562\n",
      "loss: 0.28787851333618164\n",
      "loss: 0.2829408645629883\n",
      "loss: 0.27706968784332275\n",
      "loss: 0.27026623487472534\n",
      "loss: 0.2626110315322876\n",
      "loss: 0.254241406917572\n",
      "loss: 0.24534489214420319\n",
      "loss: 0.2361222207546234\n",
      "loss: 0.2268841713666916\n",
      "loss: 0.21783548593521118\n",
      "loss: 0.2092207968235016\n",
      "loss: 0.20122641324996948\n",
      "loss: 0.1939575970172882\n",
      "loss: 0.18744924664497375\n",
      "loss: 0.1816801279783249\n",
      "loss: 0.17656609416007996\n",
      "loss: 0.17205311357975006\n",
      "loss: 0.16805900633335114\n",
      "loss: 0.16445668041706085\n",
      "loss: 0.16118735074996948\n",
      "loss: 0.15818679332733154\n",
      "loss: 0.15536712110042572\n",
      "loss: 0.15267081558704376\n",
      "loss: 0.15006297826766968\n",
      "loss: 0.14755310118198395\n",
      "loss: 0.1450900286436081\n",
      "loss: 0.14265741407871246\n",
      "loss: 0.14024673402309418\n",
      "loss: 0.13785263895988464\n",
      "loss: 0.13547584414482117\n",
      "loss: 0.13311898708343506\n",
      "loss: 0.130782812833786\n",
      "loss: 0.12847775220870972\n",
      "loss: 0.12620657682418823\n",
      "loss: 0.12397964298725128\n",
      "loss: 0.1217992976307869\n",
      "loss: 0.11967714130878448\n",
      "loss: 0.11761035025119781\n",
      "loss: 0.11561191082000732\n",
      "loss: 0.11367412656545639\n",
      "loss: 0.11179880052804947\n",
      "loss: 0.1099882498383522\n",
      "loss: 0.10824183374643326\n",
      "loss: 0.10656456649303436\n",
      "loss: 0.10497865825891495\n",
      "loss: 0.10346116125583649\n",
      "loss: 0.1020079180598259\n",
      "loss: 0.10061848163604736\n",
      "loss: 0.09929147362709045\n",
      "loss: 0.0980222299695015\n",
      "loss: 0.0968073159456253\n",
      "loss: 0.09564433991909027\n",
      "loss: 0.09453053027391434\n",
      "loss: 0.09345879405736923\n",
      "loss: 0.0924292579293251\n",
      "loss: 0.09143812954425812\n",
      "loss: 0.09048314392566681\n",
      "loss: 0.08956298977136612\n",
      "loss: 0.08867459744215012\n",
      "loss: 0.08782386779785156\n",
      "loss: 0.0870056226849556\n",
      "loss: 0.08620858937501907\n",
      "loss: 0.08543363213539124\n",
      "loss: 0.08467718213796616\n",
      "loss: 0.083938367664814\n",
      "loss: 0.08322396129369736\n",
      "loss: 0.0825401246547699\n",
      "loss: 0.08188191056251526\n",
      "loss: 0.0812402293086052\n",
      "loss: 0.08061383664608002\n",
      "loss: 0.08001205325126648\n",
      "loss: 0.07942798733711243\n",
      "loss: 0.0788569524884224\n",
      "loss: 0.078296959400177\n",
      "loss: 0.07774851471185684\n",
      "loss: 0.07722701877355576\n",
      "loss: 0.0767139345407486\n",
      "HB result\n",
      "loss: 0.4168793857097626\n",
      "loss: 0.30035385489463806\n",
      "loss: 0.24765311181545258\n",
      "loss: 0.174001082777977\n",
      "loss: 0.14441867172718048\n",
      "loss: 0.12144572287797928\n",
      "loss: 0.10359849035739899\n",
      "loss: 0.09177450835704803\n",
      "loss: 0.08369605988264084\n",
      "loss: 0.07777337729930878\n"
     ]
    }
   ],
   "source": [
    "mu = 0.6\n",
    "print(f\"reg coef {eta * (1 + mu) / (2 * (1 - mu)**2)}\")\n",
    "\n",
    "hbf_config = {\n",
    "    \"net\": DiagonalNet(**model_config),\n",
    "    \"eta\": eta,\n",
    "    \"num_it\": num_it * factor,\n",
    "    \"eta_euler\": eta_euler,\n",
    "    \"mu\": mu,\n",
    "    \"hbf\": True,\n",
    "}\n",
    "\n",
    "hb_config = {\n",
    "    \"net\": DiagonalNet(**model_config),\n",
    "    \"eta\": eta,\n",
    "    \"num_it\": num_it,\n",
    "    \"eta_euler\": eta_euler,\n",
    "    \"mu\": mu,\n",
    "}\n",
    "\n",
    "rgf_config = {\n",
    "    \"net\": DiagonalNet(**model_config),\n",
    "    \"eta\": eta,\n",
    "    \"num_it\": num_it * factor,\n",
    "    \"eta_euler\": eta_euler,\n",
    "    \"mu\": mu,\n",
    "}\n",
    "\n",
    "gd_config = {\n",
    "    \"net\": DiagonalNet(**model_config),\n",
    "    \"eta\": eta,\n",
    "    \"num_it\": num_it,\n",
    "    \"eta_euler\": eta_euler,\n",
    "    \"mu\": 0,\n",
    "}\n",
    "\n",
    "gf_config = {\n",
    "    \"net\": DiagonalNet(**model_config),\n",
    "    \"eta\": eta,\n",
    "    \"num_it\": num_it * factor,\n",
    "    \"eta_euler\": eta_euler,\n",
    "    \"mu\": 0,\n",
    "}\n",
    "\n",
    "print(\"hbf result\")\n",
    "hbf_history2, hbf_net2 = train(\n",
    "    X=X, y=y, w_star=w_star, return_loss=False, return_trajectory=True, **hbf_config\n",
    ")\n",
    "\n",
    "print(\"RGF result\")\n",
    "rgf_history2, rgf_net2 = train(\n",
    "    X=X, y=y, w_star=w_star, return_loss=False, return_trajectory=True, **rgf_config\n",
    ")\n",
    "print(\"HB result\")\n",
    "hb_history2, hb_net2 = hb_train(\n",
    "    X=X, y=y, w_star=w_star, return_loss=False, return_trajectory=True, **hb_config\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "def norm(array):\n",
    "    return (array**2).sum(dim=1)\n",
    "\n",
    "\n",
    "def cal_dis_error(hb, hbf, rgf):\n",
    "    hb_traj = torch.cat(hb, dim=1).T\n",
    "    hbf_traj = torch.cat(hbf, dim=1).T\n",
    "    rgf_traj = torch.cat(rgf, dim=1).T\n",
    "\n",
    "    hbf_diff = hbf_traj - hb_traj\n",
    "    rgf_diff = rgf_traj - hb_traj\n",
    "    return norm(hbf_diff), norm(rgf_diff)\n",
    "\n",
    "hbf_error, rgf_error = cal_dis_error(hb_history, hbf_history, rgf_history)\n",
    "hbf_error1, rgf_error1 = cal_dis_error(hb_history1, hbf_history1, rgf_history1)\n",
    "hbf_error2, rgf_error2 = cal_dis_error(hb_history2, hbf_history2, rgf_history2)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Figure 3: Discretization errors ∥wHBk − w(tk)∥2 2 for HBF (dotted lines) and RGF (solid lines), respectively, when training the 2-layer diagonal linear networks."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import numpy as np\n",
    "\n",
    "fig, ax = plt.subplots()\n",
    "num = len(rgf_error)\n",
    "pos = np.arange(0, num)\n",
    "it = 10000\n",
    "ax.plot(pos[:it], hbf_error2[:it], label=r\"HBF, $\\mu=0.6$\", linestyle=\"--\", c=\"black\")\n",
    "ax.plot(pos[:it], rgf_error2[:it], label=r\"RGF, $\\mu=0.6$\", linestyle=\"-\", c=\"black\")\n",
    "\n",
    "\n",
    "ax.plot(pos[:it], hbf_error[:it], label=r\"HBF, $\\mu=0.7$\", linestyle=\"--\", c=\"b\")\n",
    "# ax.plot(pos, hb_history, label=\"HB\", c=\"b\")\n",
    "ax.plot(pos[:it], rgf_error[:it], label=r\"RGF, $\\mu=0.7$\", linestyle=\"-\", c=\"b\")\n",
    "# ax.plot(pos, gd_history, label=\"GD\")\n",
    "# ax.plot(pos, gf_history, label=\"GF\")\n",
    "\n",
    "ax.plot(pos[:it], hbf_error1[:it], label=r\"HBF, $\\mu=0.8$\", linestyle=\"--\", c=\"m\")\n",
    "# ax.plot(pos, hb_history1, label=\"HB, mu=0.8\", c=\"m\")\n",
    "ax.plot(pos[:it], rgf_error1[:it], label=r\"RGF, $\\mu=0.8$\", linestyle=\"-\", c=\"m\")\n",
    "\n",
    "# ax.plot(pos, hbf_history2, label=\"HBF, mu=0.9\", linestyle=\"--\", c=\"r\")\n",
    "# ax.plot(pos, hb_history2, label=\"HB, mu=0.9\", c=\"r\")\n",
    "# ax.plot(pos, rgf_history2, label=\"RGF, mu=0.9\", linestyle=\"-.\", c=\"r\")\n",
    "\n",
    "# ax.set_xscale(\"log\")\n",
    "# ax.set_title()\n",
    "ax.set_ylabel(r\"$||\\mathbf{w}^{HB}_k - \\mathbf{w}(t_k)||^2_2$\", fontsize=14)\n",
    "\n",
    "ax.set_yscale(\"log\")\n",
    "ax.set_xlabel(r\"Iteration $k$\", fontsize=14)\n",
    "ax.grid(True)\n",
    "plt.tight_layout()\n",
    "plt.legend()\n",
    "plt.show()\n",
    "fig.savefig(\"./exp_fig/dln_dis_error.pdf\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "lora",
   "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.12.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
