{
 "cells": [
  {
   "cell_type": "markdown",
   "source": [
    "# NSPDE validation.\n"
   ],
   "metadata": {
    "collapsed": false
   }
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "outputs": [],
   "source": [
    "import json\n",
    "import numpy as np\n",
    "import torch\n",
    "from src.gan.generators_spde import Generator\n",
    "from src.gan.base import stopping_criterion\n",
    "import pandas as pd\n",
    "import seaborn as sns\n",
    "import matplotlib.pyplot as plt\n",
    "from scipy.stats import ks_2samp\n",
    "from scipy import stats as st\n",
    "from statsmodels.tsa.stattools import acf\n",
    "from src.evaluation_functions import generate_ks_results_nspde\n",
    "import matplotlib as mpl\n",
    "import scienceplots\n",
    "\n",
    "plt.style.use('science')\n",
    "mpl.rcParams['axes.titlesize'] = 24\n",
    "mpl.rcParams['xtick.labelsize'] = 18\n",
    "mpl.rcParams['ytick.labelsize'] = 18\n",
    "mpl.rcParams['axes.labelsize'] = 22"
   ],
   "metadata": {
    "collapsed": false
   }
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "outputs": [],
   "source": [
    "is_cuda = torch.cuda.is_available()\n",
    "device = 'cuda' if is_cuda else 'cpu'\n",
    "if not is_cuda:\n",
    "    print(\"Warning: CUDA not available; falling back to CPU but this is likely to be very slow.\")"
   ],
   "metadata": {
    "collapsed": false
   }
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "outputs": [],
   "source": [
    "def my_smooth_corr(x, a, r = 2):\n",
    "    my_eps=0.001\n",
    "    j = 1.*torch.arange(1,x.shape[0]+1).to(x.device)\n",
    "    j[-1] = 0.\n",
    "    q = j**(-(2*r+1+my_eps)/2)\n",
    "    q[-1]=0\n",
    "    res = torch.sqrt(q)*torch.sqrt(2. / a) * torch.sin(j * torch.pi * x / a)\n",
    "    return res"
   ],
   "metadata": {
    "collapsed": false
   }
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "outputs": [],
   "source": [
    "data = np.load('../../data/lob_neurips.npy')\n",
    "data =  torch.tensor(data)\n",
    "\n",
    "dim_x = 20\n",
    "dim_t = 32\n",
    "output_dim = 1\n",
    "\n",
    "gridT = torch.linspace(0,1,dim_t).repeat(dim_x,1)\n",
    "gridX = torch.linspace(0,1,dim_x).unsqueeze(-1).repeat(1,dim_t)\n",
    "grid = torch.stack([gridX, gridT], dim=-1)"
   ],
   "metadata": {
    "collapsed": false
   }
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.9666666666666667\n",
      "{'discriminator__dyadic_order': 1, 'discriminator__kernel_type': 'rbf_id', 'discriminator__max_batch': 32, 'discriminator__sigma': {'sigma': 10}, 'discriminator__transform': 'integrate', 'generator__dim': 1, 'generator__hidden_size': 16, 'generator__initial_noise_size': 1, 'generator__integration_method': 'fixed_point', 'generator__modes1': 20, 'generator__modes2': 20, 'generator__n_iter': 4, 'generator__noise_size': 2, 'generator__noise_type': 'white', 'generator__output_dim': 1, 'seed': 540769236, 'training__batch_size': 64, 'training__lr': 0.001, 'training__steps': 1500, 'training__wd': 0}\n",
      "{'discriminator__dyadic_order': 1, 'discriminator__kernel_type': 'rbf_id', 'discriminator__max_batch': 32, 'discriminator__sigma': {'sigma': 10}, 'discriminator__transform': 'integrate', 'generator__dim': 1, 'generator__hidden_size': 16, 'generator__initial_noise_size': 1, 'generator__integration_method': 'fixed_point', 'generator__modes1': 20, 'generator__modes2': 20, 'generator__n_iter': 4, 'generator__noise_size': 2, 'generator__noise_type': 'white', 'generator__output_dim': 1, 'seed': 540769236, 'training__batch_size': 64, 'training__lr': 0.001, 'training__steps': 1500, 'training__wd': 0}\n",
      "{'discriminator__dyadic_order': 1, 'discriminator__kernel_type': 'rbf_sqr', 'discriminator__max_batch': 32, 'discriminator__sigma': {'sigma1': 10, 'sigma2': 10}, 'discriminator__transform': 'integrate', 'generator__dim': 1, 'generator__hidden_size': 16, 'generator__initial_noise_size': 1, 'generator__integration_method': 'fixed_point', 'generator__modes1': 10, 'generator__modes2': 20, 'generator__n_iter': 4, 'generator__noise_size': 2, 'generator__noise_type': 'white', 'generator__output_dim': 1, 'seed': 452780310, 'training__batch_size': 64, 'training__lr': 0.001, 'training__steps': 1500, 'training__wd': 0}\n",
      "{'discriminator__dyadic_order': 1, 'discriminator__kernel_type': 'rbf_sqr', 'discriminator__max_batch': 32, 'discriminator__sigma': {'sigma1': 10, 'sigma2': 10}, 'discriminator__transform': 'integrate', 'generator__dim': 1, 'generator__hidden_size': 16, 'generator__initial_noise_size': 1, 'generator__integration_method': 'fixed_point', 'generator__modes1': 10, 'generator__modes2': 20, 'generator__n_iter': 4, 'generator__noise_size': 2, 'generator__noise_type': 'white', 'generator__output_dim': 1, 'seed': 452780310, 'training__batch_size': 64, 'training__lr': 0.001, 'training__steps': 1500, 'training__wd': 0}\n",
      "{'discriminator__dyadic_order': 1, 'discriminator__kernel_type': 'rbf_cexp', 'discriminator__max_batch': 32, 'discriminator__sigma': {'n_freqs': 5, 'sigma1': 1, 'sigma2': 1}, 'discriminator__transform': 'integrate', 'generator__dim': 1, 'generator__hidden_size': 32, 'generator__initial_noise_size': 1, 'generator__integration_method': 'fixed_point', 'generator__modes1': 10, 'generator__modes2': 20, 'generator__n_iter': 4, 'generator__noise_size': 2, 'generator__noise_type': 'white', 'generator__output_dim': 1, 'seed': 937496119, 'training__batch_size': 64, 'training__lr': 0.001, 'training__steps': 1500, 'training__wd': 0}\n",
      "{'discriminator__dyadic_order': 1, 'discriminator__kernel_type': 'rbf_cexp', 'discriminator__max_batch': 32, 'discriminator__sigma': {'n_freqs': 5, 'sigma1': 1, 'sigma2': 1}, 'discriminator__transform': 'integrate', 'generator__dim': 1, 'generator__hidden_size': 32, 'generator__initial_noise_size': 1, 'generator__integration_method': 'fixed_point', 'generator__modes1': 10, 'generator__modes2': 20, 'generator__n_iter': 4, 'generator__noise_size': 2, 'generator__noise_type': 'white', 'generator__output_dim': 1, 'seed': 937496119, 'training__batch_size': 64, 'training__lr': 0.001, 'training__steps': 1500, 'training__wd': 0}\n"
     ]
    }
   ],
   "source": [
    "discriminators = ['rbf_id/', 'rbf_sqr/', 'rbf_cexp/']\n",
    "\n",
    "base_path = \"../../scripts/results/\"\n",
    "\n",
    "eval_batch_size = 128\n",
    "dataloader = torch.utils.data.DataLoader(data, batch_size=eval_batch_size, shuffle=True)\n",
    "infinite_train_dataloader = (elem for it in iter(lambda: dataloader, None) for elem in it)\n",
    "\n",
    "marginals = tuple([i*1./30 for i in range(30)])\n",
    "alpha     = 0.95\n",
    "tol       = 1 - alpha\n",
    "n_runs    = 100\n",
    "print(marginals[-1])\n",
    "\n",
    "\n",
    "\n",
    "results_ks = []\n",
    "for dis, discriminator in enumerate(discriminators):\n",
    "    best_run_ks = 0\n",
    "    best_run_id = 0\n",
    "\n",
    "    for i in range(1,233):\n",
    "        file = base_path + discriminator + str(i) + \"/metrics.json\"\n",
    "        res = 0\n",
    "        try:\n",
    "            with open(file, 'r') as f:\n",
    "                res = json.load(f)[\"best_ks\"][\"values\"][0]\n",
    "        except:\n",
    "            pass\n",
    "        if res > best_run_ks and res<0.72:\n",
    "            best_run_ks = res\n",
    "            best_run_id = i\n",
    "    \n",
    "    file = base_path + discriminator + str(best_run_id) + \"/config.json\"\n",
    "    with open(file, 'r') as f:\n",
    "        config = json.load(f)\n",
    "        print(config)\n",
    "        # create correlation function for the noise in input to the generator\n",
    "        if config['generator__noise_type'][0] == 'r':\n",
    "            input_roughness = int(config['generator__noise_type'].split('_')[-1])\n",
    "            generator_correlation = lambda x,a : my_smooth_corr(x, a, r = input_roughness)\n",
    "            config['generator__noise_type'] = generator_correlation\n",
    "        else:\n",
    "            pass\n",
    "        print(config)\n",
    "        \n",
    "    generator = Generator(\n",
    "    dim=config['generator__dim'],\n",
    "    data_size=config['generator__output_dim'], \n",
    "    initial_noise_size=config['generator__initial_noise_size'],\n",
    "    noise_size=config['generator__noise_size'], \n",
    "    hidden_size=config['generator__hidden_size'], \n",
    "    initial_point='given',\n",
    "    noise_type=config['generator__noise_type'],\n",
    "    integration_method=config['generator__integration_method'],\n",
    "    modes1=config['generator__modes1'], \n",
    "    modes2=config['generator__modes2'],).cuda()\n",
    "\n",
    "    generator.load_state_dict(torch.load(base_path+discriminator+str(best_run_id)+'/generator_checkpoint.pt'))\n",
    "    \n",
    "    dims = dim_x\n",
    "    path_length = dim_t\n",
    "    generators = [generator]\n",
    "\n",
    "    mean_ks = np.zeros((len(generators), dims, len(marginals)))\n",
    "\n",
    "\n",
    "    total_ks_results = generate_ks_results_nspde(\n",
    "        grid.to(device), infinite_train_dataloader, generators, marginals, n_runs, dims=dims, eval_batch_size=eval_batch_size, device=device\n",
    "    )\n",
    "\n",
    "    for k in range(dims):\n",
    "        for i, m in enumerate(marginals):\n",
    "            for j, disc in enumerate(['ours']):\n",
    "\n",
    "                average_score  = np.mean(total_ks_results[j, :, k, i, 0])\n",
    "                std_score      = np.std(total_ks_results[j, :, k, i, 0])\n",
    "                percent_reject = sum(total_ks_results[j, :, k, i, 1] <= tol)/n_runs\n",
    "\n",
    "                mean_ks[j,k,i] = average_score\n",
    "\n",
    "                lci, hci = st.norm.interval(alpha, loc=average_score, scale=std_score)\n",
    "\n",
    "    results_ks.append(mean_ks[0])\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 1152x360 with 4 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, (ax1,ax2,ax3, axcb) = plt.subplots(1,4,figsize=(16,5), gridspec_kw={'width_ratios':[1,1,1,0.08]})\n",
    "ax1.get_shared_y_axes().join(ax2,ax3)\n",
    "cmap = sns.color_palette(\"mako_r\", as_cmap=True) \n",
    "vmin = np.min([np.min(results_ks[0]),np.min(results_ks[1]),np.min(results_ks[2])])\n",
    "vmax = np.max([np.max(results_ks[0]),np.max(results_ks[1]),np.max(results_ks[2])])\n",
    "\n",
    "for i in range(3):\n",
    "    if i==0:\n",
    "        sns.heatmap(results_ks[i],ax = ax1,cbar=False,vmin=vmin, vmax=vmax, cmap=cmap)\n",
    "        ax1.set_xlabel('time')\n",
    "        ax1.set_ylabel('price')\n",
    "        ax1.set_title('ID')\n",
    "    elif i==1:\n",
    "        sns.heatmap(results_ks[i],ax=ax2,cbar=False,vmin=vmin, vmax=vmax, cmap=cmap)\n",
    "        ax2.set_xlabel('time')\n",
    "        ax2.set_ylabel('price')\n",
    "        ax2.set_title('SQR')\n",
    "    else:\n",
    "        sns.heatmap(results_ks[i],ax=ax3, cbar_ax=axcb, vmin=vmin, vmax=vmax, cmap=cmap)\n",
    "        ax3.set_xlabel('time')\n",
    "        ax3.set_ylabel('price')\n",
    "        ax3.set_title('CEXP')\n",
    "plt.tight_layout()\n",
    "plt.savefig('lob.png',dpi=1000)\n",
    "plt.show()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "NSPDE",
   "language": "python",
   "name": "nspde"
  },
  "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.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
