{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import sys\n",
    "sys.path.append(\"../\")\n",
    "\n",
    "import os\n",
    "\n",
    "import numpy as np\n",
    "import random\n",
    "import tensorflow as tf\n",
    "import numpy as np\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "GLOBAL_SEED = 1\n",
    "\n",
    "os.environ['PYTHONHASHSEED']=str(GLOBAL_SEED)\n",
    "os.environ['TF_CUDNN_DETERMINISTIC'] = '1'\n",
    "random.seed(GLOBAL_SEED)\n",
    "np.random.seed(GLOBAL_SEED)\n",
    "tf.random.set_seed(GLOBAL_SEED)\n",
    "\n",
    "#os.environ[\"CUDA_VISIBLE_DEVICES\"] = \"-1\" # disable GPU\n",
    "dtype = 'float64'\n",
    "tf.keras.backend.set_floatx(dtype)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "w = np.random.randn(2) * 0.001\n",
    "b = np.random.randn(1) * 0.001\n",
    "\n",
    "eps = 1e-4  # vary this parameters: eps \\in {1e-4, 1e-3, 1e-2, 1e-1}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "os.makedirs(\"./figs\", exist_ok=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "norm_w = np.sqrt(w[0]**2 + w[1]**2)\n",
    "lambda_ = b[0] / norm_w\n",
    "theta = np.arccos(w[0] / norm_w)\n",
    "feature_loc_x = -1.0*lambda_*np.cos(theta)\n",
    "feature_loc_y = -1.0*lambda_*np.sin(theta)\n",
    "\n",
    "plt.style.use('ggplot')\n",
    "plt.figure(figsize=(6,6))\n",
    "# origin\n",
    "plt.scatter([0], [0], s=50, c='black')\n",
    "\n",
    "# original feature\n",
    "plt.axline([feature_loc_x, feature_loc_y], [(feature_loc_x**2+feature_loc_y**2)/feature_loc_x, 0], label=\"Original feature\", c='brown', linewidth=5)\n",
    "plt.scatter([feature_loc_x], [feature_loc_y], s=50, c='brown')\n",
    "plt.plot([0, feature_loc_x], [0, feature_loc_y], c=\"brown\", linestyle=\"dotted\", linewidth=5)\n",
    "\n",
    "# standard parameterization: perturbed feature\n",
    "w_eps_std = w + eps\n",
    "b_eps_std = b + eps\n",
    "norm_w_eps_std = np.sqrt(w_eps_std[0]**2 + w_eps_std[1]**2)\n",
    "lambda_eps_std = b_eps_std[0] / norm_w_eps_std\n",
    "theta_eps_std = np.arccos(w_eps_std[0] / norm_w_eps_std)\n",
    "feature_loc_x_eps_std = -1.0*lambda_eps_std*np.cos(theta_eps_std)\n",
    "feature_loc_y_eps_std = -1.0*lambda_eps_std*np.sin(theta_eps_std)\n",
    "plt.axline([feature_loc_x_eps_std, feature_loc_y_eps_std], [(feature_loc_x_eps_std**2+feature_loc_y_eps_std**2)/feature_loc_x_eps_std, 0], label=\"SP after perturbation\", c='red')\n",
    "plt.scatter([feature_loc_x_eps_std], [feature_loc_y_eps_std], s=50, c='red')\n",
    "plt.plot([0, feature_loc_x_eps_std], [0, feature_loc_y_eps_std], c=\"red\", linestyle=\"dashed\")\n",
    "\n",
    "# weightnorm: perturbed feature\n",
    "w_eps_bn = (w + eps) / np.sqrt((w+eps)[0]**2 + (w+eps)[1]**2) * (np.sqrt(w[0]**2 + w[1]**2) + eps)\n",
    "b_eps_bn = b + eps\n",
    "norm_w_eps_bn = np.sqrt(w_eps_bn[0]**2 + w_eps_bn[1]**2)\n",
    "lambda_eps_bn = b_eps_bn[0] / norm_w_eps_bn\n",
    "theta_eps_bn = np.arccos(w_eps_bn[0] / norm_w_eps_bn)\n",
    "feature_loc_x_eps_bn = -1.0*lambda_eps_bn*np.cos(theta_eps_bn)\n",
    "feature_loc_y_eps_bn = -1.0*lambda_eps_bn*np.sin(theta_eps_bn)\n",
    "plt.axline([feature_loc_x_eps_bn, feature_loc_y_eps_bn], [(feature_loc_x_eps_bn**2+feature_loc_y_eps_bn**2)/feature_loc_x_eps_bn, 0], label=\"WN after perturbation\", c='green')\n",
    "plt.scatter([feature_loc_x_eps_bn], [feature_loc_y_eps_bn], s=50, c='green')\n",
    "plt.plot([0, feature_loc_x_eps_bn], [0, feature_loc_y_eps_bn], c=\"green\", linestyle=\"dashed\")\n",
    "\n",
    "# geometric parameterization: perturbed feature\n",
    "lambda_eps = lambda_ + eps\n",
    "theta_eps = theta + eps\n",
    "feature_loc_x_eps = -1.0*lambda_eps*np.cos(theta_eps)\n",
    "feature_loc_y_eps = -1.0*lambda_eps*np.sin(theta_eps)\n",
    "plt.axline([feature_loc_x_eps, feature_loc_y_eps], [(feature_loc_x_eps**2+feature_loc_y_eps**2)/feature_loc_x_eps, 0], label=\"GmP after perturbation\", c='blue')\n",
    "plt.scatter([feature_loc_x_eps], [feature_loc_y_eps], s=50, c='blue')\n",
    "plt.plot([0, feature_loc_x_eps], [0, feature_loc_y_eps], c=\"blue\", linestyle=\"dashed\")\n",
    "\n",
    "\n",
    "plt.xlim(-1.0, 0.75)\n",
    "plt.ylim(-1.0, 0.75)\n",
    "if eps == 1e-4:\n",
    "    plt.legend(fontsize=21, loc=\"lower left\")\n",
    "plt.xlabel(\"$x_1$\", fontsize=35)\n",
    "plt.ylabel(\"$x_2$\", fontsize=35)\n",
    "plt.xticks([])\n",
    "plt.yticks([])\n",
    "plt.grid(True)\n",
    "# plt.savefig(\"./figs/parameter_perturbation_{}.pdf\".format(eps), format=\"pdf\", bbox_inches=\"tight\")"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "gp",
   "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"
  },
  "orig_nbformat": 4,
  "vscode": {
   "interpreter": {
    "hash": "630694aee0ba86fd16b724d8128d1f0c1677f298824c71c2eb5a498d077f2247"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
