{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "from copy import deepcopy\n",
    "\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "from sklearn.decomposition import PCA\n",
    "from torch import optim\n",
    "\n",
    "from utils import _build_dataset, train, plot_trajectory, hb_train, l2distance\n",
    "\n",
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "class DiagonalNet(nn.Module):\n",
    "    def __init__(self, scale, n_layer, d, gaussian=False, delta=None):\n",
    "        super(DiagonalNet, self).__init__()\n",
    "        if gaussian:\n",
    "            # p = scale * torch.randn((d, 1))\n",
    "            p = torch.normal(0, scale, size=(d, 1))\n",
    "            m = deepcopy(p)\n",
    "            # m = scale * torch.randn((d, 1))\n",
    "        else:\n",
    "            p, m = scale * torch.ones((d, 1)), scale * torch.ones((d, 1))\n",
    "\n",
    "        if delta is not None:\n",
    "            self.w_p = nn.Parameter(delta * p)\n",
    "            self.w_m = nn.Parameter(m / delta)\n",
    "        else:\n",
    "            self.w_p, self.w_m = nn.Parameter(p), nn.Parameter(m)\n",
    "        self.w_m_init = deepcopy(self.w_m)\n",
    "        self.n_layer = n_layer\n",
    "\n",
    "    def forward(self, x):\n",
    "        theta = self.w_p.pow(self.n_layer) - self.w_m.pow(self.n_layer)\n",
    "        return x @ theta\n",
    "\n",
    "    @property\n",
    "    def w(self):\n",
    "        return self.w_p.pow(self.n_layer) - self.w_m.pow(self.n_layer)\n",
    "\n",
    "    def loss(self, x, y):\n",
    "        return nn.MSELoss()(x @ self.w, y.reshape(-1, 1))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "random_seed = 1000\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": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "# X = 0.5 * X\n",
    "# y = 0.5 * y\n",
    "# w_star = 0.5 * w_star"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.11250000000000007\n"
     ]
    }
   ],
   "source": [
    "model_config = {\n",
    "    \"n_layer\": 2,\n",
    "    \"d\":d,\n",
    "    \"delta\": 0.9\n",
    "}\n",
    "lr = 1e-2\n",
    "mu = 0.8\n",
    "# factor = 10\n",
    "# lr_euler = lr / factor\n",
    "it_num = 10000\n",
    "\n",
    "# hb_net = DiagonalNet(**model_config)\n",
    "# gd_net = DiagonalNet(**model_config)\n",
    "# hbf_net = DiagonalNet(**model_config)\n",
    "# rgf_net = DiagonalNet(**model_config)\n",
    "\n",
    "coef = lambda p, eta: eta * (1 + p) / (4 * (1 - p)**2 )\n",
    "\n",
    "print(coef(mu, lr))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "from collections import defaultdict\n",
    "\n",
    "\n",
    "def train(net, lr, momentum=mu):\n",
    "    optimizer = optim.SGD(net.parameters(), lr=lr, momentum=momentum)\n",
    "    crition = nn.MSELoss()\n",
    "    it = 0\n",
    "    distances = [l2distance(net.w, w_star)]\n",
    "    while True:\n",
    "        y_hat = net(X)\n",
    "        l = crition(y_hat.reshape(-1), y.reshape(-1))\n",
    "        optimizer.zero_grad()\n",
    "        l.backward()\n",
    "        optimizer.step()\n",
    "\n",
    "        it += 1\n",
    "        # distances.append(l2distance(net.w, w_star))\n",
    "        distances.append(l.detach().clone().data)\n",
    "\n",
    "        if it % 1000 == 0:\n",
    "            print(l2distance(net.w, w_star))\n",
    "\n",
    "        if it >= it_num:\n",
    "            break\n",
    "\n",
    "    return distances\n",
    "\n",
    "\n",
    "def run(scale_list, lr, momentum_list):\n",
    "    momentum_exp_result = defaultdict(list)\n",
    "    for momentum in momentum_list:\n",
    "        for s in scale_list:\n",
    "            print(f\"training with init scale = {s}\")\n",
    "            model_config[\"scale\"] = s\n",
    "            net = DiagonalNet(**model_config)\n",
    "            test_error = train(net, lr, momentum=momentum)\n",
    "            momentum_exp_result[f\"{momentum}\"].append(test_error)\n",
    "    return momentum_exp_result"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "training with init scale = 0.01\n",
      "0.08440454\n",
      "0.0537887\n",
      "0.04426799\n",
      "0.039378617\n",
      "0.03640545\n",
      "0.034309298\n",
      "0.032709382\n",
      "0.031434886\n",
      "0.030391557\n",
      "0.029520981\n",
      "training with init scale = 0.6\n",
      "1.5331645\n",
      "1.5331645\n",
      "1.5331645\n",
      "1.5331645\n",
      "1.5331645\n",
      "1.5331645\n",
      "1.5331645\n",
      "1.5331645\n",
      "1.5331645\n",
      "1.5331645\n",
      "training with init scale = 0.01\n",
      "0.04991758\n",
      "0.04021183\n",
      "0.036766887\n",
      "0.035074245\n",
      "0.03411696\n",
      "0.033528753\n",
      "0.03314501\n",
      "0.032883678\n",
      "0.0326981\n",
      "0.03256268\n",
      "training with init scale = 0.6\n",
      "1.5175934\n",
      "1.5175934\n",
      "1.5175934\n",
      "1.5175934\n",
      "1.5175934\n",
      "1.5175934\n",
      "1.5175934\n",
      "1.5175934\n",
      "1.5175934\n",
      "1.5175934\n",
      "training with init scale = 0.01\n",
      "0.0571175\n",
      "0.051655386\n",
      "0.0501383\n",
      "0.049482495\n",
      "0.04915032\n",
      "0.048967633\n",
      "0.048861563\n",
      "0.04879744\n",
      "0.04875775\n",
      "0.048732683\n",
      "training with init scale = 0.6\n",
      "1.457758\n",
      "1.457758\n",
      "1.457758\n",
      "1.457758\n",
      "1.457758\n",
      "1.457758\n",
      "1.457758\n",
      "1.457758\n",
      "1.457758\n",
      "1.457758\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "\n",
    "scale_list = [0.01, 0.6]\n",
    "m_list = [0, 0.8, 0.9]\n",
    "exp_result = run(scale_list, lr=lr, momentum_list=m_list)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Figure 4 (b): L(w) during training processes of HB (μ = 0.9) and GD for different s."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 67,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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XvWpp1RPOEE1KMj0MbVvr3rzzaqLF5dfLPfrv75W7kW/Xf9tuefcudOsjXXTTeb9hg2qjz+ump5eo1CrraCwQOMeidMrZ11pf788ZPugimH7oCyrbnXsCFOmmNqAO3aY1lCPHmbNqX79oGtXM3x99eriz/Rr0w+AP/eU/8S3RQszb+rdd83E365dTT3AB31UIjNIghjcdnBArGllWf7KJijLp4oGWXzxhem3euUVU73iwQdN0urd21RFP7izGXtv3cs1A66p8Fz332+qt0+bZuZrDR4Mzz9vlqr3hSHthrAmbQ3peem+CcCyApxNUJbfaNLEFK797jtTff255yAmBv7zH9PqOuvUeN58Ew4dKv/zvXqZyulPPWUqUTz5pNn+++919zOUNKTdEABWpK/wTQBWndi5cycXXHABHTt2pF+/fhx33HF8/PHHpKSk0LhxY/r06UPXrl054YQT+Pzzz30dbkCxCcryS61amYEPP/9sBlk88AD8uTGTiy+G1q2VK680Sai8wRXBwab1NWCAWfNq5MjifSkppkBtXejXuh/BEsyfB+1A1PpKVRk3bhwnnHACGzZsYPHixcyYMYNU15oxQ4cOZenSpaxdu5bnnnuOa6+9lu+/9/+Vov2FTVCW32vf3owAfPaLr+DSoQwfs4d33zWP8Lp1MyMF08t5ihYUZEYTAuTlQWEh3H23WVKkLkSGRtKteTf+yvirbi5o1bkffviBsLAwJk6ceHhb+/btmTx58hHHJicnc8899/D888/XZYgBLRBKHVkWAMmte0P7nzjjtE9557XL+eAD0990ww0mgU2YYJYK6dmz+DPffGP2hYdD585mMMWsWTB1at0UnO3Tqg9f/vFl7V/I4oYbzJw7b0pONoN3KrJq1Sr69u3r9vn69u3L448/XuO4GgrbgrICRsf4jkSHRrN853JiYszgigULTDHbc881E4R79YLhw+GDDyA/36xl9dtv5lFh27amysXWrUeu5ltb+rbuy57cPWw/uL1uLmj51KRJk+jduzcDBgwod79WNeHPKsW2oKyAESRBHNX0KP7cW7pPp18/M1T98cfN1xdeMAkrKQluugmGDoWffoLLL4ecHPOZ334zran33oN//at46Q9v69va/HW9dMdSWse2rp2LWEDlLZ3a0qNHD2bOnHn4/dSpU0lLS6N///7lHr906VK61XSd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",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import numpy as np\n",
    "gd_result = exp_result[f\"{0}\"]\n",
    "hb_result = exp_result[f'{m_list[1]}']\n",
    "hb_result1 = exp_result[f'{m_list[2]}']\n",
    "\n",
    "# hb_result2 = exp_result[f\"{m_list[3]}\"]\n",
    "it = 5000\n",
    "colors = ['c', 'm', 'black', \"r\", \"g\"]\n",
    "\n",
    "fig, ax = plt.subplots()\n",
    "num = len(gd_result[0][:it])\n",
    "pos = np.arange(0, num)\n",
    "# for i, c in enumerate(colors[:len(gd_result)]):\n",
    "# i=1\n",
    "# ax.plot(pos, gd_result[i][:5000], label=f\"GD\", linestyle=\"-\", c=c)\n",
    "# ax.plot(pos, hb_result[i][:5000], label=f\"HB, mu={m_list[1]}\", linestyle=\"--\", c=c)\n",
    "# ax.plot(pos, hb_result1[i][:5000], label=f\"HB, mu={m_list[2]}\", linestyle=\"dotted\", c=c)\n",
    "# ax.plot(pos, hb_result2[i], label=f\"HB, mu={m_list[3]}\", linestyle=\"-.\", c=c)\n",
    "\n",
    "i = 0\n",
    "p1, = ax.plot(pos, gd_result[i][:it], label=r\"GD\", linestyle=\"-\", c=\"g\")\n",
    "# ax.plot(pos, hb_result[i][:it], label=r\"HB\", linestyle=\"--\", c=\"m\")\n",
    "p2, = ax.plot(pos, hb_result1[i][:it], label=f\"HB\", linestyle=\"--\", c=\"g\")\n",
    "\n",
    "# i = 1\n",
    "# ax.plot(pos, gd_result[i][:5000], label=f\"GD\", linestyle=\"-\", c=\"b\")\n",
    "# ax.plot(pos, hb_result[i][:5000], label=f\"HB, mu={m_list[1]}\", linestyle=\"--\", c=\"b\")\n",
    "# # ax.plot(pos, hb_result1[i][:5000], label=f\"HB, mu={m_list[2]}\", linestyle=\"dotted\", c=\"b\")\n",
    "\n",
    "par = ax.twinx()\n",
    "\n",
    "i = 1\n",
    "# ax.plot(pos, gd_result[i][:it], label=r\"GD, $s=0.6$\", linestyle=\"-\", c=\"b\")\n",
    "# ax.plot(pos, hb_result[i][:it], label=r\"HB\", linestyle=\"--\", c=\"b\")\n",
    "# ax.plot(pos, hb_result1[i][:it], label=f\"HB, mu={m_list[2]}\", linestyle=\"dotted\", c=\"b\")\n",
    "p3, = par.plot(pos, gd_result[i][:it], label=r\"GD\", linestyle=\"-\", c=\"b\")\n",
    "# par.plot(pos, hb_result[i][:it], label=r\"HB\", linestyle=\"--\", c=\"b\")\n",
    "p4, = par.plot(pos, hb_result1[i][:it], label=f\"HB\", linestyle=\"--\", c=\"b\")\n",
    "\n",
    "\n",
    "ax.set_xscale(\"log\")\n",
    "ax.tick_params(axis=\"y\", labelcolor=\"g\")\n",
    "# ax.set_yscale(\"log\")\n",
    "ax.grid(True)\n",
    "ax.set_xlabel(r\"Iteration $k$\", fontsize=14,)\n",
    "ax.set_ylabel(r\"$||\\mathbf{w}_k - \\mathbf{w}^{*}||_2$\", fontsize=14, color=\"g\")\n",
    "par.tick_params(axis=\"y\", labelcolor=\"b\")\n",
    "par.set_yscale(\"log\")\n",
    "par.set_ylabel(r\"$L(\\mathbf{w})$, $s=0.6$\", fontsize=\"14\", color=\"b\")\n",
    "ax.set_ylabel(r\"$L(\\mathbf{w})$, $s=0.01$\", fontsize=\"14\", color=\"g\")\n",
    "ax.legend(handles=[p1, p2,p3,p4])\n",
    "plt.tight_layout()\n",
    "plt.show()\n",
    "fig.savefig(\"./exp_fig/dln_train_error.pdf\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "2"
      ]
     },
     "execution_count": 46,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(gd_result)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "base",
   "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.9.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
