{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## BO with Warped Gaussian Processes\n",
    "\n",
    "In this tutorial, we illustrate how to use learned input warping functions for robust bayesian optimization when the outcome may be non-stationary functions. When the lenglescales are non-stationarity in the raw input space, learning a warping function  that maps raw inputs to a warped space where the lengthscales are stationary can be useful because then standard stationary kernels can be used for to effectively model the function.\n",
    "\n",
    "In general, we recommend for a relatively simple setup (like this one) to use [Ax](https://ax.dev), since this will simplify your setup (including the amount of code you need to write) considerably. See the [Using BoTorch with Ax](./custom_botorch_model_in_ax) tutorial. To use input warping with `MODULAR_BOTORCH`, we can pass the `warp_tf`, constructed as below, by adding `input_transform=warp_tf` argument to the `Surrogate(...)` call. \n",
    "\n",
    "We consider use a Kumaraswamy CDF as the class of input warping function and learn the concentration parameters ($a>0$ and $b>0$). Kumaraswamy CDFs are quite flexible and map inputs in [0, 1] to outputs in [0, 1]. This work follows the Beta CDF input warping proposed by Snoek et al., but replaces the Beta distribution Kumaraswamy distribution, which has a *differentiable* and closed-form CDF. \n",
    "   \n",
    "   $$K_\\text{cdf}(x) = 1 - (1-x^a)^b$$\n",
    "   \n",
    "This enables maximum likelihood (or maximum a posteriori) estimation of the CDF hyperparameters using gradient methods to maximize the likelihood (or posterior probability) jointly with the GP hyperparameters. (Snoek et al. use a fully Bayesian treatment of the CDF parameters). Each input dimension is transformed using a separate warping function.\n",
    "\n",
    "We use the Noisy Expected Improvement (qNEI) acquisition function to optimize a synthetic Hartmann6 test function. The standard problem is\n",
    "\n",
    "$$f(x) = -\\sum_{i=1}^4 \\alpha_i \\exp \\left( -\\sum_{j=1}^6 A_{ij} (x_j - P_{ij})^2  \\right)$$\n",
    "\n",
    "over $x \\in [0,1]^6$ (parameter values can be found in `botorch/test_functions/hartmann6.py`). For this demonstration,\n",
    "We first warp each input dimension through a different inverse Kumaraswamy CDF.\n",
    "\n",
    "Since botorch assumes a maximization problem, we will attempt to maximize $-f(x)$ to achieve $\\max_{x} -f(x) = 3.32237$.\n",
    "\n",
    "[1] [J. Snoek, K. Swersky, R. S. Zemel, R. P. Adams. Input Warping for Bayesian Optimization of Non-Stationary Functions. Proceedings of the 31st International Conference on Machine Learning, PMLR 32(2):1674-1682, 2014.](http://proceedings.mlr.press/v32/snoek14.pdf)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import torch\n",
    "\n",
    "\n",
    "device = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
    "dtype = torch.double\n",
    "SMOKE_TEST = os.environ.get(\"SMOKE_TEST\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Problem setup\n",
    "\n",
    "First, we define the sample parameters for the sigmoid functions that transform the respective inputs."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Text(0, 0.5, 'Transformed Value')"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 360x360 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from torch.distributions import Kumaraswamy\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "%matplotlib inline\n",
    "\n",
    "\n",
    "fontdict = {\"fontsize\": 15}\n",
    "torch.manual_seed(1234567890)\n",
    "c1 = torch.rand(6, dtype=dtype, device=device) * 3 + 0.1\n",
    "c0 = torch.rand(6, dtype=dtype, device=device) * 3 + 0.1\n",
    "x = torch.linspace(0, 1, 101, dtype=dtype, device=device)\n",
    "k = Kumaraswamy(concentration1=c1, concentration0=c0)\n",
    "k_icdfs = k.icdf(x.unsqueeze(1).expand(101, 6))\n",
    "fig, ax = plt.subplots(1, 1, figsize=(5, 5))\n",
    "\n",
    "for i in range(6):\n",
    "    ax.plot(x.cpu(), k_icdfs[:, i].cpu())\n",
    "ax.set_xlabel(\"Raw Value\", **fontdict)\n",
    "ax.set_ylabel(\"Transformed Value\", **fontdict)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "from botorch.test_functions import Hartmann\n",
    "\n",
    "neg_hartmann6 = Hartmann(negate=True)\n",
    "\n",
    "\n",
    "def obj(X):\n",
    "    X_warp = k.icdf(X)\n",
    "    return neg_hartmann6(X_warp)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Initial design\n",
    "\n",
    "The models are initialized with 14 points in $[0,1]^6$ drawn from a scrambled sobol sequence.\n",
    "\n",
    "We add observe the objectives with additive Gaussian noise with a standard deviation of 0.05."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "from botorch.models import FixedNoiseGP\n",
    "from gpytorch.mlls.sum_marginal_log_likelihood import ExactMarginalLogLikelihood\n",
    "from botorch.utils.sampling import draw_sobol_samples\n",
    "\n",
    "NOISE_SE = 0.05\n",
    "train_yvar = torch.tensor(NOISE_SE**2, device=device, dtype=dtype)\n",
    "\n",
    "bounds = torch.tensor([[0.0] * 6, [1.0] * 6], device=device, dtype=dtype)\n",
    "\n",
    "\n",
    "def generate_initial_data(n=14):\n",
    "    # generate training data\n",
    "    train_x = draw_sobol_samples(\n",
    "        bounds=bounds, n=n, q=1, seed=torch.randint(0, 10000, (1,)).item()\n",
    "    ).squeeze(1)\n",
    "    exact_obj = obj(train_x).unsqueeze(-1)  # add output dimension\n",
    "\n",
    "    best_observed_value = exact_obj.max().item()\n",
    "    train_obj = exact_obj + NOISE_SE * torch.randn_like(exact_obj)\n",
    "    return train_x, train_obj, best_observed_value"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Input warping and model initialization\n",
    "We initialize the `Warp` input transformation and pass it a `FixedNoiseGP` to model the noiseless objective. The `Warp` object is a `torch.nn.Module` that contains the concentration parameters and applies the warping function in the `Model`'s `forward` pass."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "from botorch.utils.transforms import standardize\n",
    "from botorch.models.transforms.input import Warp\n",
    "from gpytorch.priors.torch_priors import LogNormalPrior\n",
    "\n",
    "\n",
    "def initialize_model(train_x, train_obj, use_input_warping):\n",
    "    if use_input_warping:\n",
    "        # initialize input_warping transformation\n",
    "        warp_tf = Warp(\n",
    "            indices=list(range(train_x.shape[-1])),\n",
    "            # use a prior with median at 1.\n",
    "            # when a=1 and b=1, the Kumaraswamy CDF is the identity function\n",
    "            concentration1_prior=LogNormalPrior(0.0, 0.75**0.5),\n",
    "            concentration0_prior=LogNormalPrior(0.0, 0.75**0.5),\n",
    "        )\n",
    "    else:\n",
    "        warp_tf = None\n",
    "    # define the model for objective\n",
    "    model = FixedNoiseGP(\n",
    "        train_x,\n",
    "        standardize(train_obj),\n",
    "        train_yvar.expand_as(train_obj),\n",
    "        input_transform=warp_tf,\n",
    "    ).to(train_x)\n",
    "    mll = ExactMarginalLogLikelihood(model.likelihood, model)\n",
    "    return mll, model"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Define a helper function that performs the essential BO step\n",
    "The helper function below takes an acquisition function as an argument, optimizes it, and returns the batch $\\{x_1, x_2, \\ldots x_q\\}$ along with the observed function values. For this example, we'll use sequential $q=1$ optimization. A simple initialization heuristic is used to select the 20 restart initial locations from a set of 512 random points. "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "from botorch.optim import optimize_acqf\n",
    "\n",
    "\n",
    "num_restarts = 20 if not SMOKE_TEST else 2\n",
    "raw_samples = 512 if not SMOKE_TEST else 32\n",
    "\n",
    "\n",
    "def optimize_acqf_and_get_observation(acq_func):\n",
    "    \"\"\"Optimizes the acquisition function, and returns a new candidate and a noisy observation.\"\"\"\n",
    "    # optimize\n",
    "    candidates, _ = optimize_acqf(\n",
    "        acq_function=acq_func,\n",
    "        bounds=bounds,\n",
    "        q=1,\n",
    "        num_restarts=num_restarts,\n",
    "        raw_samples=raw_samples,  # used for intialization heuristic\n",
    "        options={\"batch_limit\": 5, \"maxiter\": 200},\n",
    "    )\n",
    "    # observe new values\n",
    "    new_x = candidates.detach()\n",
    "    exact_obj = obj(new_x).unsqueeze(-1)  # add output dimension\n",
    "    train_obj = exact_obj + NOISE_SE * torch.randn_like(exact_obj)\n",
    "    return new_x, train_obj\n",
    "\n",
    "\n",
    "def update_random_observations(best_random):\n",
    "    \"\"\"Simulates a quasi-random policy by taking a the current list of best values observed randomly,\n",
    "    drawing a new random point, observing its value, and updating the list.\n",
    "    \"\"\"\n",
    "    rand_x = draw_sobol_samples(bounds=bounds, n=1, q=1).squeeze(1)\n",
    "    next_random_best = obj(rand_x).max().item()\n",
    "    best_random.append(max(best_random[-1], next_random_best))\n",
    "    return best_random"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Trial  1 of 3 ..................................................\n",
      "Trial  2 of 3 ..................................................\n",
      "Trial  3 of 3 .................................................."
     ]
    }
   ],
   "source": [
    "from botorch import fit_gpytorch_mll\n",
    "from botorch.acquisition.monte_carlo import qNoisyExpectedImprovement\n",
    "from botorch.exceptions import BadInitialCandidatesWarning\n",
    "\n",
    "import time\n",
    "import warnings\n",
    "\n",
    "\n",
    "warnings.filterwarnings(\"ignore\", category=BadInitialCandidatesWarning)\n",
    "warnings.filterwarnings(\"ignore\", category=RuntimeWarning)\n",
    "\n",
    "\n",
    "N_TRIALS = 3 if not SMOKE_TEST else 2\n",
    "N_BATCH = 50 if not SMOKE_TEST else 5\n",
    "\n",
    "verbose = False\n",
    "\n",
    "best_observed_all_ei, best_observed_all_warp, best_random_all = [], [], []\n",
    "\n",
    "torch.manual_seed(0)\n",
    "\n",
    "\n",
    "# average over multiple trials\n",
    "for trial in range(1, N_TRIALS + 1):\n",
    "\n",
    "    print(f\"\\nTrial {trial:>2} of {N_TRIALS} \", end=\"\")\n",
    "    best_observed_ei, best_observed_warp, best_random = [], [], []\n",
    "\n",
    "    # call helper functions to generate initial training data and initialize model\n",
    "    train_x_ei, train_obj_ei, best_observed_value_ei = generate_initial_data(n=14)\n",
    "    mll_ei, model_ei = initialize_model(\n",
    "        train_x_ei, train_obj_ei, use_input_warping=False\n",
    "    )\n",
    "\n",
    "    train_x_warp, train_obj_warp, = (\n",
    "        train_x_ei,\n",
    "        train_obj_ei,\n",
    "    )\n",
    "    best_observed_value_warp = best_observed_value_ei\n",
    "    # use input warping\n",
    "    mll_warp, model_warp = initialize_model(\n",
    "        train_x_warp, train_obj_warp, use_input_warping=True\n",
    "    )\n",
    "\n",
    "    best_observed_ei.append(best_observed_value_ei)\n",
    "    best_observed_warp.append(best_observed_value_warp)\n",
    "    best_random.append(best_observed_value_ei)\n",
    "\n",
    "    # run N_BATCH rounds of BayesOpt after the initial random batch\n",
    "    for iteration in range(1, N_BATCH + 1):\n",
    "\n",
    "        t0 = time.monotonic()\n",
    "\n",
    "        # fit the models\n",
    "        fit_gpytorch_mll(mll_ei)\n",
    "        fit_gpytorch_mll(mll_warp)\n",
    "\n",
    "        ei = qNoisyExpectedImprovement(\n",
    "            model=model_ei,\n",
    "            X_baseline=train_x_ei,\n",
    "        )\n",
    "\n",
    "        ei_warp = qNoisyExpectedImprovement(\n",
    "            model=model_warp,\n",
    "            X_baseline=train_x_warp,\n",
    "        )\n",
    "\n",
    "        # optimize and get new observation\n",
    "        new_x_ei, new_obj_ei = optimize_acqf_and_get_observation(ei)\n",
    "        new_x_warp, new_obj_warp = optimize_acqf_and_get_observation(ei_warp)\n",
    "\n",
    "        # update training points\n",
    "        train_x_ei = torch.cat([train_x_ei, new_x_ei])\n",
    "        train_obj_ei = torch.cat([train_obj_ei, new_obj_ei])\n",
    "\n",
    "        train_x_warp = torch.cat([train_x_warp, new_x_warp])\n",
    "        train_obj_warp = torch.cat([train_obj_warp, new_obj_warp])\n",
    "\n",
    "        # update progress\n",
    "        best_random = update_random_observations(best_random)\n",
    "        best_value_ei = obj(train_x_ei).max().item()\n",
    "        best_value_warp = obj(train_x_warp).max().item()\n",
    "        best_observed_ei.append(best_value_ei)\n",
    "        best_observed_warp.append(best_value_warp)\n",
    "\n",
    "        mll_ei, model_ei = initialize_model(\n",
    "            train_x_ei, train_obj_ei, use_input_warping=False\n",
    "        )\n",
    "        mll_warp, model_warp = initialize_model(\n",
    "            train_x_warp, train_obj_warp, use_input_warping=True\n",
    "        )\n",
    "\n",
    "        t1 = time.monotonic()\n",
    "\n",
    "        if verbose:\n",
    "            print(\n",
    "                f\"\\nBatch {iteration:>2}: best_value (random, ei, ei_warp) = \"\n",
    "                f\"({max(best_random):>4.2f}, {best_value_ei:>4.2f}, {best_value_warp:>4.2f}), \"\n",
    "                f\"time = {t1-t0:>4.2f}.\",\n",
    "                end=\"\",\n",
    "            )\n",
    "        else:\n",
    "            print(\".\", end=\"\")\n",
    "\n",
    "    best_observed_all_ei.append(best_observed_ei)\n",
    "    best_observed_all_warp.append(best_observed_warp)\n",
    "    best_random_all.append(best_random)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Perform Bayesian Optimization\n",
    "The Bayesian optimization \"loop\" simply iterates the following steps:\n",
    "1. given a surrogate model, choose a candidate point\n",
    "2. observe $f(x)$ for each $x$ in the batch \n",
    "3. update the surrogate model. \n",
    "\n",
    "\n",
    "Just for illustration purposes, we run three trials each of which do `N_BATCH=50` rounds of optimization.\n",
    "\n",
    "*Note*: Running this may take a little while."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Plot the results\n",
    "The plot below shows the log regret at each step of the optimization for each of the algorithms. The confidence intervals represent the variance at that step in the optimization across the trial runs. In order to get a better estimate of the average performance early on, one would have to run a much larger number of trials `N_TRIALS` (we avoid this here to limit the runtime of this tutorial). "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x7fec08ec4eb0>"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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tTPvQELVGS8uLo6SkhJKSkoYNWoUM2bMaLV+057gLfUKV29xERFpjdroO8m6devYsGFD4+tly5aRn5/P1q1bG9vfn3zySSZPntxY569//Wvjz0mTJgFw3nnn8eyzzwLw9NNPc8EFF3TWWxARkS6oW57RR8KxY8e46667OHToEPHx8QwaNIiHH36Ym266iS984QvU1tZy7rnn8rWvfa1xnaqqKiZMmEB9fT1/+ctfAPjNb37D7bffzs9//nNyc3P585//HKm3JCIiXYASfScZO3Ys8+bNO2X5xRdfzEcffXTK8q1btwJw//33n7S8qKiId99995T6DzzwQEjilO4n3HOph7tcRFqnRB9Aw8h4Db77/ApAI+NJbAr3XOrhLtcXAZHWKdEH0DAynohEP/WaF2mdOuOJiIjEMCV6ERGRGKZELyIiEsPURh/Iutdh/Ruwe6X3On+k93PI5VB8ReTiEhEROU06ow+k+Ar47K9h+LXe47O/9h4dTPJmxr333tv4+he/+EXjbXEPPPAAffr0YcyYMY2PQ4cOMXv2bKZMmdKh/YqISPelRN+JkpKSeP7559m3b1/A8m9961ssW7as8dGjR4/ODVBERGKOEn0nio+P58477+Shhx6KdCgiItJNdM82+pXPw5Edbdfb+n7w28zsAyM/12a1r3/965x11lkBZ5176KGHeOqppwDo2bMns2bNCn7/IiIiAXTPRB9BmZmZTJ06ld/85jekpKScVPatb32Lb3/72xGKTEREYlH3TPRBnHmf5LzQjrR1zz33cM455/DlL385pNsVERFpTm30EZCdnc0NN9zAY489FulQREQkxinRR8i99957Su/7hx566KTb6xpmsBMREWmv7nnpPkKOHTvW+Dw/P58TJ040vn7ggQcCTjVbVFRESUlJJ0QnXVLD4E4NXr7b+6nBnUTET4k+EH14SldRfIX+JkWkVUr0gejDU2JFMF9a26qjL74iXZoSvUgsC+ZLa1t19MVXpEtTZzwREZEYpjN6EZFw62jzSKTLY+E9dIVjFCYRSfRmlg38FSgCtgI3OOcOBqjXA3gUGAk44Hbn3Pxwxze7dDZzyuaw/sB6AIZkDwFgct/JlPQrCffuRSTWdLR5JNLl0RBDtJeHahthEKlL998BZjrnBgMz/a8D+TXwhnNuKDAaWNMZwZX0K+H+SfdzWdFlXFZ0GfdPup/7J93f4SQfyWlqS0pKWLx4cYe309yPf/zjgMt//etfc8899zS+/upXv8oll1zS+Pq3v/0t3/zmNzu8/507d3L99dd3eDsiIrEqUon+GmCG//kM4NrmFcwsE7gQeAzAOVftnDvUSfGFRbimqX3ggQeYPn166AI9DS0l+vPOO4958+Y1vl62bBmHDx+mrq4OgHnz5nH++ecHtY/a2toWy3r37s3f//7304hYpJl1r3uXUPdv9B4v3+091r0efJ1gtiESIZFqo893zpUDOOfKzSwvQJ0zgL3An81sNLAEuNs5dzzQBs3sTuBOgP79+4cn6g5qOk3tj370o4jFkZ6ezt13380rr7xCSkoKL774Ivn5+dx2220kJyezatUqdu/ezS9/+UumTJnC9OnTWbx4Mb/73e8AmDJlCt/+9rd54403qKioYMyYMYwYMYKnn366cR9nn30269evp6KigurqalJTUxk0aBArVqxgzJgxzJs3j5/97Gc88sgjPPzww1RXVzNo0CCefPJJUlNTue2228jOzuajjz7inHPOISMjg02bNrFjxw5KS0u57777uOOOO9i6dStTpkxh5cqVTJ8+nZdeeokTJ06wadMmrrvuOn72s58B8Nhjj/HTn/6U3r17M3jwYJKSkhrfj3RzXfiSrEgwwpbozewdoCBA0X8FuYl44BzgLufch2b2a7xL/N8PVNk59zDwMMC4ceNcaxt+Y+sb7D6+u80AFu1aFGSokJ+Wz+VFl7dZLxqmqT1+/DgTJ07kRz/6Effddx+PPPII3/ve9wDYunUrc+bMYdOmTXz6059m48aNLW7nJz/5Cb/73e9YtmzZKWXx8fGMGTOGRYsWUVFRwYQJExg8eDDz5s0jLy8P5xz9+vXjc5/7HHfccQcA3/ve93jssce46y5vEqH169fzzjvvEBcXxwMPPMDy5ctZsGABx48f5+yzz+aqq646Zb/Lli3jo48+IikpieLiYu666y7i4uL44Q9/yNKlS8nIyOCiiy5i9OjRITiSIiLRL2yJ3jl3SUtlZrbbzAr9Z/OFwJ4A1cqAMufch/7Xf6fltvwuI1TT1K5YsYJbb70VgF27dpGYmMivfvUrAGbOnEmvXr1aXDcxMbGx3X/s2LG8/fbbjWU33HADPp+PwYMHc8YZZ7B27drTeXsnOf/885k3bx4VFRVMmjSJwYMH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\n",
      "text/plain": [
       "<Figure size 576x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import numpy as np\n",
    "from matplotlib import pyplot as plt\n",
    "\n",
    "%matplotlib inline\n",
    "\n",
    "\n",
    "def ci(y):\n",
    "    return 1.96 * y.std(axis=0) / np.sqrt(N_TRIALS)\n",
    "\n",
    "\n",
    "GLOBAL_MAXIMUM = neg_hartmann6.optimal_value\n",
    "\n",
    "\n",
    "iters = np.arange(N_BATCH + 1)\n",
    "y_ei = np.log10(GLOBAL_MAXIMUM - np.asarray(best_observed_all_ei))\n",
    "y_ei_warp = np.log10(GLOBAL_MAXIMUM - np.asarray(best_observed_all_warp))\n",
    "y_rnd = np.log10(GLOBAL_MAXIMUM - np.asarray(best_random_all))\n",
    "\n",
    "fig, ax = plt.subplots(1, 1, figsize=(8, 6))\n",
    "ax.errorbar(\n",
    "    iters,\n",
    "    y_rnd.mean(axis=0),\n",
    "    yerr=ci(y_rnd),\n",
    "    label=\"Sobol\",\n",
    "    linewidth=1.5,\n",
    "    capsize=3,\n",
    "    alpha=0.6,\n",
    ")\n",
    "ax.errorbar(\n",
    "    iters,\n",
    "    y_ei.mean(axis=0),\n",
    "    yerr=ci(y_ei),\n",
    "    label=\"NEI\",\n",
    "    linewidth=1.5,\n",
    "    capsize=3,\n",
    "    alpha=0.6,\n",
    ")\n",
    "ax.errorbar(\n",
    "    iters,\n",
    "    y_ei_warp.mean(axis=0),\n",
    "    yerr=ci(y_ei_warp),\n",
    "    label=\"NEI + Input Warping\",\n",
    "    linewidth=1.5,\n",
    "    capsize=3,\n",
    "    alpha=0.6,\n",
    ")\n",
    "ax.set(xlabel=\"number of observations (beyond initial points)\", ylabel=\"Log10 Regret\")\n",
    "ax.legend(loc=\"lower left\")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.7.11"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
