{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "sys.path.append('/path/to/the/folder')\n",
    "from GradMethods.algorithms import Algorithm, GD\n",
    "import torch\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib\n",
    "import math\n",
    "from collections import defaultdict\n",
    "import numpy as np\n",
    "from matplotlib.patches import Rectangle\n",
    "import matplotlib.patches as patches"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.rcParams[\"figure.autolayout\"] = True\n",
    "plt.rcParams.update({\n",
    "    \"text.usetex\": True,\n",
    "    'font.size'   : 20})\n",
    "\n",
    "import matplotlib as mpl\n",
    "mpl.rcParams.update(mpl.rcParamsDefault)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "def power_norm(x, p=2.1):\n",
    "    return (torch.norm(x) ** p)/p\n",
    "\n",
    "def grad_power_norm(x, p=2.1):\n",
    "    return torch.norm(x) ** (p - 2) * x"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "def experiment_run(p=4 , n_iter=500,n_dim=10):\n",
    "    L_1= p-2\n",
    "    L_0 = ((p-2)/ L_1)**(p-2)\n",
    "    solution = torch.zeros(n_dim)\n",
    "    method = GD(L_0, L_1, power_norm, grad_power_norm, solution, lr_type=\"first_type\")\n",
    "    res = method.run(n_iter)\n",
    "    distances = np.array(res[\"Dist2Sol\"])\n",
    "    func_rs = np.array(res[\"func_residual\"])\n",
    "\n",
    "    dif_L_1 = 2\n",
    "    dif_L_0 = ((p-2)/ dif_L_1 )**(p-2)\n",
    "    print(dif_L_0, dif_L_1)\n",
    "    method_diff = GD(dif_L_0, dif_L_1, power_norm, grad_power_norm, solution, lr_type=\"second_type\")\n",
    "    res_diff = method_diff.run(n_iter)\n",
    "    distances_diff = np.array(res_diff[\"Dist2Sol\"])\n",
    "    func_rs_diff = np.array(res_diff[\"func_residual\"])\n",
    "\n",
    "    dif_L_1 = 2\n",
    "    dif_L_0 = ((p-2)/ dif_L_1 )**(p-2)\n",
    "    print(dif_L_0, dif_L_1)\n",
    "    method_clip = GD(dif_L_0, dif_L_1, power_norm, grad_power_norm, solution, lr_type=\"clipped\")\n",
    "    res_clip = method_clip.run(n_iter)\n",
    "    distances_clip = np.array(res_clip[\"Dist2Sol\"])\n",
    "    func_rs_clip = np.array(res_clip[\"func_residual\"])\n",
    "\n",
    "    method_polyak = GD(L_0, L_1, power_norm, grad_power_norm, solution, lr_type=\"polyak\")\n",
    "    res_polyak = method_polyak.run(n_iter)\n",
    "    distances_polyak = np.array(res_polyak[\"Dist2Sol\"])\n",
    "    func_rs_polyak = np.array(res_polyak[\"func_residual\"])\n",
    "\n",
    "\n",
    "    method_norm = GD(L_0, L_1, power_norm, grad_power_norm, solution, lr_type=\"normalized\")\n",
    "    res_normalized = method_norm.run(n_iter)\n",
    "    distances_normalized = np.array(res_normalized[\"Dist2Sol\"])\n",
    "    func_rs_normalized = np.array(res_normalized[\"func_residual\"])\n",
    "\n",
    "    plt.plot(func_rs_normalized, label=\"NGD\",   marker=\".\", markersize=20, markevery=50, linewidth=0.75)\n",
    "    plt.plot(func_rs, label=\"GD 1\",   marker=\"s\", markevery=50, linewidth=0.75)\n",
    "    plt.plot(func_rs_diff, label=\"GD 2\",   marker=\"s\", markevery=50, linewidth=0.75)\n",
    "    plt.plot(func_rs_clip, label=\"Clip GD\",   marker=\"v\", markevery=50, linewidth=0.75)\n",
    "    plt.plot(func_rs_polyak, label=\"PS GD\",   marker=\"x\", markevery=50, linewidth=0.75)\n",
    "\n",
    "\n",
    "    plt.yscale('log')\n",
    "    plt.title('Convergence', fontsize=20)\n",
    "    plt.ylabel(\"$f(x_k) - f^*$\", fontsize=\"15\")\n",
    "    plt.xlabel(\"Number of iterations\", fontsize=\"15\")\n",
    "    plt.legend( fontsize=\"15\")\n",
    "    plt.savefig(\"exp1_p_{}.pdf\".format(p), dpi=1200)\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "n_dim = 10\n",
    "n_iter = 500\n",
    "ps = [4.0, 6.0, 8.0]\n",
    "for p in ps:\n",
    "    experiment_run(p , n_iter,n_dim)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "n_dim = 10\n",
    "n_iter = 500\n",
    "y = torch.zeros(n_dim)\n",
    "p = 8\n",
    "def experiment_run_second(L_1s, p, n_iter,n_dim):\n",
    "    results = defaultdict()\n",
    "    for L_1 in L_1s:\n",
    "        L_0 = ((p-2)/ L_1)**(p-2)\n",
    "        method = GD(L_0, L_1, power_norm, grad_power_norm, y, lr_type=\"first_type\")\n",
    "        res = method.run(n_iter)\n",
    "        distances = np.array(res[\"Dist2Sol\"])\n",
    "        func_rs = np.array(res[\"func_residual\"])\n",
    "        results[L_1] = func_rs\n",
    "    for L_1 in L_1s:\n",
    "        plt.plot(results[L_1], label=\"$L_1 = {}$\".format(L_1),   marker=\"s\", markevery=50, linewidth=0.75)\n",
    "    plt.yscale('log')\n",
    "    plt.title('Convergence', fontsize=20)\n",
    "    plt.ylabel(\"$f(x_k) - f^*$\", fontsize=\"15\")\n",
    "    plt.xlabel(\"Number of iterations\", fontsize=\"15\")\n",
    "    plt.legend( fontsize=\"15\")\n",
    "    plt.savefig(\"exp2_p_{}.pdf\".format(p), dpi=1200)\n",
    "    plt.show()\n",
    "    plt.yscale('log')\n",
    "    plt.title('Convergence', fontsize=20)\n",
    "    plt.ylabel(\"$f(x_k) - f^*$\", fontsize=\"15\")\n",
    "    plt.xlabel(\"Number of iterations\", fontsize=\"15\")\n",
    "    plt.legend( fontsize=\"15\")\n",
    "    plt.savefig(\"exp2_p8.pdf\", dpi=1200)\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "L_1s = [1.0, 2.0, 4.0, 8.0, 16.0]\n",
    "ps = [4.0, 6.0, 8.0]\n",
    "for p in ps:\n",
    "    experiment_run_second(L_1s, p, n_iter,n_dim)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3.8.2 64-bit",
   "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.2"
  },
  "orig_nbformat": 4,
  "vscode": {
   "interpreter": {
    "hash": "aee8b7b246df8f9039afb4144a1f6fd8d2ca17a180786b69acc140d282b71a49"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
