{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# new plot for fBM to demonstrate the \\sqrt{H} convergence rates\n",
    "import numpy as np\n",
    "import numpy.linalg as la\n",
    "import matplotlib.pyplot as plt\n",
    "from fbm import FBM\n",
    "from tikzplotlib import save as tikz_save\n",
    "from funcs import *\n",
    "import seaborn as sns\n",
    "colors = sns.color_palette()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# define vector field\n",
    "d = 1 # dimensionality\n",
    "theta = 1. # strenght of drift\n",
    "sig = 0.001\n",
    "Hess = theta * np.eye(d) # Hessian"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def f(x):\n",
    "    return x.T @ Hess @ x\n",
    "def nabla_f(x):\n",
    "    return Hess @ x"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "N_sim = 1000 # number of simulations\n",
    "N_iter = 10000 # number of steps\n",
    "print('Running code with sig = %.3f and N_iter = %d.'%(sig,N_iter))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "eta = 1./N_iter"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# define gradient descent perturbed by specific increments\n",
    "def GD_perturbed_by_incrms(f,nabla_f,x0,eta,sig,N_iter,increments):\n",
    "    dim = len(x0)\n",
    "    xs = np.zeros([N_iter,dim])\n",
    "    xs[0] = x0\n",
    "    for i in range(N_iter-1):\n",
    "        xs[i+1] = xs[i] - eta*nabla_f(xs[i]) + sig*increments[i]\n",
    "    return xs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "x0 = np.zeros([d,])\n",
    "Hs = np.round(np.arange(0.1,0.95,0.05),2).tolist()\n",
    "\n",
    "xs_fbm = np.zeros([len(Hs),N_sim,N_iter,d])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "for sim_idx in range(N_sim):\n",
    "    if sim_idx%100 == 0: print('number of sims = %d'%sim_idx)\n",
    "    for idx,H in enumerate(Hs):\n",
    "        fbmobj = FBM(n=N_iter, hurst=H, length=1.)\n",
    "        incrs = sig*sample_fbm_perturbations(fbmobj,N_iter,d)\n",
    "        xs_fbm[idx,sim_idx] = GD_perturbed_by_incrms(f=f,nabla_f=nabla_f,x0=x0,eta=eta,sig=sig,N_iter=N_iter,increments=incrs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "### data post-processing, computing the mean over the supremum\n",
    "sup_fbm = np.max(np.linalg.norm(xs_fbm[:,:,:,:],axis=3),axis=2)\n",
    "mean_sup_fbm = np.mean(sup_fbm,axis=1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# compute the least squares\n",
    "Phi = np.zeros([len(Hs),2])\n",
    "Phi[:,0] = np.ones([len(Hs)])\n",
    "sqrtm_H = [1/np.sqrt(H) for H in Hs]\n",
    "Phi[:,1] = sqrtm_H\n",
    "w_ML = np.linalg.inv(Phi.T @ Phi) @ Phi.T @ mean_sup_fbm\n",
    "def ml_predictor(H):\n",
    "    return w_ML[0] + w_ML[1]/np.sqrt(H)\n",
    "ys = [ml_predictor(H) for H in Hs]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.title('$\\sigma$ = %.3f'%sig)\n",
    "plt.plot(Hs,mean_sup_fbm,label='$\\mathbb{E}\\, \\sup_{t \\in [0,1]}\\, X^H_t$',color='c')\n",
    "# plt.plot(Hs,.0001/np.sqrt(Hs),label='$1/\\sqrt{H}$')\n",
    "plt.plot(Hs,ys,label='$1/\\sqrt{H}$',color='k',linestyle='dotted')\n",
    "plt.legend()\n",
    "plt.xlabel('H')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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.8.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
