{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "5b26f937-d2bb-4f9f-b439-9494049f80a1",
   "metadata": {},
   "source": [
    "# Investigate development of MC error"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "eeadafbe-25a5-41c2-a11d-b5a8b4bfddca",
   "metadata": {},
   "outputs": [],
   "source": [
    "from hypnettorch.utils import misc\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "import os\n",
    "from scipy.spatial.distance import cdist\n",
    "import sys\n",
    "from time import time\n",
    "import torch\n",
    "\n",
    "# Pretend that notebook is located in base directory of this repo.\n",
    "curr_dir = os.path.basename(os.path.abspath(os.curdir))\n",
    "base_dir = os.path.abspath('../..')\n",
    "if curr_dir == 'nngp' and base_dir != sys.path[0]:\n",
    "    sys.path.insert(0, base_dir)\n",
    "\n",
    "from data.gmm_utils import get_circle_gmm_instance\n",
    "from nngp import MLPKernel\n",
    "\n",
    "%matplotlib inline\n",
    "%load_ext autoreload\n",
    "%autoreload 2\n",
    "\n",
    "device = 'cpu'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "5d5e405e-2706-40dd-af65-654199440174",
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_kernels(data, config_dict):\n",
    "    # Compute kernel matrices.\n",
    "    for k in config_dict:\n",
    "        start = time()\n",
    "        if k['name'] == 'rbf':\n",
    "            rbf_kernel = RBF()\n",
    "            K = rbf_kernel(data.detach().cpu().numpy())\n",
    "        else:\n",
    "            mlp_kernel = MLPKernel(sigma2_w=1., sigma2_b=1., **k['params'])\n",
    "            if k['name'].startswith('analytic'):\n",
    "                K = mlp_kernel.kernel_analytic(data, **k['kernel_params']).numpy() \n",
    "            elif k['name'].startswith('efficient'):\n",
    "                K = mlp_kernel.kernel_efficient(data, **k['kernel_params']).numpy()\n",
    "            else:\n",
    "                K = mlp_kernel.kernel_mc(data, **k['kernel_params']).numpy() \n",
    "        k['kernel'] = K\n",
    "        print('Kernel \"%s\" computation took %f seconds.' % (k['name'], time()-start))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "21fcfffa-0608-4223-9cfc-615ca24b9ed5",
   "metadata": {},
   "outputs": [],
   "source": [
    "def kernel_estimation_error(title, kernel_dicts, show_plots=True):\n",
    "    # Note, `mc_size == -1` means the efficient kernel was used.\n",
    "    K_dict = {}\n",
    "    for d in kernel_dicts:\n",
    "        if d['name'].startswith('analytic'):\n",
    "            assert 0 not in K_dict.keys()\n",
    "            K_dict[0] = d['kernel']\n",
    "            \n",
    "            nan_mask = np.isnan(d['kernel'])\n",
    "            if np.any(nan_mask):\n",
    "                print('%s has %d nan entries!' % (d['name'], nan_mask.sum()))\n",
    "        elif d['name'].startswith('efficient'):\n",
    "            assert -1 not in K_dict.keys()\n",
    "            K_dict[-1] = d['kernel']\n",
    "        else:\n",
    "            assert 'num_samples' in d['kernel_params']\n",
    "            num_mc = d['kernel_params']['num_samples']\n",
    "            assert num_mc not in K_dict.keys()\n",
    "            K_dict[num_mc] = d['kernel']\n",
    "\n",
    "    err_dict = {}\n",
    "    for k, v in K_dict.items():\n",
    "        if k == 0:\n",
    "            continue\n",
    "        # Compute relative error\n",
    "        err_dict[k] = np.abs(K_dict[k] - K_dict[0]) / K_dict[0]\n",
    "\n",
    "    mc_sizes = list(err_dict.keys())\n",
    "    mc_sizes.sort()\n",
    "    err_means = [err_dict[n][~nan_mask].mean() for n in mc_sizes]\n",
    "    err_stds = [err_dict[n][~nan_mask].std() for n in mc_sizes]\n",
    "    \n",
    "    if show_plots:\n",
    "        plt.title(title)\n",
    "        plt.errorbar(mc_sizes, err_means, yerr=err_stds, fmt='.k')\n",
    "        plt.xscale('log')\n",
    "        plt.show()\n",
    "\n",
    "        for n in mc_sizes:\n",
    "            plt.title(title + ' MC-%d' % n)\n",
    "            plt.hist(K_dict[n].flatten())\n",
    "            plt.show()\n",
    "    \n",
    "    return (mc_sizes, err_means, err_stds)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "45e92b5d-dfcf-4313-bb3f-33a601cbcac9",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "if False:\n",
    "    means = [np.array([-2, -2]), np.array([2, 2])]\n",
    "    covs = [1 * np.eye(len(mean)) for mean in means]\n",
    "    modes = get_gmm_tasks(means=means, covs=covs, num_train=10,\n",
    "                          num_test=100)\n",
    "    data = GMMData(modes, classification=True, use_one_hot=False)\n",
    "else:\n",
    "    data = get_circle_gmm_instance(sigmas=[.5]*2, num_train=20,\n",
    "                                   num_test=100, use_one_hot=False,\n",
    "                                   radius=4, offset=-np.pi/4,\n",
    "                                   rseed=1)\n",
    "\n",
    "X = data.get_train_inputs()\n",
    "Y = data.get_train_outputs().squeeze()\n",
    "\n",
    "colors = misc.get_colorbrewer2_colors(family='Dark2')\n",
    "\n",
    "plt.title('GMM')\n",
    "plt.scatter(X[Y==0,0], X[Y==0,1], c=colors[4])\n",
    "plt.scatter(X[Y==1,0], X[Y==1,1], c=colors[5])\n",
    "plt.ylim(-10, 10)\n",
    "plt.xlim(-10, 10)\n",
    "plt.show()\n",
    "\n",
    "# We use symmetrical values -1 / 1 for classification\n",
    "Y[Y==0] = -1\n",
    "\n",
    "X_torch = data.input_to_torch_tensor(X, device=device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "5ed917f4-5d0e-4c02-ba58-2403ba28b77e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Kernel \"analytic_relu_1l\" computation took 0.000959 seconds.\n",
      "Kernel \"relu_1l_mc100\" computation took 0.003509 seconds.\n",
      "Kernel \"relu_1l\" computation took 0.005993 seconds.\n",
      "Kernel \"relu_1l_mc10k\" computation took 0.030450 seconds.\n",
      "Kernel \"relu_1l_mc100k\" computation took 0.224828 seconds.\n",
      "Kernel \"analytic_relu_2l\" computation took 0.000590 seconds.\n",
      "Kernel \"relu_2l_mc100\" computation took 0.005468 seconds.\n",
      "Kernel \"relu_2l\" computation took 0.006491 seconds.\n",
      "Kernel \"relu_2l_mc10k\" computation took 0.046554 seconds.\n",
      "Kernel \"relu_2l_mc100k\" computation took 0.419679 seconds.\n",
      "Kernel \"analytic_relu_4l\" computation took 0.000900 seconds.\n",
      "Kernel \"relu_4l_mc100\" computation took 0.013290 seconds.\n",
      "Kernel \"relu_4l\" computation took 0.018671 seconds.\n",
      "Kernel \"relu_4l_mc10k\" computation took 0.092457 seconds.\n",
      "Kernel \"relu_4l_mc100k\" computation took 0.850583 seconds.\n",
      "Kernel \"analytic_relu_8l\" computation took 0.001196 seconds.\n",
      "Kernel \"relu_8l_mc100\" computation took 0.031066 seconds.\n",
      "Kernel \"relu_8l\" computation took 0.046576 seconds.\n",
      "Kernel \"relu_8l_mc10k\" computation took 0.197809 seconds.\n",
      "Kernel \"relu_8l_mc100k\" computation took 1.602156 seconds.\n",
      "Kernel \"analytic_erf_1l\" computation took 0.000570 seconds.\n",
      "Kernel \"erf_1l_mc100\" computation took 0.001240 seconds.\n",
      "Kernel \"erf_1l\" computation took 0.002681 seconds.\n",
      "Kernel \"erf_1l_mc10k\" computation took 0.022864 seconds.\n",
      "Kernel \"erf_1l_mc100k\" computation took 0.237521 seconds.\n",
      "Kernel \"analytic_erf_2l\" computation took 0.000375 seconds.\n",
      "Kernel \"erf_2l_mc100\" computation took 0.005336 seconds.\n",
      "Kernel \"erf_2l\" computation took 0.007095 seconds.\n",
      "Kernel \"erf_2l_mc10k\" computation took 0.049567 seconds.\n",
      "Kernel \"erf_2l_mc100k\" computation took 0.452584 seconds.\n",
      "Kernel \"analytic_erf_4l\" computation took 0.000758 seconds.\n",
      "Kernel \"erf_4l_mc100\" computation took 0.013013 seconds.\n",
      "Kernel \"erf_4l\" computation took 0.017242 seconds.\n",
      "Kernel \"erf_4l_mc10k\" computation took 0.102113 seconds.\n",
      "Kernel \"erf_4l_mc100k\" computation took 0.932604 seconds.\n",
      "Kernel \"analytic_erf_8l\" computation took 0.001060 seconds.\n",
      "Kernel \"erf_8l_mc100\" computation took 0.025725 seconds.\n",
      "Kernel \"erf_8l\" computation took 0.034508 seconds.\n",
      "Kernel \"erf_8l_mc10k\" computation took 0.166872 seconds.\n",
      "Kernel \"erf_8l_mc100k\" computation took 1.708460 seconds.\n",
      "Kernel \"analytic_cos_1l\" computation took 0.000640 seconds.\n",
      "Kernel \"cos_1l_mc100\" computation took 0.001194 seconds.\n",
      "Kernel \"cos_1l\" computation took 0.001941 seconds.\n",
      "Kernel \"cos_1l_mc10k\" computation took 0.021924 seconds.\n",
      "Kernel \"cos_1l_mc100k\" computation took 0.212739 seconds.\n"
     ]
    }
   ],
   "source": [
    "kernels = [\n",
    "    ### Relu ###\n",
    "    {'name': 'analytic_relu_1l', 'params': {'n_layer': 1, 'nonlinearity': torch.nn.ReLU()},\n",
    "     'kernel_params': {}},\n",
    "    {'name': 'relu_1l_mc100', 'params': {'n_layer': 1, 'nonlinearity': torch.nn.ReLU()},\n",
    "     'kernel_params': {'num_samples': 100}},\n",
    "    {'name': 'relu_1l', 'params': {'n_layer': 1, 'nonlinearity': torch.nn.ReLU()},\n",
    "     'kernel_params': {'num_samples': 1000}},\n",
    "    {'name': 'relu_1l_mc10k', 'params': {'n_layer': 1, 'nonlinearity': torch.nn.ReLU()},\n",
    "     'kernel_params': {'num_samples': 10000}},\n",
    "    {'name': 'relu_1l_mc100k', 'params': {'n_layer': 1, 'nonlinearity': torch.nn.ReLU()},\n",
    "     'kernel_params': {'num_samples': 100000}},\n",
    "    \n",
    "    {'name': 'analytic_relu_2l', 'params': {'n_layer': 2, 'nonlinearity': torch.nn.ReLU()},\n",
    "     'kernel_params': {}},\n",
    "    {'name': 'relu_2l_mc100', 'params': {'n_layer': 2, 'nonlinearity': torch.nn.ReLU()},\n",
    "     'kernel_params': {'num_samples': 100}},\n",
    "    {'name': 'relu_2l', 'params': {'n_layer': 2, 'nonlinearity': torch.nn.ReLU()},\n",
    "     'kernel_params': {'num_samples': 1000}},\n",
    "    {'name': 'relu_2l_mc10k', 'params': {'n_layer': 2, 'nonlinearity': torch.nn.ReLU()},\n",
    "     'kernel_params': {'num_samples': 10000}},\n",
    "    {'name': 'relu_2l_mc100k', 'params': {'n_layer': 2, 'nonlinearity': torch.nn.ReLU()},\n",
    "     'kernel_params': {'num_samples': 100000}},\n",
    "    \n",
    "    {'name': 'analytic_relu_4l', 'params': {'n_layer': 4, 'nonlinearity': torch.nn.ReLU()},\n",
    "     'kernel_params': {}},\n",
    "    {'name': 'relu_4l_mc100', 'params': {'n_layer': 4, 'nonlinearity': torch.nn.ReLU()},\n",
    "     'kernel_params': {'num_samples': 100}},\n",
    "    {'name': 'relu_4l', 'params': {'n_layer': 4, 'nonlinearity': torch.nn.ReLU()},\n",
    "     'kernel_params': {'num_samples': 1000}},\n",
    "    {'name': 'relu_4l_mc10k', 'params': {'n_layer': 4, 'nonlinearity': torch.nn.ReLU()},\n",
    "     'kernel_params': {'num_samples': 10000}},\n",
    "    {'name': 'relu_4l_mc100k', 'params': {'n_layer': 4, 'nonlinearity': torch.nn.ReLU()},\n",
    "     'kernel_params': {'num_samples': 100000}},\n",
    "    \n",
    "    {'name': 'analytic_relu_8l', 'params': {'n_layer': 8, 'nonlinearity': torch.nn.ReLU()},\n",
    "     'kernel_params': {}},\n",
    "    {'name': 'relu_8l_mc100', 'params': {'n_layer': 8, 'nonlinearity': torch.nn.ReLU()},\n",
    "     'kernel_params': {'num_samples': 100}},\n",
    "    {'name': 'relu_8l', 'params': {'n_layer': 8, 'nonlinearity': torch.nn.ReLU()},\n",
    "     'kernel_params': {'num_samples': 1000}},\n",
    "    {'name': 'relu_8l_mc10k', 'params': {'n_layer': 8, 'nonlinearity': torch.nn.ReLU()},\n",
    "     'kernel_params': {'num_samples': 10000}},\n",
    "    {'name': 'relu_8l_mc100k', 'params': {'n_layer': 8, 'nonlinearity': torch.nn.ReLU()},\n",
    "     'kernel_params': {'num_samples': 100000}},\n",
    "    \n",
    "    ### Error Function ###\n",
    "    {'name': 'analytic_erf_1l', 'params': {'n_layer': 1, 'nonlinearity': torch.erf},\n",
    "     'kernel_params': {}},\n",
    "    {'name': 'erf_1l_mc100', 'params': {'n_layer': 1, 'nonlinearity': torch.erf},\n",
    "     'kernel_params': {'num_samples': 100}},\n",
    "    {'name': 'erf_1l', 'params': {'n_layer': 1, 'nonlinearity': torch.erf},\n",
    "     'kernel_params': {'num_samples': 1000}},\n",
    "    {'name': 'erf_1l_mc10k', 'params': {'n_layer': 1, 'nonlinearity': torch.erf},\n",
    "     'kernel_params': {'num_samples': 10000}},\n",
    "    {'name': 'erf_1l_mc100k', 'params': {'n_layer': 1, 'nonlinearity': torch.erf},\n",
    "     'kernel_params': {'num_samples': 100000}},\n",
    "\n",
    "    {'name': 'analytic_erf_2l', 'params': {'n_layer': 2, 'nonlinearity': torch.erf},\n",
    "     'kernel_params': {}},\n",
    "    {'name': 'erf_2l_mc100', 'params': {'n_layer': 2, 'nonlinearity': torch.erf},\n",
    "     'kernel_params': {'num_samples': 100}},\n",
    "    {'name': 'erf_2l', 'params': {'n_layer': 2, 'nonlinearity': torch.erf},\n",
    "     'kernel_params': {'num_samples': 1000}},\n",
    "    {'name': 'erf_2l_mc10k', 'params': {'n_layer': 2, 'nonlinearity': torch.erf},\n",
    "     'kernel_params': {'num_samples': 10000}},\n",
    "    {'name': 'erf_2l_mc100k', 'params': {'n_layer': 2, 'nonlinearity': torch.erf},\n",
    "     'kernel_params': {'num_samples': 100000}},\n",
    "    \n",
    "    {'name': 'analytic_erf_4l', 'params': {'n_layer': 4, 'nonlinearity': torch.erf},\n",
    "     'kernel_params': {}},\n",
    "    {'name': 'erf_4l_mc100', 'params': {'n_layer': 4, 'nonlinearity': torch.erf},\n",
    "     'kernel_params': {'num_samples': 100}},\n",
    "    {'name': 'erf_4l', 'params': {'n_layer': 4, 'nonlinearity': torch.erf},\n",
    "     'kernel_params': {'num_samples': 1000}},\n",
    "    {'name': 'erf_4l_mc10k', 'params': {'n_layer': 4, 'nonlinearity': torch.erf},\n",
    "     'kernel_params': {'num_samples': 10000}},\n",
    "    {'name': 'erf_4l_mc100k', 'params': {'n_layer': 4, 'nonlinearity': torch.erf},\n",
    "     'kernel_params': {'num_samples': 100000}},\n",
    "    \n",
    "    {'name': 'analytic_erf_8l', 'params': {'n_layer': 8, 'nonlinearity': torch.erf},\n",
    "     'kernel_params': {}},\n",
    "    {'name': 'erf_8l_mc100', 'params': {'n_layer': 8, 'nonlinearity': torch.erf},\n",
    "     'kernel_params': {'num_samples': 100}},\n",
    "    {'name': 'erf_8l', 'params': {'n_layer': 8, 'nonlinearity': torch.erf},\n",
    "     'kernel_params': {'num_samples': 1000}},\n",
    "    {'name': 'erf_8l_mc10k', 'params': {'n_layer': 8, 'nonlinearity': torch.erf},\n",
    "     'kernel_params': {'num_samples': 10000}},\n",
    "    {'name': 'erf_8l_mc100k', 'params': {'n_layer': 8, 'nonlinearity': torch.erf},\n",
    "     'kernel_params': {'num_samples': 100000}},\n",
    "    \n",
    "    ### Cosine ###\n",
    "    {'name': 'analytic_cos_1l', 'params': {'n_layer': 1, 'nonlinearity': torch.cos},\n",
    "     'kernel_params': {}},\n",
    "    {'name': 'cos_1l_mc100', 'params': {'n_layer': 1, 'nonlinearity': torch.cos},\n",
    "     'kernel_params': {'num_samples': 100}},\n",
    "    {'name': 'cos_1l', 'params': {'n_layer': 1, 'nonlinearity': torch.cos},\n",
    "     'kernel_params': {'num_samples': 1000}},\n",
    "    {'name': 'cos_1l_mc10k', 'params': {'n_layer': 1, 'nonlinearity': torch.cos},\n",
    "     'kernel_params': {'num_samples': 10000}},\n",
    "    {'name': 'cos_1l_mc100k', 'params': {'n_layer': 1, 'nonlinearity': torch.cos},\n",
    "     'kernel_params': {'num_samples': 100000}},\n",
    "]\n",
    "\n",
    "compute_kernels(X_torch, kernels)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "a42f4393-5cc9-4467-aca9-3bec67e3fae3",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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/PLLP3vyySfnTZgwYdmf/vSnPocccsiweDxe7PLy6sADD1z74osvznn++efnXH/99YsXLVpUccIJJ+w4d+7csmLXhuZ58803Ox177LHDY7GY3XrrrZ888MADH+22224bJk6cOPjGG2/sU+z68ol+3H505Pfj1ohGozrzzDOHbLPNNtGnnnpq3osvvjinR48evGAt0JHfS1ujLfXBDnFTpPZmypQp8/v37x+r//2oo45aX1VVFT///PMHP/fcc12PPfbYdcWsr6USiYTq6uqsoqIi4+L/VVVVsUMPPbRGkg4//PCa4cOH1x5xxBEj7r///qrrr7/+88JVi9Z64IEHquLxuJ5//vn53bt3T0jS8ccfv/b999/v9Oc//7nXxRdfvLzYNbYE/bhj6cjvxy0RiUSstLTUFyxYUFZTU1NywgknrDryyCPX5/IxOpqO/F7aEm2xDzKy3gJ18Zg9Mf9fPX71r+e3eXL+v3rUxWMFXW6m4f8Y6h1wwAE1krR48eLSlpxz4sSJ/UeOHLlT165dd+vZs+eo/fbbb9j06dMr69sXLVoULi0t3eOaa66pbnzsf//3f/fv1KnT7suXL9/09dH999/fY9SoUSM6deq0e9euXXc78sgjh3z44YebjRpuu+22u4wdO3b7W265pdf222//tbKysj0ee+yx7s2p+4ADDtiQqm+zcz/33HNd9t9//2GVlZW7d+rUafcDDzxwx3/+858VzTl3e+exiK39vwd6rPjfK7dZ+38P9vBYpKD9uK6uzsLhsFdWViYabu/WrVs8kUhkOqxJ9OOOJxZL2Mx/ftpj2pS528x8+9MesViC9+MC9ONoNKpLLrmkb/2+1dXVu55xxhnbbdiwYdPrP3fu3DIz2/OXv/xln7PPPnu76urqXTt16rTH6aefPmD48OG7pJ7rYDPbc5999kk7Vast8Lq4bXz6/R7rbn11m41/fb+H18V5L6UP5hQj68307+WLO/3g+XuGRRKxkmg8bqWhkF/55jOJB789Ye5ufQbUFquuF198sask7bLLLi2qYcmSJaXnnnvuFwMHDoyuX7++5KGHHuo1evTo4a+++uoH++6778aBAwfGDj/88C/vu+++Ppdffvmy+uNisZgefvjh3mPGjFndp0+fuCTddNNNfS6++OKB48aNW3nppZcuXbt2bckNN9zQ/5BDDhn+3nvvvd+zZ89N7x6vv/5619mzZ3f+n//5n6V9+/aN7rDDDnXNqXvevHnlkjR06NBI/bZHHnmk+/e///0dDj744C8nT568QJImTZrU99BDDx3xr3/96/0ddtgh2pLXqD3Z+PFbnT6bNGaYRyMlHqszC5f5sod/ktjup1PmVgzZuyD9+IwzzljxwAMP9DnttNMGXnPNNUu7dOmSuP/++3u+/vrrXW+//fYFLTkn/bhjWfTJ6k5/vOutYbFYoiQeS1goXOJ//cv7iR+ete/cAQN78H6s/PXj448/fsj06dO7n3vuuZ8feOCB699///1Ov/zlL/svWrSofNq0aR813Pfmm2/ut+uuu9b89re//SQej9t+++1Xc9BBB62fMGHCkAsuuGDpscceuyao0w+2pm7W0k6rz3pimNfFSlQXN5WFfO0NLyWq7ho3t3SXfryXij6YC4T1ZqiLx+wHz98z7Mu6jZtet1gsYRsVLfn+8/cM/9fJP59VFgrn9OuabCxYsKD0l7/8Zf/9999/7UEHHdSiG0A9+uijn9T/PRaLady4cWt23HHHne+8887e++6772JJOuecc5Yfc8wxw6ZOndpl9OjR61PHdf/iiy9KzznnnOWStGbNmpJrrrlm23Hjxq18/PHHF9af86CDDqrZeeedd77tttt6X3HFFZv+Ya9bty709ttvzx44cOAWo1PpuLui0aji8bi9//775WefffagQYMGRc4777wV9ftcdNFFA/bee+9106dP3/SPdcyYMWuHDh26y/XXX9/3nnvuWdyS16i98FjEPps0ZliiZvWmfux1MfO6DSWfTjpy+NDffjbLwuV578d777137dSpU+eedNJJOzz44IN9JCkcDvuvfvWrRWeeeebqlpyTftxxxGIJ++Ndbw3buCG6qR8n6uIWrYuX/GHym8Mvv/rwWeFwCe/HeejHU6dO7fLcc8/1vO222xaed955KyXpuOOOW1dVVRU755xztn/ttdc6HXDAARvr9+/du3f0+eef/6ik5Ksv82trazdIyQ+o9VPC2hqvi9vqs54Y5mtqv8pSG2PmG2Mlq858Ynj1jHNmWVmI91L6YKsxDaYZnlkwq3skEUv7mkUSsZJnFsxq1lffubBmzZqSY445ZodwOOwPPPDAwoZt0Wh0s5+mPPXUU1333XffYT169NittLR0z7Kysj0/+eST8vnz52/6yv3oo49eN3To0No77rhj0wUrd999d59hw4ZtrO/oL730Upf169eHvv/9769s+NhDhgyp23777WtfffXVrg0fd9SoUTXZBhxJeuaZZ6rKysr27NSp0x577bXX1z788MNOTz/99If1n+Dffffd8sWLF5ePHz9+s8fv2rVrYvfdd6954403WrTiRnuy7s3Hu3s0krYfezRSsu7NxwvSj999993yk08+eeiOO+648eGHH57/1FNPzfve9763/MILLxx4xx13VNXvRz+mH6fzzn+WdI/FEmn7cSyWKHnnP0t4P85TP37uuee6l5aW+imnnLK64XnHjh27NvW4m513zJgxXzYMSe1F7dQ53b0ufSbwulhJ7dQ5vJfSB3OCkfVm+HjtivJoPP1ctGg8bgvWrizo8mcbNmywI444YofFixeXv/DCC3OHDh266V/f3Llzy0aMGLHZUkxz5sx5d/jw4Vt8pfTqq692PvHEE3f8xje+sfa2225buO2220bD4bCfeeaZgyORzUPdhAkTll199dUDPv/880Vr164N/eMf/+h+/fXXL6pv//zzz8OSdNxxxw1LV3P37t03+5qpurq6WV/lH3TQQWuuueaaJXV1dfb6669XXn/99duecMIJO8yaNWt2586dfenSpWEpOQdt4sSJgxsf369fv6ynJ4TDYa+rq0v7rzue6gelpaUFH7lrrbovPiz3WF3afuzxOqtbNr8g/fiiiy7aNhwO+/Tp0+eXlydH8seOHbtu1apV4UsvvXTAmWeeuWr+/Pn0Y/pxWiuW15THM8xPj8cStnJ5De/HeerHy5cvD0ejUevevfvu6dpXrly5Wbbo169fu5yyFVu4ulyZ5qfXJSz2yWreS+mDOUFYb4Yh3XpHSkMhT3cBU2ko5Nt36xVJd1w+RCIRGzNmzNBZs2ZVPvPMM/P22WefjQ3bBw0aFJ0xY8YHjbelO9ejjz7aMxwO+5QpUz6q/4cuSWvXrg1169Zts39IZ5999srrrrtuuzvuuKP36tWrQ+Xl5YkzzjhjZX17nz59YpJ06623Lhw1atRmNUlb/sM0a951OD169IjXf7V82GGH1XTv3j3+4x//ePANN9xQfc0113xRPzJ5ySWXfDZ69Oi1jY9v+Py2plevXtGFCxemHcFcsmRJaUlJifr27Zv1aGpQlG2zY8TCZe51W14YbaEyL6veoSD9eM6cOZ122mmnjY3/m+y99941zzzzTNVnn30Wph/TjzPp3acyEgqXeCJNWAqFS7xXn0rej/PUj6uqqmLl5eX+/PPPz0nXPnDgwM2eW3P/fbQV4cE9IyoLuTamWWSirMTDg3ryXkofzAnCejMcs/2ua65885nERkW3GKUqLwknjtl+1zWFqCMej+v444/f/vXXX+/22GOPfZhurlVFRYVnO19yw4YNJSUlJSop+Wp+51//+teuS5cuLdtuu+02+9RdVVWVGDt27Mr777+/z4YNG0rGjh27qqqqatPFId/61rfWV1ZWJubPn19+/vnnr1SenXfeeSsnT55c/fvf/77vRRddtHzUqFG1/fv3r5s9e3an1i6Bd8ghh6x79tlnq1555ZXODV/LRCKh5557rsfOO+9cU79MVlvSdd8T1yx7+CcJr9uwRT+20vJE131PLEg/7tOnT2z27Nmda2trN1ua66233qosLy/36urqOP2YfpzJqN36r/nrX95PROviW/TjcLgkMWq3/rwf56kfjxkzZu0dd9zRd/Xq1aGxY8e2yaUpc6Fi9Ig1a294KeEbt5wKY2XhRMXoEbyX0gdzgrDeDGWhsD/47Qlzv//8PcMbrgZTXhJOPPjtCXMLdXHpKaecMnDKlCk9zz///KVdunRJNFxOafDgwXUNv37NxpgxY9bcc8891ePGjdt+woQJK+bMmVMxadKkfpm+jvrxj3+8/M9//nMfSTr33HM3W7+1qqoqceWVVy6+5JJLBi1fvjw8ZsyYtT169IgvXry49JVXXul68MEHrzv77LNXteR5p1NSUqLLL798yX/913/t8Otf/7rP1Vdf/cXNN9+86Hvf+97Qo446yk488cRVffr0iS1durT0tdde6zJw4MC6q6666ov64yORSMm9997bs/F5hw8fXnvWWWetnDx5cvXYsWN3nDhx4tJRo0ZtXLZsWfiee+7pM2/evM5PPvnkvFw9j0KycLlv99Mpcz+ddORwj0ZKPF5nFipzKy1PbPfTKXMLcXGpJP3oRz9aNmHChCGHHXbYDmefffbyzp07J5566qkezz77bNXpp5/+RXPX1qUfd6x+HA6X+A/P2nfuHya/ObzhajDhcEnih2ftO7dQF5d2xPfjo48+et3RRx+96gc/+MHQs84664v99tuvpqSkRB9//HHZ1KlTu0+aNOnTXXfdtWDfbBSLlYW86q5xc1ed+cTw5GowCVNZiVtZOFF117i5hbi4VOqY76UdrQ8S1ptptz4Dav918s9nPbNgVvcFa1eWb9+tV+SY7XddU8hVYF5++eXuknTbbbf1u+222/o1bJs4ceLSm2++eUlzznfCCSesvfbaaxfffvvt20ybNq3nDjvssPGuu+5acP311/dPt/++++67cdCgQZEuXbrEDzzwwC0+qV944YUrBg4cGL355pu3Ofvss3vFYjGrrq6u23fffdfvvffeLVodoSknn3zymptuuqnm9ttv3+bCCy9cNn78+DW9evWae9111/W74IILBkcikZLevXtHd99995rvfve7m70prFmzJjRhwoQhjc/5gx/8YPmf/vSnRf/4xz/mXnzxxf0nT568zRdffFHaqVOnxKhRo2qeffbZufVXv7dFFUP2rh36289mrXvz8e51y+aXl1XvEOm674lrChXUJem0005bXVlZ+eGvf/3rfueff/6gurq6kgEDBkRuuOGGRT/72c+afRMP+nHH68cDBvaovfzqw2e9858l3Vcurynv1acyMmq3/msKuQpMR30/fuqppxZcf/311Q8++GDvW2+9tV9ZWVmif//+dd/85jfXbrvttm1uWlVLle7Sr7Z6xjmzaqfO6R77ZHV5eFDPSMXoEWsKFdSljvte2pH6oLm3ueuKmuWdd95ZOGrUqBVb3xPZmjVrVvluu+2286RJkz6ZOHEiry3aJPox2gP6MYqtI/TBd955p/eoUaMGF+vxGVlH1j766KPSDz74oOLqq6/u37t372jDi0iAtoJ+jPaAfoxiow8WTttddBIF9/vf/77PscceO2zFihXhe++99+MuXbq0769l0C7Rj9Ee0I9RbPTBwmEaDAAAAJBBsafBMLIOAAAABBRhHQAAAAiojhDWvb1P9QEAAEDuJRIJk1TUG8e1+7BuZgtramo6F7sOAAAAtC0bN26sMLNW3Um6tdp9WI9GozctXLgwvGLFih51dXVhRtkBAADQlEQiYTU1NZ0WLlxYFovFri5mLe1+NRhJmjlz5o7hcPgSMxvl7j3VAT6kAAAAoMUSZvZ5LBa7eo899phWzEI6RFgHAAAA2iJGmAEAAICAIqwDAAAAAUVYBwAAAAKKsA4AAAAEFGEdAAAACKj/D/Zl4fG5qyCaAAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 864x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from matplotlib.font_manager import FontProperties\n",
    "from matplotlib.ticker import FuncFormatter, MaxNLocator\n",
    "\n",
    "arch_names = ['relu_1l', 'relu_2l', 'relu_4l', 'relu_8l',\n",
    "              'erf_1l', 'erf_2l', 'erf_4l', 'erf_8l']\n",
    "\n",
    "arch_names = ['relu_2l', 'relu_8l',\n",
    "              'erf_2l', 'erf_8l']\n",
    "\n",
    "arch_labels = ['2-layer ReLU', '8-layer ReLU',\n",
    "               '2-layer erf', '8-layer erf']\n",
    "\n",
    "#arch_names = ['cos_1l']\n",
    "#arch_labels = ['1-layer Cosine']\n",
    "\n",
    "errs = {}\n",
    "errs_lbls = {}\n",
    "\n",
    "for i, name in enumerate(arch_names):\n",
    "    curr_kernels = [k for k in kernels if name in k['name']]\n",
    "    curr_errs = kernel_estimation_error(name, curr_kernels, show_plots=False)\n",
    "\n",
    "    errs[name] = curr_errs\n",
    "    errs_lbls[name] = arch_labels[i]\n",
    "\n",
    "colors = misc.get_colorbrewer2_colors(family='Dark2')\n",
    "    \n",
    "small_font = FontProperties()\n",
    "small_font.set_size(16)\n",
    "ts, lw, ms = 15, 8, 140 # text fontsize, line width, marker size\n",
    "figsize = (12, 6)\n",
    "\n",
    "fig, axes = plt.subplots(figsize=figsize)\n",
    "    \n",
    "#plt.title('MC estimation error')\n",
    "for i, aname in enumerate(errs.keys()):\n",
    "    sizes = [s + 2*(i-1.5)*s/10 for s in errs[aname][0]]\n",
    "    plt.scatter(sizes, errs[aname][1], c=colors[i], s=lw*6,\n",
    "                label=errs_lbls[aname])\n",
    "    plt.errorbar(sizes, errs[aname][1], yerr=errs[aname][2], c=colors[i],\n",
    "                 fmt='.')\n",
    "plt.xscale('log')\n",
    "\n",
    "plt.legend(prop=small_font, loc='upper center', \n",
    "           bbox_to_anchor=(0.47, -0.2), ncol=5)\n",
    "plt.ylabel('relative MC error', fontsize=ts)\n",
    "plt.xlabel('No. of MC samples', fontsize=ts)\n",
    "\n",
    "plt.axhline(y=0, color='k', linestyle=(0, (5, 10)))\n",
    "\n",
    "axes.grid(False)\n",
    "axes.set_facecolor('w')\n",
    "axes.axhline(y=axes.get_ylim()[0], color='k', lw=lw)\n",
    "axes.axvline(x=axes.get_xlim()[0], color='k', lw=lw)\n",
    "#plt.xticks([0, 1, 2, 3], fontsize=ts)\n",
    "plt.xticks(fontsize=ts)\n",
    "plt.yticks([0, .05, .1, .15], fontsize=ts)\n",
    "axes.tick_params(axis='both', length=lw, direction='out', width=lw/2.)\n",
    "\n",
    "filename = 'mc_error'\n",
    "if filename is not None:\n",
    "    fpath = filename\n",
    "    plt.savefig(fpath + '.pdf', bbox_inches='tight')\n",
    "    plt.savefig(fpath + '.png', bbox_inches='tight')\n",
    "\n",
    "plt.show()\n",
    "\n",
    "\n",
    "#########################\n",
    "plt.title('MC estimation error')\n",
    "for i, aname in enumerate(errs.keys()):\n",
    "    sizes = [s + i*s/10 for s in errs[aname][0]]\n",
    "    plt.scatter(sizes, errs[aname][1], label=aname)\n",
    "plt.xscale('log')\n",
    "plt.xlabel('num samples')\n",
    "plt.ylabel('error')\n",
    "plt.legend()\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "39d1f282-9624-4889-ac8f-564e79f18cc3",
   "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.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
