{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "from src.utils.training import get_objective, get_optimizer\n",
    "from src.utils.data import load_dataset\n",
    "from scipy.optimize import minimize\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "import torch\n",
    "from tqdm import tqdm\n",
    " \n",
    "from src.optim.SOREL import Sorel\n",
    "from src.optim.SOREL_batch import SorelBatch"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Load Dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "dataset = \"yacht\" # [\"yacht\", \"concrete\", \"energy\", \"kin8nm\", \"power\"]\n",
    "\n",
    "X_train, y_train, X_test, y_test = load_dataset(dataset)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Get Objective and Optimizer"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Build objective.\n",
    "model_cfg = {\n",
    "    \"objective\": \"cvar\", # Options: 'cvar', 'extremile', 'esrm'. \n",
    "    \"para_value\": 0.5,   # to apply alpha-cvar, rho-esrm or r-extremile,  set para_value = 1-alpha (cvar) or rho (esrm) or r (extremile).\n",
    "    \"l2_reg\": 1.0,\n",
    "    \"loss\": \"squared_error\",  # Options: 'squared_error', 'binary_cross_entropy', 'multinomial_cross_entropy'.\n",
    "    \"n_class\": None,\n",
    "    \"shift_cost\": 0,\n",
    "}\n",
    "autodiff = False # non-autodiff variants\n",
    "train_obj = get_objective(model_cfg, X_train, y_train, autodiff=autodiff)\n",
    "val_obj   = get_objective(model_cfg, X_test, y_test, autodiff=autodiff)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "lr = 0.03\n",
    "lrd = 0.2\n",
    "seed = 1\n",
    "optimizer = Sorel(\n",
    "                train_obj,\n",
    "                lr=lr,\n",
    "                smooth_coef=0,\n",
    "                smoothing=\"l2\",\n",
    "                seed=seed,\n",
    "                length_epoch=train_obj.n,\n",
    "                lrdcon=lrd,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "# You can get baseline optimizers by running the following code.\n",
    "# optim_cfg = {\n",
    "#         \"optimizer\": \"sgd\", # Options: 'sgd', 'lsvrg', 'prospect'\n",
    "#         \"lr\": lr,\n",
    "#         \"epoch_len\": train_obj.n, # Used as an update interval for LSVRG, and otherwise is simply a logging interval for other methods.\n",
    "#         \"shift_cost\": 0,\n",
    "#     }\n",
    "\n",
    "# optimizer = get_optimizer(optim_cfg, train_obj, seed)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Training"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "L-BGFS sucsess! Minimum loss on yacht: 0.2959\n"
     ]
    }
   ],
   "source": [
    "# Get the optimal solution by L-BFGS\n",
    "train_obj_ = get_objective(model_cfg, X_train, y_train)\n",
    "\n",
    "# Define function and Jacobian oracles.\n",
    "def fun(w):\n",
    "    return train_obj_.get_batch_loss(torch.tensor(w, dtype=torch.float64)).item()\n",
    "\n",
    "def jac(w):\n",
    "    return (\n",
    "        train_obj_.get_batch_subgrad(\n",
    "            torch.tensor(w, dtype=torch.float64, requires_grad=True)\n",
    "        )\n",
    "        .detach()\n",
    "        .numpy()\n",
    "    )\n",
    "\n",
    "# Run optimizer.\n",
    "d = train_obj.d\n",
    "init = np.zeros((d,), dtype=np.float64)\n",
    "if model_cfg[\"n_class\"]:\n",
    "    init = np.zeros((model_cfg[\"n_class\"] * d,), dtype=np.float64)\n",
    "else:\n",
    "    init = np.zeros((d,), dtype=np.float64)\n",
    "output = minimize(fun, init, method=\"L-BFGS-B\", jac=jac)\n",
    "if output.success:\n",
    "    print(f\"L-BGFS sucsess! Minimum loss on {dataset}: {output.fun:0.4f}\")\n",
    "else:\n",
    "    raise Exception(output.message)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [00:01<00:00, 62.15it/s]\n"
     ]
    }
   ],
   "source": [
    "n_epochs = 100\n",
    "\n",
    "train_losses = []\n",
    "epoch_len = optimizer.get_epoch_len()\n",
    "for epoch in tqdm(range(n_epochs)):\n",
    "\n",
    "    optimizer.start_epoch()\n",
    "    for _ in range(epoch_len):\n",
    "        optimizer.step()\n",
    "    optimizer.end_epoch()\n",
    "    train_losses.append(train_obj.get_batch_loss(optimizer.weights).item())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 500x500 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Display training suboptimality.\n",
    "\n",
    "fig, ax = plt.subplots(1, 1, figsize=(5, 5))\n",
    "\n",
    "t = np.arange(len(train_losses))\n",
    "\n",
    "minimum_loss = output.fun\n",
    "train_losses = np.array(train_losses)\n",
    "subopt = (train_losses - minimum_loss) / (train_losses[0] - minimum_loss)\n",
    "\n",
    "ax.plot(np.arange(len(subopt)), subopt, linewidth=3, label=r\"Train Suboptimality\")\n",
    "ax.set_xlabel(\"Epochs\")\n",
    "\n",
    "ax.legend()\n",
    "ax.set_yscale(\"log\")\n",
    "plt.show()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "lrm",
   "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.8.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
