{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import sys \n",
    "sys.path.append('path/to/folder')\n",
    "from algorithms import Algorithm, SGD, Popov, Extragradient\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\n",
    "from concurrent.futures import ProcessPoolExecutor, as_completed\n",
    "from itertools import product"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "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": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def run_experiment(n_exper=10, n_samples=1, dim=10, mu=1.0, L=1.0, n_steps=20000,\\\n",
    "     sampler=None, trashold=None, noise=False, twod=False, p=1.5, lr_p=None, \\\n",
    "        scheduler_lr=\"q\", init_points = None, closeness=1, q=2/3, noise_ratio=5.0):\n",
    "    results = defaultdict(list)\n",
    "    results[\"projection\"] = []\n",
    "    results[\"popov\"] = []\n",
    "    results[\"kappa\"] = []\n",
    "    if init_points is None:\n",
    "        init_points = [closeness * torch.randn(2 * dim), closeness * torch.randn(2 * dim)]\n",
    "    for i in range(n_exper):\n",
    "        if i % 5 == 0:\n",
    "            print(i)\n",
    "        # J = torch.tensor([[[0.0, 1.0],\n",
    "        # [-1.0, 0.0 ]]])\n",
    "        # abs =  torch.zeros(1, 2)\n",
    "        print(\"dim\", dim)\n",
    "        \n",
    "        sgd = SGD(p=p, dim=dim, noise_ratio = noise_ratio,lr_0=lr_p,  noise=noise, init=init_points)\n",
    "        res = sgd.run(n_steps=n_steps, sampler=sampler, scheduler=\"q\", q=q, trashold=1)\n",
    "        results[\"projection\"].append(res[\"Dist2Sol\"])\n",
    "\n",
    "        sgd = SGD(p=p, dim=dim, noise_ratio=noise_ratio, lr_0=lr_p, noise=noise, init=init_points, same=True)\n",
    "        res = sgd.run(n_steps=n_steps, sampler=sampler, scheduler=\"q\", q=q, trashold=1)\n",
    "        results[\"projection_same\"].append(res[\"Dist2Sol\"])\n",
    "        \n",
    "        popov = Popov(p=p, dim=dim, noise_ratio=noise_ratio, lr_0=lr_p, noise=noise, init=init_points)\n",
    "        res_popov = popov.run(n_steps=n_steps, sampler=sampler, scheduler=scheduler_lr, q=q, trashold=trashold)\n",
    "        results[\"popov\"].append(res_popov[\"Dist2Sol\"])\n",
    "\n",
    "        extra = Extragradient( p=p, dim=dim, noise_ratio=noise_ratio,  lr_0=lr_p, noise=noise, init=init_points)\n",
    "        res_extra = extra.run(n_steps=n_steps, sampler=sampler, scheduler=scheduler_lr, q=q, trashold=trashold)\n",
    "\n",
    "        results[\"extra\"].append(res_extra[\"Dist2Sol\"])\n",
    "        results[\"kappa\"].append(extra.L / extra.mu)\n",
    "    results[\"extra\"] = np.array(results[\"extra\"])\n",
    "    results[\"popov\"] = np.array(results[\"popov\"])\n",
    "    results[\"projection\"] = np.array(results[\"projection\"])\n",
    "    results[\"projection_same\"] = np.array(results[\"projection_same\"])\n",
    "    results[\"kappa\"] = np.array(results[\"kappa\"])\n",
    "    results[\"points_proj\"] = res[\"u\"]\n",
    "    results[\"points_pop\"] = res_popov[\"u\"]\n",
    "    results[\"points_extr\"] = res_extra[\"u\"]\n",
    "    plt.figure(figsize=(8,6))\n",
    "    plt.plot(results[\"projection\"].mean(0), label=\"Projection\", marker=\"o\", markevery=n_steps//10)\n",
    "    plt.plot(results[\"projection_same\"].mean(0), label=\"Projection-Same\", marker=\"o\", markevery=n_steps//10)\n",
    "    plt.plot(results[\"popov\"].mean(0), label=\"Popov\",   marker=\"o\", markevery=n_steps//10)\n",
    "    plt.plot(results[\"extra\"].mean(0), label=\"Korpelevich\",   marker=\"s\", markevery=n_steps//10)\n",
    "    plt.yscale('log')\n",
    "\n",
    "    plt.title('Convergence', fontsize=20)\n",
    "    plt.ylabel(\"Distance to solution\", fontsize=\"15\")\n",
    "    plt.xlabel(\"Number of iterations\", fontsize=\"15\")\n",
    "    if trashold is  None:\n",
    "        trashold = n_steps//2\n",
    "    plt.legend( fontsize=\"15\")\n",
    "    plt.savefig(\"/Users/davankov/Downloads/generalized-svi-tmlr/experiment/run_200k/hh_Exp_p_{:.1f}_sch_{}_close_{}_noise{}{}_q{}_n_steps{}.pdf\".format(p, scheduler_lr, closeness, noise,noise_ratio, q, n_steps))\n",
    "    plt.close() \n",
    "        \n",
    "    return results"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Experiments ##"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "dim=10\n",
    "n_samples = 1\n",
    "mu = 0.05\n",
    "L = 1.0\n",
    "# qs = [2/3]\n",
    "qs = [1/2+0.001, 2/3, 1 - 0.001]\n",
    "n_exper=20\n",
    "ps = [2.5]  #[1.2, 1.7]#[2.5, 3.0, 6.0]\n",
    "close = [1.0]\n",
    "n_steps = 10000\n",
    "schs = [\"q\"]\n",
    "for q in qs:\n",
    "    for p in ps:\n",
    "        for cl in close:\n",
    "            for sch in schs:\n",
    "                res__ =run_experiment(n_exper, n_samples, dim, mu, L, n_steps=n_steps, p=p, sampler=None, trashold=1, noise=True, scheduler_lr=sch, closeness=cl, q=q)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "a = torch.rand(10)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# m = a.shape[0] //2\n",
    "# a[0:m], a[m:]\n",
    "x = torch.rand(10)\n",
    "midle = x.shape[0] //2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "tens = torch.cat([torch.pow(torch.norm(x[0: midle]), 4 - 2) * x[0: midle] + x[midle: ],\\\n",
    "             torch.pow(torch.norm(x[midle: ]), 4- 2) * x[midle: ] -  x[0: midle]])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.pow(torch.norm(x[0: midle]), 4 - 2) * x[0: midle] + x[midle: ]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.pow(torch.norm(x[midle: ]), 4- 2) * x[midle: ] -  x[0: midle]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "tens"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "dim=5\n",
    "n_samples = 1\n",
    "mu = 0.05\n",
    "L = 1.0\n",
    "# qs = [2/3]\n",
    "qs = [0.99]#[1/2+0.001, 2/3, 1 - 0.001]\n",
    "n_exper=10\n",
    "ps = [2.5, 3.0, 6.0]\n",
    "close = [0.1]\n",
    "n_steps = 4000000\n",
    "schs = [\"q\"]\n",
    "for q in qs:\n",
    "    for p in ps:\n",
    "        for cl in close:\n",
    "            for sch in schs:\n",
    "                res__ =run_experiment(n_exper, n_samples, dim, mu, L, n_steps=n_steps, p=p, sampler=None, trashold=1, noise=True, scheduler_lr=sch, closeness=cl, q=q)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "\n",
    "param_grid = {\n",
    "    \"dim\": [5],\n",
    "    \"n_samples\": [1],\n",
    "    \"mu\" : [0.05],\n",
    "    \"L\" : [1.0],\n",
    "    \"q\" : [0.99], #[1/2+0.001, 2/3, 1 - 0.001]\n",
    "    \"n_exper\" : [10],\n",
    "    \"p\" : [1.5, 2.5, 3.0, 6.0],\n",
    "    \"closeness\" : [0.1],\n",
    "    \"n_steps\" : [1000000],\n",
    "    \"scheduler_lr\" : [\"q\"],\n",
    "    \"sampler\": [None],\n",
    "    \"trashold\": [1],\n",
    "    \"noise\" : [True],\n",
    "    \"lr_p\" : []\n",
    "}\n",
    "\n",
    "configs = [dict(zip(param_grid.keys(), values)) \n",
    "           for values in product(*param_grid.values())]\n",
    "results = []"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from joblib import Parallel, delayed\n",
    "# 3) parallel map\n",
    "results = Parallel(n_jobs=-1, prefer=\"processes\")(\n",
    "    delayed(run_experiment)(**cfg) for cfg in configs\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# init_points = [ 10.0 * torch.randn(2 * 5), 10.0 * torch.randn(2 * 5)]\n",
    "param_grid = {\n",
    "    \"dim\": [5],\n",
    "    \"n_samples\": [1],\n",
    "    \"mu\" : [0.05],\n",
    "    \"L\" : [1.0],\n",
    "    \"q\" : [1/2+0.001, 2/3, 1 - 0.001],\n",
    "    \"n_exper\" : [10],\n",
    "    \"p\" : [1.5, 2.5, 4.0, 6.0],\n",
    "    \"closeness\" : [10.0],\n",
    "    \"n_steps\" : [20000],\n",
    "    \"scheduler_lr\" : [\"q\"],\n",
    "    \"sampler\": [None],\n",
    "    \"trashold\": [1],\n",
    "    \"noise\" : [True],\n",
    "    \"lr_p\" : [1.0],\n",
    "    \"noise_ratio\" : [0.5, 5.0, 10.0],\n",
    "    \"init_points\" : [init_points]\n",
    "}\n",
    "\n",
    "configs = [dict(zip(param_grid.keys(), values)) \n",
    "           for values in product(*param_grid.values())]\n",
    "new_results = []"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from joblib import Parallel, delayed\n",
    "# 3) parallel map\n",
    "new_results = Parallel(n_jobs=-1, prefer=\"processes\")(\n",
    "    delayed(run_experiment)(**cfg) for cfg in configs\n",
    ")"
   ]
  }
 ],
 "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.9.6"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
