{
 "metadata": {
  "dataExplorerConfig": {},
  "kernelspec": {
   "display_name": "python3",
   "language": "python",
   "name": "python3",
   "cinder_runtime": true,
   "ipyflow_runtime": false,
   "metadata": {
    "kernel_name": "bento_kernel_ae",
    "nightly_builds": false,
    "fbpkg_supported": true,
    "cinder_runtime": true,
    "ipyflow_runtime": false,
    "is_prebuilt": true
   }
  },
  "last_server_session_id": "d1d86616-d7d7-4c21-b422-e2ae4e86bb3d",
  "last_kernel_id": "4cefec4e-9828-4758-942f-279b2a07fd74",
  "last_base_url": "https://bento.edge.x2p.facebook.net/",
  "last_msg_id": "8c31aad4-82bcf477a336e81950061e63_324",
  "outputWidgetContext": {}
 },
 "nbformat": 4,
 "nbformat_minor": 4,
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {
    "originalKey": "1bc3d568-b16a-4fe5-9667-c0e187f9a366",
    "showInput": false
   },
   "source": [
    "## BO with TuRBO-1 and TS/qEI\n",
    "\n",
    "In this tutorial, we show how to implement Trust Region Bayesian Optimization (TuRBO) [1] in a closed loop in BoTorch.\n",
    "\n",
    "This implementation uses one trust region (TuRBO-1) and supports either parallel expected improvement (qEI) or Thompson sampling (TS). We optimize the $20D$ Ackley function on the domain $[-5, 10]^{20}$ and show that TuRBO-1 outperforms qEI as well as Sobol.\n",
    "\n",
    "Since botorch assumes a maximization problem, we will attempt to maximize $-f(x)$ to achieve $\\max_x -f(x)=0$.\n",
    "\n",
    "[1]: [Eriksson, David, et al. Scalable global optimization via local Bayesian optimization. Advances in Neural Information Processing Systems. 2019](https://proceedings.neurips.cc/paper/2019/file/6c990b7aca7bc7058f5e98ea909e924b-Paper.pdf)\n",
    ""
   ]
  },
  {
   "cell_type": "code",
   "metadata": {
    "originalKey": "c11881c9-13f5-4e35-bdc8-b8f817089713",
    "collapsed": false,
    "requestMsgId": "b21eda64-89d8-461f-a9d1-57117892e0c9",
    "executionStartTime": 1674921563794,
    "executionStopTime": 1674921566438
   },
   "source": [
    "import os\n",
    "import math\n",
    "from dataclasses import dataclass\n",
    "\n",
    "import torch\n",
    "from botorch.acquisition import qExpectedImprovement\n",
    "from botorch.fit import fit_gpytorch_mll\n",
    "from botorch.generation import MaxPosteriorSampling\n",
    "from botorch.models import SingleTaskGP\n",
    "from botorch.optim import optimize_acqf\n",
    "from botorch.test_functions import Ackley\n",
    "from botorch.utils.transforms import unnormalize\n",
    "from torch.quasirandom import SobolEngine\n",
    "\n",
    "import gpytorch\n",
    "from gpytorch.constraints import Interval\n",
    "from gpytorch.kernels import MaternKernel, ScaleKernel\n",
    "from gpytorch.likelihoods import GaussianLikelihood\n",
    "from gpytorch.mlls import ExactMarginalLogLikelihood\n",
    "from gpytorch.priors import HorseshoePrior\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\")"
   ],
   "execution_count": 1,
   "outputs": [
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "I0128 075923.913 _utils_internal.py:179] NCCL_DEBUG env var is set to None\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "I0128 075923.914 _utils_internal.py:197] NCCL_DEBUG is forced to WARN from None\n"
     ]
    }
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "originalKey": "5be02873-2895-4451-8bf6-35e3cd0e6f99",
    "showInput": false
   },
   "source": [
    "## Optimize the 20-dimensional Ackley function\n",
    "\n",
    "The goal is to minimize the popular Ackley function:\n",
    "\n",
    "$f(x_1,\\ldots,x_d) = -20\\exp\\left(-0.2 \\sqrt{\\frac{1}{d} \\sum_{j=1}^d x_j^2} \\right) -\\exp \\left( \\frac{1}{d} \\sum_{j=1}^d \\cos(2 \\pi x_j) \\right) + 20 + e$\n",
    "\n",
    "over the domain  $[-5, 10]^{20}$.  The global optimal value of $0$ is attained at $x_1 = \\ldots = x_d = 0$.\n",
    "\n",
    "As mentioned above, since botorch assumes a maximization problem, we instead maximize $-f(x)$."
   ]
  },
  {
   "cell_type": "code",
   "metadata": {
    "originalKey": "069fba29-e308-4a40-b92e-b1a5bdc8dcd8",
    "collapsed": false,
    "requestMsgId": "40b2ab4c-067e-4e9f-9330-93dcda5f3e8c",
    "executionStartTime": 1674921566576,
    "executionStopTime": 1674921566582
   },
   "source": [
    "fun = Ackley(dim=20, negate=True).to(dtype=dtype, device=device)\n",
    "fun.bounds[0, :].fill_(-5)\n",
    "fun.bounds[1, :].fill_(10)\n",
    "dim = fun.dim\n",
    "lb, ub = fun.bounds\n",
    "\n",
    "batch_size = 4\n",
    "n_init = 2 * dim\n",
    "max_cholesky_size = float(\"inf\")  # Always use Cholesky\n",
    "\n",
    "\n",
    "def eval_objective(x):\n",
    "    \"\"\"This is a helper function we use to unnormalize and evalaute a point\"\"\"\n",
    "    return fun(unnormalize(x, fun.bounds))"
   ],
   "execution_count": 2,
   "outputs": []
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "originalKey": "6e19c4b3-1364-4789-833d-c7ae648e7a78",
    "showInput": false
   },
   "source": [
    "## Maintain the TuRBO state\n",
    "TuRBO needs to maintain a state, which includes the length of the trust region, success and failure counters, success and failure tolerance, etc. \n",
    "\n",
    "In this tutorial we store the state in a dataclass and update the state of TuRBO after each batch evaluation. \n",
    "\n",
    "**Note**: These settings assume that the domain has been scaled to $[0, 1]^d$ and that the same batch size is used for each iteration."
   ]
  },
  {
   "cell_type": "code",
   "metadata": {
    "originalKey": "4c419a40-d6cf-43de-8c60-e8445c3ca473",
    "collapsed": false,
    "requestMsgId": "5fb06df5-5815-47f9-bfa5-73155751345f",
    "executionStartTime": 1674921566718,
    "executionStopTime": 1674921566731
   },
   "source": [
    "@dataclass\n",
    "class TurboState:\n",
    "    dim: int\n",
    "    batch_size: int\n",
    "    length: float = 0.8\n",
    "    length_min: float = 0.5**7\n",
    "    length_max: float = 1.6\n",
    "    failure_counter: int = 0\n",
    "    failure_tolerance: int = float(\"nan\")  # Note: Post-initialized\n",
    "    success_counter: int = 0\n",
    "    success_tolerance: int = 10  # Note: The original paper uses 3\n",
    "    best_value: float = -float(\"inf\")\n",
    "    restart_triggered: bool = False\n",
    "\n",
    "    def __post_init__(self):\n",
    "        self.failure_tolerance = math.ceil(\n",
    "            max([4.0 / self.batch_size, float(self.dim) / self.batch_size])\n",
    "        )\n",
    "\n",
    "\n",
    "def update_state(state, Y_next):\n",
    "    if max(Y_next) > state.best_value + 1e-3 * math.fabs(state.best_value):\n",
    "        state.success_counter += 1\n",
    "        state.failure_counter = 0\n",
    "    else:\n",
    "        state.success_counter = 0\n",
    "        state.failure_counter += 1\n",
    "\n",
    "    if state.success_counter == state.success_tolerance:  # Expand trust region\n",
    "        state.length = min(2.0 * state.length, state.length_max)\n",
    "        state.success_counter = 0\n",
    "    elif state.failure_counter == state.failure_tolerance:  # Shrink trust region\n",
    "        state.length /= 2.0\n",
    "        state.failure_counter = 0\n",
    "\n",
    "    state.best_value = max(state.best_value, max(Y_next).item())\n",
    "    if state.length < state.length_min:\n",
    "        state.restart_triggered = True\n",
    "    return state"
   ],
   "execution_count": 3,
   "outputs": []
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "originalKey": "e03f6fa1-83d1-4f7e-8dfd-0a0a53a9ad1c",
    "showInput": false
   },
   "source": [
    "## Take a look at the state"
   ]
  },
  {
   "cell_type": "code",
   "metadata": {
    "originalKey": "e06a71f5-ab79-4c11-a798-2dd5f3cf40e1",
    "collapsed": false,
    "requestMsgId": "af20e76d-b6b3-4f59-82ae-e1d3a3159b8d",
    "executionStartTime": 1674921566859,
    "executionStopTime": 1674921566868
   },
   "source": [
    "state = TurboState(dim=dim, batch_size=batch_size)\n",
    "print(state)"
   ],
   "execution_count": 4,
   "outputs": [
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "TurboState(dim=20, batch_size=4, length=0.8, length_min=0.0078125, length_max=1.6, failure_counter=0, failure_tolerance=5, success_counter=0, success_tolerance=10, best_value=-inf, restart_triggered=False)\n"
     ]
    }
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "originalKey": "9fc2a1a5-1b3e-429a-933f-49739c0e9a6b",
    "showInput": false
   },
   "source": [
    "## Generate initial points\n",
    "This generates an initial set of Sobol points that we use to start of the BO loop."
   ]
  },
  {
   "cell_type": "code",
   "metadata": {
    "originalKey": "f0a7d80a-efba-4b9d-b5bc-64fdf62d0e99",
    "collapsed": false,
    "requestMsgId": "890e6347-f465-428c-a332-f6e3bbe34aa6",
    "executionStartTime": 1674921567266,
    "executionStopTime": 1674921567271
   },
   "source": [
    "def get_initial_points(dim, n_pts, seed=0):\n",
    "    sobol = SobolEngine(dimension=dim, scramble=True, seed=seed)\n",
    "    X_init = sobol.draw(n=n_pts).to(dtype=dtype, device=device)\n",
    "    return X_init"
   ],
   "execution_count": 5,
   "outputs": []
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "originalKey": "d7ed19a9-4662-496c-880b-c2e0717c4117",
    "showInput": false
   },
   "source": [
    "## Generate new batch\n",
    "Given the current `state` and a probabilistic (GP) `model` built from observations `X` and `Y`, we generate a new batch of points.  \n",
    "\n",
    "This method works on the domain $[0, 1]^d$, so make sure to not pass in observations from the true domain.  `unnormalize` is called before the true function is evaluated which will first map the points back to the original domain.\n",
    "\n",
    "We support either TS and qEI which can be specified via the `acqf` argument."
   ]
  },
  {
   "cell_type": "code",
   "metadata": {
    "originalKey": "f4a1f540-1959-4f95-92b1-696525a50347",
    "collapsed": false,
    "requestMsgId": "90e9fc43-786b-4027-b89e-f76dc8e472f0",
    "executionStartTime": 1674921567409,
    "executionStopTime": 1674921567429
   },
   "source": [
    "def generate_batch(\n",
    "    state,\n",
    "    model,  # GP model\n",
    "    X,  # Evaluated points on the domain [0, 1]^d\n",
    "    Y,  # Function values\n",
    "    batch_size,\n",
    "    n_candidates=None,  # Number of candidates for Thompson sampling\n",
    "    num_restarts=10,\n",
    "    raw_samples=512,\n",
    "    acqf=\"ts\",  # \"ei\" or \"ts\"\n",
    "):\n",
    "    assert acqf in (\"ts\", \"ei\")\n",
    "    assert X.min() >= 0.0 and X.max() <= 1.0 and torch.all(torch.isfinite(Y))\n",
    "    if n_candidates is None:\n",
    "        n_candidates = min(5000, max(2000, 200 * X.shape[-1]))\n",
    "\n",
    "    # Scale the TR to be proportional to the lengthscales\n",
    "    x_center = X[Y.argmax(), :].clone()\n",
    "    weights = model.covar_module.base_kernel.lengthscale.squeeze().detach()\n",
    "    weights = weights / weights.mean()\n",
    "    weights = weights / torch.prod(weights.pow(1.0 / len(weights)))\n",
    "    tr_lb = torch.clamp(x_center - weights * state.length / 2.0, 0.0, 1.0)\n",
    "    tr_ub = torch.clamp(x_center + weights * state.length / 2.0, 0.0, 1.0)\n",
    "\n",
    "    if acqf == \"ts\":\n",
    "        dim = X.shape[-1]\n",
    "        sobol = SobolEngine(dim, scramble=True)\n",
    "        pert = sobol.draw(n_candidates).to(dtype=dtype, device=device)\n",
    "        pert = tr_lb + (tr_ub - tr_lb) * pert\n",
    "\n",
    "        # Create a perturbation mask\n",
    "        prob_perturb = min(20.0 / dim, 1.0)\n",
    "        mask = torch.rand(n_candidates, dim, dtype=dtype, device=device) <= prob_perturb\n",
    "        ind = torch.where(mask.sum(dim=1) == 0)[0]\n",
    "        mask[ind, torch.randint(0, dim - 1, size=(len(ind),), device=device)] = 1\n",
    "\n",
    "        # Create candidate points from the perturbations and the mask\n",
    "        X_cand = x_center.expand(n_candidates, dim).clone()\n",
    "        X_cand[mask] = pert[mask]\n",
    "\n",
    "        # Sample on the candidate points\n",
    "        thompson_sampling = MaxPosteriorSampling(model=model, replacement=False)\n",
    "        with torch.no_grad():  # We don't need gradients when using TS\n",
    "            X_next = thompson_sampling(X_cand, num_samples=batch_size)\n",
    "\n",
    "    elif acqf == \"ei\":\n",
    "        ei = qExpectedImprovement(model, train_Y.max(), maximize=True)\n",
    "        X_next, acq_value = optimize_acqf(\n",
    "            ei,\n",
    "            bounds=torch.stack([tr_lb, tr_ub]),\n",
    "            q=batch_size,\n",
    "            num_restarts=num_restarts,\n",
    "            raw_samples=raw_samples,\n",
    "        )\n",
    "\n",
    "    return X_next"
   ],
   "execution_count": 6,
   "outputs": []
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "originalKey": "6b3dceba-35f5-4678-b21a-3b4ca22d3190",
    "showInput": false
   },
   "source": [
    "## Optimization loop\n",
    "This simple loop runs one instance of TuRBO-1 with Thompson sampling until convergence.\n",
    "\n",
    "TuRBO-1 is a local optimizer that can be used for a fixed evaluation budget in a multi-start fashion.  Once TuRBO converges, `state[\"restart_triggered\"]` will be set to true and the run should be aborted.  If you want to run more evaluations with TuRBO, you simply generate a new set of initial points and then keep generating batches until convergence or when the evaluation budget has been exceeded.  It's important to note that evaluations from previous instances are discarded when TuRBO restarts.\n",
    "\n",
    "NOTE: We use a `SingleTaskGP` with a noise constraint to keep the noise from getting too large as the problem is noise-free. "
   ]
  },
  {
   "cell_type": "code",
   "metadata": {
    "originalKey": "89258ea0-2a0c-4b88-8606-79ed531f0d97",
    "collapsed": false,
    "requestMsgId": "98ebf52b-fddf-485c-a250-d857b501eb19",
    "executionStartTime": 1674921567583,
    "executionStopTime": 1674921663734
   },
   "source": [
    "X_turbo = get_initial_points(dim, n_init)\n",
    "Y_turbo = torch.tensor(\n",
    "    [eval_objective(x) for x in X_turbo], dtype=dtype, device=device\n",
    ").unsqueeze(-1)\n",
    "\n",
    "state = TurboState(dim, batch_size=batch_size)\n",
    "\n",
    "NUM_RESTARTS = 10 if not SMOKE_TEST else 2\n",
    "RAW_SAMPLES = 512 if not SMOKE_TEST else 4\n",
    "N_CANDIDATES = min(5000, max(2000, 200 * dim)) if not SMOKE_TEST else 4\n",
    "\n",
    "torch.manual_seed(0)\n",
    "\n",
    "while not state.restart_triggered:  # Run until TuRBO converges\n",
    "    # Fit a GP model\n",
    "    train_Y = (Y_turbo - Y_turbo.mean()) / Y_turbo.std()\n",
    "    likelihood = GaussianLikelihood(noise_constraint=Interval(1e-8, 1e-3))\n",
    "    covar_module = ScaleKernel(  # Use the same lengthscale prior as in the TuRBO paper\n",
    "        MaternKernel(\n",
    "            nu=2.5, ard_num_dims=dim, lengthscale_constraint=Interval(0.005, 4.0)\n",
    "        )\n",
    "    )\n",
    "    model = SingleTaskGP(\n",
    "        X_turbo, train_Y, covar_module=covar_module, likelihood=likelihood\n",
    "    )\n",
    "    mll = ExactMarginalLogLikelihood(model.likelihood, model)\n",
    "\n",
    "    # Do the fitting and acquisition function optimization inside the Cholesky context\n",
    "    with gpytorch.settings.max_cholesky_size(max_cholesky_size):\n",
    "        # Fit the model\n",
    "        fit_gpytorch_mll(mll)\n",
    "\n",
    "        # Create a batch\n",
    "        X_next = generate_batch(\n",
    "            state=state,\n",
    "            model=model,\n",
    "            X=X_turbo,\n",
    "            Y=train_Y,\n",
    "            batch_size=batch_size,\n",
    "            n_candidates=N_CANDIDATES,\n",
    "            num_restarts=NUM_RESTARTS,\n",
    "            raw_samples=RAW_SAMPLES,\n",
    "            acqf=\"ts\",\n",
    "        )\n",
    "\n",
    "    Y_next = torch.tensor(\n",
    "        [eval_objective(x) for x in X_next], dtype=dtype, device=device\n",
    "    ).unsqueeze(-1)\n",
    "\n",
    "    # Update state\n",
    "    state = update_state(state=state, Y_next=Y_next)\n",
    "\n",
    "    # Append data\n",
    "    X_turbo = torch.cat((X_turbo, X_next), dim=0)\n",
    "    Y_turbo = torch.cat((Y_turbo, Y_next), dim=0)\n",
    "\n",
    "    # Print current status\n",
    "    print(\n",
    "        f\"{len(X_turbo)}) Best value: {state.best_value:.2e}, TR length: {state.length:.2e}\"\n",
    "    )"
   ],
   "execution_count": 7,
   "outputs": [
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "44) Best value: -1.17e+01, TR length: 8.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "48) Best value: -1.17e+01, TR length: 8.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "52) Best value: -1.11e+01, TR length: 8.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "56) Best value: -1.04e+01, TR length: 8.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "60) Best value: -1.04e+01, TR length: 8.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "64) Best value: -9.41e+00, TR length: 8.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "68) Best value: -9.41e+00, TR length: 8.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "72) Best value: -9.41e+00, TR length: 8.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "76) Best value: -9.41e+00, TR length: 8.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "80) Best value: -9.41e+00, TR length: 8.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "84) Best value: -8.82e+00, TR length: 8.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "88) Best value: -8.82e+00, TR length: 8.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "92) Best value: -8.82e+00, TR length: 8.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "96) Best value: -8.36e+00, TR length: 8.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "100) Best value: -7.93e+00, TR length: 8.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "104) Best value: -7.93e+00, TR length: 8.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "108) Best value: -7.93e+00, TR length: 8.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "112) Best value: -7.93e+00, TR length: 8.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "116) Best value: -7.93e+00, TR length: 8.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "120) Best value: -7.93e+00, TR length: 4.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "124) Best value: -6.72e+00, TR length: 4.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "128) Best value: -6.34e+00, TR length: 4.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "132) Best value: -6.01e+00, TR length: 4.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "136) Best value: -5.51e+00, TR length: 4.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "140) Best value: -5.51e+00, TR length: 4.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "144) Best value: -5.51e+00, TR length: 4.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "148) Best value: -5.45e+00, TR length: 4.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "152) Best value: -5.27e+00, TR length: 4.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "156) Best value: -5.27e+00, TR length: 4.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "160) Best value: -5.27e+00, TR length: 4.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "164) Best value: -5.27e+00, TR length: 4.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "168) Best value: -5.06e+00, TR length: 4.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "172) Best value: -5.06e+00, TR length: 4.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "176) Best value: -5.06e+00, TR length: 4.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "180) Best value: -5.06e+00, TR length: 4.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "184) Best value: -5.06e+00, TR length: 4.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "188) Best value: -5.06e+00, TR length: 2.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "192) Best value: -4.32e+00, TR length: 2.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "196) Best value: -3.86e+00, TR length: 2.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "200) Best value: -3.69e+00, TR length: 2.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "204) Best value: -3.69e+00, TR length: 2.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "208) Best value: -3.69e+00, TR length: 2.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "212) Best value: -3.69e+00, TR length: 2.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "216) Best value: -3.41e+00, TR length: 2.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "220) Best value: -3.41e+00, TR length: 2.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "224) Best value: -3.41e+00, TR length: 2.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "228) Best value: -3.41e+00, TR length: 2.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "232) Best value: -3.41e+00, TR length: 2.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "236) Best value: -3.41e+00, TR length: 1.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "240) Best value: -2.95e+00, TR length: 1.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "244) Best value: -2.65e+00, TR length: 1.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "248) Best value: -2.65e+00, TR length: 1.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "252) Best value: -2.65e+00, TR length: 1.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "256) Best value: -2.54e+00, TR length: 1.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "260) Best value: -2.54e+00, TR length: 1.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "264) Best value: -2.54e+00, TR length: 1.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "268) Best value: -2.38e+00, TR length: 1.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "272) Best value: -2.38e+00, TR length: 1.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "276) Best value: -2.38e+00, TR length: 1.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "280) Best value: -2.38e+00, TR length: 1.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "284) Best value: -2.38e+00, TR length: 1.00e-01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "288) Best value: -2.38e+00, TR length: 5.00e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "292) Best value: -1.91e+00, TR length: 5.00e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "296) Best value: -1.91e+00, TR length: 5.00e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "300) Best value: -1.91e+00, TR length: 5.00e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "304) Best value: -1.31e+00, TR length: 5.00e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "308) Best value: -1.31e+00, TR length: 5.00e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "312) Best value: -1.31e+00, TR length: 5.00e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "316) Best value: -1.31e+00, TR length: 5.00e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "320) Best value: -1.31e+00, TR length: 5.00e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "324) Best value: -1.31e+00, TR length: 2.50e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "328) Best value: -1.31e+00, TR length: 2.50e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "332) Best value: -1.31e+00, TR length: 2.50e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "336) Best value: -1.31e+00, TR length: 2.50e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "340) Best value: -1.31e+00, TR length: 2.50e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "344) Best value: -1.31e+00, TR length: 1.25e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "348) Best value: -1.14e+00, TR length: 1.25e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "352) Best value: -1.14e+00, TR length: 1.25e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "356) Best value: -1.14e+00, TR length: 1.25e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "360) Best value: -1.14e+00, TR length: 1.25e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "364) Best value: -1.12e+00, TR length: 1.25e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "368) Best value: -1.12e+00, TR length: 1.25e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "372) Best value: -1.12e+00, TR length: 1.25e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "376) Best value: -1.12e+00, TR length: 1.25e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "380) Best value: -1.12e+00, TR length: 1.25e-02\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "384) Best value: -1.12e+00, TR length: 6.25e-03\n"
     ]
    }
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "originalKey": "518bbb5e-84f6-4062-bf28-25ccf7650c01",
    "showInput": false
   },
   "source": [
    "## GP-EI\n",
    "As a baseline, we compare TuRBO to qEI"
   ]
  },
  {
   "cell_type": "code",
   "metadata": {
    "originalKey": "8cc7262f-36ac-427f-b7a1-d94b0ceeae5e",
    "collapsed": false,
    "requestMsgId": "20905f90-c4bf-4073-9f15-9ac77e1f9c22",
    "executionStartTime": 1674921663896,
    "executionStopTime": 1674921754833
   },
   "source": [
    "torch.manual_seed(0)\n",
    "\n",
    "X_ei = get_initial_points(dim, n_init)\n",
    "Y_ei = torch.tensor(\n",
    "    [eval_objective(x) for x in X_ei], dtype=dtype, device=device\n",
    ").unsqueeze(-1)\n",
    "\n",
    "while len(Y_ei) < len(Y_turbo):\n",
    "    train_Y = (Y_ei - Y_ei.mean()) / Y_ei.std()\n",
    "    likelihood = GaussianLikelihood(noise_constraint=Interval(1e-8, 1e-3))\n",
    "    model = SingleTaskGP(X_ei, train_Y, likelihood=likelihood)\n",
    "    mll = ExactMarginalLogLikelihood(model.likelihood, model)\n",
    "    fit_gpytorch_mll(mll)\n",
    "\n",
    "    # Create a batch\n",
    "    ei = qExpectedImprovement(model, train_Y.max(), maximize=True)\n",
    "    candidate, acq_value = optimize_acqf(\n",
    "        ei,\n",
    "        bounds=torch.stack(\n",
    "            [\n",
    "                torch.zeros(dim, dtype=dtype, device=device),\n",
    "                torch.ones(dim, dtype=dtype, device=device),\n",
    "            ]\n",
    "        ),\n",
    "        q=batch_size,\n",
    "        num_restarts=NUM_RESTARTS,\n",
    "        raw_samples=RAW_SAMPLES,\n",
    "    )\n",
    "    Y_next = torch.tensor(\n",
    "        [eval_objective(x) for x in candidate], dtype=dtype, device=device\n",
    "    ).unsqueeze(-1)\n",
    "\n",
    "    # Append data\n",
    "    X_ei = torch.cat((X_ei, candidate), axis=0)\n",
    "    Y_ei = torch.cat((Y_ei, Y_next), axis=0)\n",
    "\n",
    "    # Print current status\n",
    "    print(f\"{len(X_ei)}) Best value: {Y_ei.max().item():.2e}\")"
   ],
   "execution_count": 8,
   "outputs": [
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "44) Best value: -1.15e+01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "48) Best value: -1.05e+01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "52) Best value: -1.02e+01\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "56) Best value: -9.78e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "60) Best value: -9.30e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "64) Best value: -9.30e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "68) Best value: -8.58e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "72) Best value: -8.58e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "76) Best value: -8.58e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "80) Best value: -8.58e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "84) Best value: -8.58e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "88) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "92) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "96) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "100) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "104) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "108) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "112) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "116) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "120) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "124) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "128) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "132) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "136) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "140) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "144) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "148) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "152) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "156) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "160) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "164) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "168) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "172) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "176) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "180) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "184) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "188) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "192) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "196) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "200) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "204) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "208) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "212) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "216) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "220) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "224) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "228) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "232) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "236) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "240) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "244) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "248) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "252) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "256) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "260) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "264) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "268) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "272) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "276) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "280) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "284) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "288) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "292) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "296) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "300) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "304) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "308) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "312) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "316) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "320) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "324) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "328) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "332) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "336) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "340) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "344) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "348) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "352) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "356) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "360) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "364) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "368) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "372) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "376) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "380) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stdout",
     "text": [
      "384) Best value: -8.19e+00\n"
     ]
    },
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "/mnt/xarfuse/uid-25696/50ffb8fb-seed-nspid4026533582_cgpid14602017-ns-4026533579/botorch/optim/initializers.py:208: BadInitialCandidatesWarning:\n\nUnable to find non-zero acquisition function values - initial conditions are being selected randomly.\n\n"
     ]
    }
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "originalKey": "38f8ac21-d9ae-41f7-ba42-0ff6abde0a2c",
    "showInput": false
   },
   "source": [
    "## Sobol"
   ]
  },
  {
   "cell_type": "code",
   "metadata": {
    "originalKey": "a6333e87-1fcf-4174-9cb8-111598dd7780",
    "collapsed": false,
    "requestMsgId": "629a258e-fa1d-44f8-848c-3335a98e3421",
    "executionStartTime": 1674921754972,
    "executionStopTime": 1674921755010
   },
   "source": [
    "X_Sobol = (\n",
    "    SobolEngine(dim, scramble=True, seed=0)\n",
    "    .draw(len(X_turbo))\n",
    "    .to(dtype=dtype, device=device)\n",
    ")\n",
    "Y_Sobol = torch.tensor(\n",
    "    [eval_objective(x) for x in X_Sobol], dtype=dtype, device=device\n",
    ").unsqueeze(-1)"
   ],
   "execution_count": 9,
   "outputs": []
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "originalKey": "e20c8975-af02-4308-a1ef-3f12afb85ffd",
    "showInput": false
   },
   "source": [
    "## Compare the methods"
   ]
  },
  {
   "cell_type": "code",
   "metadata": {
    "originalKey": "b57b38e5-da03-4511-a301-7252eb6c7013",
    "collapsed": false,
    "requestMsgId": "7c4c1c41-4852-4498-a42e-14e08dc88afb",
    "executionStartTime": 1674921755158,
    "executionStopTime": 1674921757156
   },
   "source": [
    "import matplotlib\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "from matplotlib import rc\n",
    "\n",
    "%matplotlib inline\n",
    "\n",
    "\n",
    "names = [\"TuRBO-1\", \"EI\", \"Sobol\"]\n",
    "runs = [Y_turbo, Y_ei, Y_Sobol]\n",
    "fig, ax = plt.subplots(figsize=(8, 6))\n",
    "\n",
    "for name, run in zip(names, runs):\n",
    "    fx = np.maximum.accumulate(run.cpu())\n",
    "    plt.plot(fx, marker=\"\", lw=3)\n",
    "\n",
    "plt.plot([0, len(Y_turbo)], [fun.optimal_value, fun.optimal_value], \"k--\", lw=3)\n",
    "plt.xlabel(\"Function value\", fontsize=18)\n",
    "plt.xlabel(\"Number of evaluations\", fontsize=18)\n",
    "plt.title(\"20D Ackley\", fontsize=24)\n",
    "plt.xlim([0, len(Y_turbo)])\n",
    "plt.ylim([-15, 1])\n",
    "\n",
    "plt.grid(True)\n",
    "plt.tight_layout()\n",
    "plt.legend(\n",
    "    names + [\"Global optimal value\"],\n",
    "    loc=\"lower center\",\n",
    "    bbox_to_anchor=(0, -0.08, 1, 1),\n",
    "    bbox_transform=plt.gcf().transFigure,\n",
    "    ncol=4,\n",
    "    fontsize=16,\n",
    ")\n",
    "plt.show()"
   ],
   "execution_count": 10,
   "outputs": [
    {
     "output_type": "stream",
     "name": "stderr",
     "text": [
      "I0128 080236.236 font_manager.py:1349] generated new fontManager\n"
     ]
    },
    {
     "output_type": "display_data",
     "data": {
      "text/plain": "<Figure size 576x432 with 1 Axes>",
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAjsAAAHOCAYAAAB+YchsAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4xLjMsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy+AADFEAAAgAElEQVR4nOzdd5wcZeHH8e/e7fW7lEvvCb1XEQQUUAZhpDgKCoioIMWCgqgoNkRBBXv/oYgKioo6lDCCgzSVpihdElrq5ZJckksuuVzZ8vsjz92Wu93bu9tsmf28X6+8tkx79rnJ3vee55lnQvF4XAAAAEFVVewCAAAA7EyEHQAAEGiEHQAAEGiEHQAAEGiEHQAAEGiEHQAAEGjhYhcAQP5YtrO3pE9JerOk2ZJ6JS2RdJuk7/me2zPMNrWSPibpLEm7SwpJelnSrZK+63tuX9r6x0p6YJjD90vaJGmZpH9KusX33P/k6XO5kt5uXr7Z99zhjj/Wfd8i6T2SvuB77lfzvT6A4qNlBwgIy3beIem/kj4gaaOkX0u6T9Kekr4u6e+W7TSnbdMg6UFJ10uaIelP5t8sSd+QdL9lO/UZDhmV9L2kfz+X9LDZz2WSnrRs5w7LdqaO83PNlHRy0lsfHM/+AFQeWnaAALBsZ64JN3WSLvM997tJy+ZJelTS6yRdIekLSZt+RdIbJN0j6R2+524327RI+qukoyR92WyXLuJ77qUZyvMmE4BOlfSEZTuH+567fowf7zzzXfV7Se+U9A7Ldib7nrtpjPsDUGFo2QGC4X2SmiQ9lhx0JMn33JWm5UYmLEiJQPNh00Jz4UDQMdt0SfqIefkRy3aaRlMY33MflnSk6c5aJOk3Y/lQlu2EJJ1vXv7UtFTVSzpnLPsDUJlo2QGC4S+S1kp6NcPypeZxTtJ7x0tqkPRPE4hS+J77H8t2XjNh5S2S7hxNgXzP3W7ZznskvSTJsmznGN9zHxrVp9pRxl0krTJdZHMlnWi6sn6QaSPLdqokvd+0Cu1v/rB7xbQO/cD33K25HNyynQsk3SBpvaQ3+p67ZIT1d5P0aVPu2ZJ6JL0g6RZJN/ieGzHrvU3SYtPdONv33N5h9tVojtso6Qjfcx/PrcoApKNlBwgA33P/43vuz33PvT/DKrPN47qk9w4xj9kGET9pHg8dY7mWJ4Wkd41hFxeYx1/5nhuT9GdJWyQdYNnOYcNtYFqDbpV0o6Q9JN1hBmhPkHStGbs0YaQDW7ZjS/qxpC5JJ+UQdI6X9JQpc4ekmyQ9JGkfST+SdLdlOzVm9XskrZbUarr6hnOKCTr/I+gA40PYAQLO/PIfGNR7R9KiXc3jkFadJCvT1h2LB83jkaPZyLKdaZJOkxQ3wUG+53abIKMsA5UvMMHqeUl7+J57ru+550na21xFdpAJPdmOfaikP5guvtN8z31yhPUnm3I1Sbrc99zX+577Id9zT5O0r6RnJJ0g6XLzOaKSfmk2/0CG3b7bPP4yw3IAOSLsAMF3haQjTJfJdUnvTzSPW7Js22UeJ43j+CvM48xRbvd+SbWSHvI995Wk9280j2dlGEv0MfP4Fd9zOwfeNF1FX5L0XLayWLazSNLdZmzQWTle5v5+SVMl/VvSd5IX+J67WtLAQO6PmPApSb8wQe4Ey3ZmJ29jWp5OMmHr5hyODyALwg4QYJbtXG5aMSKSzvY9N7kba+CS8r4Mm8vM05O87lgMzO3TMMrtBlpubkx+0/fcf5mWkpb0rjFzmfu+5uXD6Tv0Pffvvufu73vu6cMd0LKdVjP+aYaki3zPdXMs61vM4199z40Ps/xhSVvNmKNdTVleNa1e1ZLOTVv/NFPn9/qeuybHMgDIgAHKQACZ1oNvSvqECTNn+557b9pqAyGkNsuu6szj9izrjGS6eVyb6wZm4sI9JG028/6ku9Fc2v7BgS4uYxfzGB1DSKg344v2NIOJb8xhm/TjHmnZznczrNNvHnc3kzYOfI7jTFfW15PWHejCukkAxo2wAwSMmQTw15LOkNRp5s8ZritmoItn4jDLlLZsPHPaDIz3WTWKbQYGJscl/cWynfTlA5MjHmnZzt6+5/7PvG40j9laqzK5LGn70y3b+YrvubmWeWC7Y82/bJLr+0+SfihpD8t2jvQ99xEz/ucE0+04qivgAAyPsAMEiLn1w2LTrfKaJNv33BczrP6SeVyQZZcDy5ZmWWckp5lHP5eVTVfSO8zLSZKOGWGTDw4M/JW0zTzWWbZTZa7gylWjpGtMODtT0m8s23mzGUw8koFL2c/3PfcXuR7Q99wey3Z+a+Y7OkfSI+az10j6bfqtOgCMDWEHCAgzt8wtJug8JelE33OzdR39yzy+Lss6rzePY7r02XRHHWxaaG7NYROZ8Sv1kp7yPffgLPs+SZIn6VzLdj5rgsFrZnGVpIVZ5h0azvd9z/28ZTsTzYDuN5nZpq/KYduXzVihbMExk5+bsHO6ZTuXmHuUiS4sIH8YoAwEx5dM19ULko4fIehI0t/MlViHmcnwUli2c4yZn6cj6fLxnJlLx28wL39k5tzJxcDA5JF+2f9VUpu5Cuo07Wgp6TCXnMtMPphepgMs24lYtrMlac6bAevNPjZLOtsM6v68ufXFSAbmN3pn0tVWycetsmznTMt2ZqUv8z33v+aeZtNM685xkp7N101UARB2gEAwdzu/0gwkfrvvuRtG2sbcAf075i7nP0u+jNuynSlJMxR/zffc/sx7GrY8bzatQbtLetrMKpzLdkeZFpI+Sb8dofxR05KltDl3vm8eP2fuCzaw77Ckq83VT3/O9pl8z33UtOhUm+6sKSMU/WYTCveV9Mm0zxQyx701y2caGAz9PfO9TKsOkEd0YwHB8Bnz/3m5mcsl27pX+5670Ty/xtzs83hJSy3buceMF7ElTZHkSsp0dVE47cqjkLlb+gHmiiaZmYvPS77v1ggGBibfZVppRnLTwO0ZLNtZYFqPfma68t4l6TnLdhabK8+ONVdNLZX0qRz2/TVTL8eaOXFOy7Si77mbLNs5S9Ltkq6zbOdsE/YaTP3uaiZovDjDLn5jrp6baFqUxnQvMQDDo2UHCIaBsSK7Svr4CP8Gb5VgWjdOMlcirTWXPDvmPlIXS3pnlkG+1Wn7/Zi59UHYtHQc73vuu0ZxH6qJphtOuc4abAZfP26+y84z78XNuJcPmgkET5b0XjNB3zckHZbLHdjN5z5H0gZJp1q287ER1r/PzM58gwkt7zcBqVvSVyW9LtMtJ8zkh382L+9Omw8JwDiF4vHh5r8CABSSZTuupLdLOsX33MXFLg8QJLTsAECRWbazr2kVe8VcYQYgjwg7AFBEZvDzreb7+EujnBsIQA4YoAwARWDZzhWS9jKDwadLutX3XAYmAzsBLTsAUBzHmIHTA3djf2+xCwQEFQOUAQBAoJVFN9bG9e3x1mkzi12MkrFxfbuojwTqIxX1kYr6SEV9pKI+UpVxfQyZuTxZWXRjxaKM10tGfaSiPlJRH6moj1TURyrqI1VQ66Mswg4AAMBYEXYAAECgEXYAAECgEXYAAECgEXYAAECgEXYAAECgEXYAAECgEXYAAECgEXYAAECgEXYAAECgEXYAAECgEXYAAECgEXYAAECgEXYAAECgEXYAAECghQt5MMt2GiRdL+kMSRMkPS/p077n3l/IcgAAgMpR6JadH0s6XtJxkqZKulXS3Zbt7F7gcgAAgApRsJYdy3ZaJZ0j6XTfc18wb3/Lsp33SLpY0uWFKgsAAKgchWzZOcSEq3+lvf+EpCMKWA4AAFBBChl2ppvHDWnvd0iamW3D6775LYVCoZz+XXjhhUO2v/DCC3Pe/qqrrhqy/SmnnJLz9jfccMOQ7Q899NCct7/rrruGbD979uyUdabNmpNx+yeffHLI9rkeOxQKqa2tLWXbtra2UW2f7sknn8x529mzZw/Z/q677hpxu4H6OPTQQ4dsf8MNN+R8/FNOOWXI9ldddRXnXo7bl+q5l+3/y3jPvYF/5XTuDVcflXzuTZs1pyy/98rx3Bvu384494ZT0AHKGYQkxbOtEItFc95ZT3e3OtrbhryXq+6tXUO27+vtyXn7rZs7h2wf6e/PefstmzYM2T4Wi+W8fWfH+iHbj8bGde2qrUp9PRrpx+7sWJ/ztrFYbMj2WzalZ+PMIv39Q7bfurkz5+37enuGbN+9tSvn7Uvl3ItEEvXAuZcbzj2+9zj3zP4i/WV57k2dOTQ0Jitk2Bk4e6ZJWpX0/vSkZcOqqqrO+SD1jY1DPnR9Y2PO2zc2twzZvrauPuftmydOGrJ9uKYm5+0nTJ4yZPuqqtwb4CZNnTbiDz2b1ukzU7bvy/37RpKGHHvS6jU5b1tVVTVk+wmTp+S8fbimZsj2zRMn5bx9bV39kO0bm1ty3r5Uzr3k//ice7nh3ON7j3Nvh472trI/94YTisezNqrkjWU7EyWtk3SO77m3Jb3/vKTFvudekWnbjva2+HhO5KDJJcVWEuojFfWRivpIRX2koj5SlXF9DO1PTFKwlh3fczdbtnOjpK9ZtvOCpOXmCqwF5pJ0AACAvCv0PDuXSVos6QFJ6yWdKOkE33OXF7gcAACgQhR0gLLvub2SLjX/AAAAdjrujQUAAAKNsAMAAAKNsAMAAAKNsAMAAAKNsAMAAAKNsAMAAAKNsAMAAAKNsAMAAAKNsAMAAAKNsAMAAAKNsAMAAAKNsAMAAAKNsAMAAAKNsAMAAAKNsAMAAAKNsAMAAAItXOwCAACAwumLxPXahpji8aHLNnVUqUPRMe23JyI99lpUL62NaZhd7zSNtdJXT63Pug5hBwCACtGxNaYTvt+tNVsyxZEmSd0FLtX4tDaF9NVTs69DNxYAABXi5sf7swSd4KJlBwCAChCPx/Wn//YPvl7QGlJ9TShlnWikX9XhmjEfY/7kkI7eLazmunEVdVTqwqER1yHsAAAq1lNtVXr5pb5iF6MgNnXH9UrHjlad5jrp/kub1FibGhQ62ts0debsIpVw5yHsAAAq0pMronrvH5ok9Ra7KAX3tv3CQ4JOkDFmBwBQkR5+KVLsIhTNe15fW+wiFBQtOwCAitSeNFD3jbtVa++Zwf/7PyTpqF3DOmxBdbGLUlCEHQBARVqbFHbOPbxGJ+8/9oG5KG3Bj7EAAAxjzZbY4POZE/h1GGT8dAEAFSm5G2vmxMoZrFuJCDsAgIrTH41r/dZE2JnRQtgJMsIOAKDirN8aH7w31NTmkGqqCTtBRtgBAFSc9s1JXVgTCDpBR9gBAFSc5MHJswg7gUfYAQBUnOTLzmdwJVbg8RMGAFSc5Dt/07ITfAWdVNCynRmSviHpREkNkp6XdKXvuQ8WshwAgGBp2xyT91xEvZF4DmtL/3g5casILjsPvkLPoHyHpE2SDpLUKelLkhZbtrOH77ltBS4LACAAtvbGddpPurWqM7egk44JBYOvYD9hy3YmmJacy3zPbfc9t8e08jRJOqJQ5QAABMfL62N618/HHnTqw3HtP5uwE3QFa9nxPXeLpPPT3t7FPNKqAwAYlV8+2qfP3tGb8t4Zh4Q1rTm3bqmqqpCOmLFJ01om7KQSolSE4vGxpeF0lu2EJTVnWu57bmfa+hMk/V3SSt9zT8627/ZVy+PhMDdoGxCJ9Iv6SKA+UlEfqaiPVEGqj7NubdRzaxN37z58XkQ/e8d2hUYxBCdI9ZEP5VofU2fOzvpTz2fLzrGS/EwLLdtpMF1XsmxngaTFktZJOmukHYfDNZo6c3Yei1reOtrbqI8k1Ecq6iMV9ZEqKPURi8X16qatg6/ftHu1vn9Gk6ZNmDyq/QSlPvIlqPWRt7Dje+59kkbM05btHGaCjivpEt9z+/NVBgBAZVjdGVd3347nkxtD+t15DQqNpkkHFaXQl57vJ+keSVf7nvu9Qh4bABAcS9YlZkDec0YVQQdZFfJqrGpJv5L0Y4IOAGA8lqyNDj7fcwZXUyG7QrbsvEHSIZL2s2znU2nLbvY994IClgUAUMaWrE1t2QGyKeSl5//IZUwPAKB8xONxrdgYV3d/fq7szdVzbUlhZzphB9kVegZlAECAXP6nXt367+JeZ7IHLTsYAWcIAGBMXloXLXrQmTMppKnN/CpDdrTsAADG5NePJ4JOa1NI03OcuThfWupD+sRbagt6TJQnwg4AVKhINK4r7+zV/UsiGstk+uu3Jjb60bvrdewe/EpBaeLMBIAK9Zt/9evmx8ffDbVoSkhv2q06hzWB4iDsAECZe64tqr++EFEkNvK63dtq1di04+aZv3xs/EGntlr63El1qqriYluULsIOAJSxDdtiesf/daurN4eVJUl1kvpS3pk7KaQ/Xdio6jGM853cGFJjLUEHpY2wAwBl7LHXoqMIOsP70tvqNL+VK5oQXIQdAChjT61M9F0dvWu13rAo+9iZbVu71NTcMvj6wLnVeste/CpAsHGGA0AZe2pV4h5R7zuiRifvX5N1/Y72DZo6s64AJQNKB+2WAFCmYrG4nk4KOwfN44ooYDi07ABAmYinTYbzckdscLzO1OaQ5kxkoDAwHMIOAJSBnz7cp+v8Xm3PcLX4wXOrFAoRdoDhEHYAoMR198X1tXt71RfNvM7BdGEBGTFmBwBK3FOrolmDzv5zqvTew7MPTAYqGS07AFDinliWSDpnH1ajb72zvqjlAcoNLTsAUOL+tTwRdg5fSHcVMFqEHQAoYeu7Yrp/SSLsvJ6wA4waYQcAStTtT/frgGu2Db6e3hLSglauuAJGi7ADACXqBw+k3rDzyF2qubwcGAPCDgCUoGUbYnqhPXHfq8MXVuuKE7jNAzAWXI0FACXoL89HBp+fsHe1fvW+xqKWByhnhB0AyEE8HtcX7urVnc9EFI3nsME4be1JHMTelzl0gPEg7ABADh55NaobH8lwr4adqLpKsvbmqxoYD8bsAEAObn86ksNa+VUVkj5yTK1amxiUDIwHfy4AwAj6o3Hd/Vwi7NzygQYdOGfn/61YXxNScx1BBxgvwg4AjOAfr0S1qXvHGJpZE0I6bvdqVVURQoByQTcWAGTRF4nr2nt6B1+fckCYoAOUGVp2AATOH57s1zf+2qttvXHF4s2qCnWNeV+RmLTNzO1XF5bOPbw2fwUFUBCEHQCB0t0X12fv6FH34OTD+WuF+cxb67TrNBrEgXLD/1oAgfKX5yNJQSd/Ttk/rAuPYr4boBzRsgOgrEVjcS3fmJiA7w9PJubC+cRbavWO3TdoyvRZ4zpGuFpcFQWUsaKFHct2jpL0sKSv+J57VbHKAaB8PfZaRB++tUdrtgw/pfGZh9aooU+a1EhQASpZUbqxLNtpkHSTpK3FOD6A8ve3FyN698+3Zww6hy+s1rxWeuoBFK9l51pJL0pqK9LxAZSxjq0xfey2HvVFd7xurpOmNSdab6a3VOna07hDOIAdCh52LNs5WtJ7Je0v6dZCHx9A+draG9f5t2zXwy9FB9+bPTGk2y9qpBUHQEZ5CzuW7YQlNWda7ntup2U7jab76lLfc9dYtpPTviORfnW00wg0gPpIRX2kCnJ93PyfGj38Un3Ke188bpsa+raoo334bYJcH2NBfaSiPlKVa31MnTk76/J8tuwcK8nPtNCM07lW0gu+594ymh2HwzUjfpBK0tHeRn0koT5SBbk+lnRul5S4R9Ulx9bqtCNasm4T5PoYC+ojFfWRKqj1kbew43vufdlm7zLdV2dLOiBfxwRQWZ5enei+ci9q0BGLmD0DwMgK+U1xvqQJkp5L6r6aKOn1lu2c6nvuIQUsC4Ay09kd17INO668qqmWDp5XXewiASgThQw7n5D0hbT3bpP0qKTrClgOAGUouVVn75lVqgszdw6A3BQs7Pieu0nSpuT3LNvplbTF99wMQwsBQNrWG9f3H0jcA+KgubTqAMhdUTu8fc89tpjHB1D6evrjeusPtumVjsTkgQfO5TJzALnjGwNASfvnK9GUoBMKSYcvZGAygNwRdgCUtAeWRlJef/f0eu06ja8uALnjzyMAJe3BpLDzu/MbdMzufG0BGB2+NQCUpK6euH76977BLqz6mh039wSA0SLsAChJn/pzj+54JtGqc9Qu1aqv4XJzAKNHxzeAkvTQS6ljdU49oKZoZQFQ3gg7AErO1t64OrcnXn/7nfU64xAaogGMDd8eAErOqk2xweeLpoR01mG06gAYO1p2AJScVZ2JeXXmTuZrCsD48C0CoOQkt+zMncTXFIDx4VsEQMlZndyyM4krsACMD2EHQMlZ1ZnUskM3FoBx4lsEQMlJ7caiZQfA+BB2AJQcBigDyCcuPQcwohUbY1rXFc9hzfGLxuNaa45VFZJmTaRlB8D4EHYAZHXTo3268o7eohx75oSQaqoJOwDGh/ZhABnF43H95OG+oh1/75l8RQEYP1p2AGT0v/aYVm4ydx0PS/vOLlz4mN5SpStOqC3Y8QAEF2EHQEb3vJC4GeeJ+4b1k7MailoeABgLwg4QAE+uiOrae3q1qTuuSKRR4fC2vOw3+RLwt+7D1wWA8sS3F1Dm4vG4PvHHHi1dNxBMqiXFRthqdGqqpTfvydcFgPLE6D+gzL24NpYUdHaO9x9Rown1XBUFoDzxpxpQ5u58JjGu5oS9q3XhIVs0eeq0vO2/pS6kea38XQSgfBF2gDK1vium79zfp5se7R98712H1mjPqTFNnVld1LIBQCkh7AAFFIvF9eBLUS1ZO/5up9/8q1+vrE/sp7F2x7iabRvGvWsACBTCDlBAi5+L6KLf9uyUfZ93ZK0aakLKz3VYABAchB2ggB5cGs3r/sJV0seOq9WpB4S1x3TG1QDAcAg7QAGtTJq35pT9w5ozaexXONWFQzrtwLD2ZnwOAGRF2AEKaEVS2PmkVas9phNUAGBno90bKJBINK62zvjg63mT+e8HAIXAty1QIGu2xBUxDTvTmkNqqGGSPgAoBMIOUCArNya6sOa3EnQAoFAKPmbHsp3zJF0haYGkNkk/8D33O4UuB1BoKzfRhQUAxVDQb1zLdt4t6YuSzpU0UdIlki62bOewQpYDKIbkwcmEHQAonEK37HxJ0qd8z33cvL7b/AMCb2VK2KEbCwAKpWBhx7KdWZL2llRt2c6/Je0p6VVJ1/ie+4dClQMohp/+vU+3/Sdxw8753FgTAAomb2HHsp2wpOYsq8w3jxdLOsuM1/mgpN9btrPW99yH8lUWoJQ8tSqqL9/dm/LefLqxAKBg8tmyc6wkP8vyo83jl33Pfck8/55lO++R9AFJGcNOJNKvjva2PBa1vFEfqUq9Ph58tkZS/eDrQ2ZH1Nzfro72nXO8Uq+PQqM+UlEfqaiPVOVaH1Nnzs66PG9hx/fc+yRlHIhg2c6e5ummtEWvSpqVbd/hcM2IH6SSdLS3UR9JSr0+XuvaLmlHF9Y7Dw7re2c0q7pq8k47XqnXR6FRH6moj1TUR6qg1kch29JfltQh6fC093eT9FoBywEU1HNtiYHJ7zmsRtVVDE4GgEIq2ABl33Ojlu18W9KXLNv5r6RnJF0g6WDzCARObySupesSYWff2dwLCwAKrdCXnn/dtCb9UdIUSS9KOsX33P8WuBxAQbzYHhu8RcSC1pAm1NOqAwCFVtCw43tuXNI15h8QOBu2xfSTh/u1unNHwlmddOPP/WjVAYCiKPjtIoAg+9Z9fbrp0f5hl+03m8vNAaAY+PYF8ujx16LDvh+ukuz9+NsCAIqBb18gT6KxuF7pSAxG/u7p9ao1/8MOmVetBVP42wIAioGwA+TJio1x9Zo7QkxvCendr6spdpEAAHRjAfnzUtIl5rtP478WAJQKvpGBPFm6LjFeZ48Z/NcCgFLBNzKQJyktO9P5rwUApYIxO8AY9UXienp1TP2RHXPpPLM6EXb2IOwAQMkg7ABjEInGdfJPuvVsUsBJRssOAJQOvpGBMXhqVSxj0JkzKaRpzdwWAgBKBS07wBg89lpk8PnsiSEtaN3xd0NLvXTRG2sVChF2AKBUEHaAMXh8WeLKq8uPr9XZh9UWtTwAgMzoxgJGKRaL619JYefwhfzNAACljLADjML6rpg++vsebe7Z8Xpqc0i7TKXLCgBKGX+SAqPw2Tt6dfdzifE6RyyqZnwOAJQ4WnaAUfjPitS7mp+6P38vAECp45sayFE8HlfHtvjg6xvPqddJ+/JfCABKHd/UQI46t0v9pmGnuU6y9+Ou5gBQDujGAnK0fmtiEkEmDQSA8kHYAXLUsTXRhTW1mf86AFAu+MYGcpQadmjZAYByQdgBcrS+KxF26MYCgPJB2AFytH4rYQcAyhFhB8hR8mXn01oIOwBQLgg7QI46krqxpjQRdgCgXBB2gBylXHpOyw4AlA3CDpCjlG4sLj0HgLLBNzaQI67GAoDyRNgBcrCtN67t/Tue14V33C4CAFAeCDtADtanTSgYCtGyAwDlgrAD5OB/7dHB57MnEnQAoJwQdoAc/POVRNg5fGG4qGUBAIwOYQfIwd9fToSdo3erLmpZAACjU9A/US3b2UvSdZKOlFQj6UVJV/uee3chy4FgiMbi6o1I2/ul7r54DluMzYatcS1dt2OOndpq6bAFhB0AKCcFCzuW7YQk3SvpH5J2ldQt6UOSbrdsZ3/fc18sVFlQ/v7+ckQfurVHG7bFJbVI2lqQ4x46v1qNtYzZAYByUshurOmS5kv6re+5m33P7Zd0owlcBxawHAiAHz/cZ4JOYb2RLiwAKDsFa9nxPXetZTsPSTrPsp3HJXVJukjSBkkPFqocCIaX1iVu3VBXHVeoaue3tuw/u1ofeEPtTj8OACC/QvF4fv46tmwnLKk503Lfczst25ku6S+SDpEUl9Qh6V2+52YNO+2rlsfD4Zq8lDMIIpF+VXJ99ESkw37YIkmqDsX16MWb1FBXufWRrtLPj3TURyrqIxX1kapc62PqzNlZ/+LNZ8vOsZL8TAst25loxuy8KOkkM8jifZLusmznDb7nPpexkOEaTZ05O49FLW8d7W0VXR875rzpliTNb61SQx3nR7JKPz/SUR+pqI9U1EeqoNZH3sKO77n3ScqYrCzbsSUdJOkk33PXmbd/YtnOhySdJ+kT+SoLSt/6vj75nZ3qiTOXAAMAACAASURBVMVyWDvVi+1RNe8dkSS1TA3p1q4eNal9J5SyPG3r2kp9JKE+UlEfqaiPVOVYH/VVVTp7+vSs6xRjdrT0QBQ2XVqoEPF4XB975RUt3b59zPuYsP+Oxw5Jv+yS1FWYq7HKBvWRivpIRX2koj5SlVl9TAqHSyrsPCKpXdLXLdu5VNI2SedI2tMMVEaF6IhExhV0AAAYjUJejdVp2c5bJV0raamkWvN4hu+5fy9UOVB8r/X0DD6fWVurkyZPHtX2t/0nojVbdnR/OQeGNaW2W41NGcfGV5zubVupjyTURyrqIxX1kaoc66O+auRZdArajeV77jOSTi7kMVF6liWFncOam/XROXOGrPPIKxFdc0+vNnUP7eFcuSmuiBnqc8nbmlTf0x7IAXVjFdQBhmNFfaSiPlJRH6mCWh/c0RAFl9yys6i+fth1vrS4V8+tyT54ub5Gmj0hpI09WVcDAFQ4bgSKghsp7ERjcS1ZN/JVWhccVauqAkwmCAAob7TsoOBeTRqcPFzYadscV7+5yfiUppDu/FDjkHWa66TpLWR1AMDICDuQzOXghdAVjWpDZMccObWhkGbX1Q1ZZ/mGRKvOoikh7TKVUAMAGDvCDtTe16dLX3lFLxX4cvD5dXWqDg3thlq2MRF2Fkwh6AAAxoffJNCdGzYUPOhI0m4NDcO+v3xjopVpYSunKABgfPhNArX39RX8mHPr6vTeGTOGXZbcjTWfsAMAGCe6saBNZgyNJH1zl1103KRJRS1PcjfWwilcbQUAGB/+bIY29PcPPm8NFz//rtiQHHY4RQEA41P832wouuSWnXyHnd5IXJ+7s1f/WRHNaf14XNpspuFpqJGmNdOyAwAYH8JOhYvH46ktOzU1ed3/7U9F9Jsn+nNYc6gFrVUKDXO1FgAAo0EfQYXbHoup18yxUxcKqTGHG6qNxotrc2vRSRcKSe87Ir/BCwBQmWjZqXAbk7qwJtfU5L0lZXVn4jLyK0+s1Vv2zO2Um9IU0owJZHEAwPgRdircxqQurCk7YXDyqs7EYOPDFlRrn1nVeT8GAADZ8KdzhUtp2dkZYWdTomVn7mRONwBA4fHbp8Ilh518D07u6Y9r/dYdYae6SprZwmBjAEDhEXYq3MadOMdO8nidmRNCClcTdgAAhUfYqXAbd+IcO6uTxuvMncSpBgAoDn4DVbhNO7EbK3lw8tzJtOoAAIqDq7ECZk1fnx7s7FR/PJ7D2tKL3d2Dz/PdspM8OHkOLTsAgCIh7ARIfyymC5cuVdsY72Kej7ATjcX1hycjem1DTA8sSbQazZ1Eyw4AoDgIOwGyord3zEFnUnVYs2rqFInm1iI0nLikK+/o1S3D3B6Cy84BAMVC2AmQ1b29g8/n1tXpuIkTh6zz7OqYHl8WUX/SXRzisZDWrWzR7rd2D1k/H5rrpIPmMpkgAKA4CDsBsjqpVef1LS26dO7cIevs84subdo5mSaFtVe1Dl1QrXCVdPxeYU1upBsLAFAchJ0ASe7Cml1bO2R5XySeEnSqd0LPUnVIOnn/sL57Rr1qmFcHAFACCDsBktyNNaeubsjyzdsT43EmN4b0whebC1Y2AACKhVGjAZISdoZp2elMCjuTGgpWLAAAioqwExDxeDy1G2vYlp3E84kNdDEBACoDYScgOiMRdcd2zFjcWFWlSdVDr35K7sYi7AAAKgVhJyCSr8SaU1enUGhomEntxiLsAAAqA2EnINqSxusMdyWW0lp2JnEpOACgQhB2AmJVWsvOcDq76cYCAFSevF56btnOIkk3STpG0iLfc5elLW+QdL2kMyRNkPS8pE/7nnt/PstRiV7r6Rl8vjBT2KEbCwBQgfLWsmPZjiPpMUnLs6z2Y0nHSzpO0lRJt0q627Kd3fNVjkr16vbEpVaL6uuHXSd1gHJBigUAQNHls2WnVdKbJM2TdG76Qst2WiWdI+l033NfMG9/y7Kd90i6WNLleSxLRYnF41qWNGYnc9hJPKcbCwBQKfIWdnzPvVE7Qs28DKscYo73r7T3n5B0RL7KUYnW9vWpx1x2PrG6WpNraoZdj24sAEAlyinsWLYTlpTx3gK+53bmsJvp5nFD2vsdkmZm2zAS6VdHe1suRa0I6fXxVE+iVWdedVXGutrQ1Shpx/w78W0d6miPFaC0Ox/nRyrqIxX1kYr6SEV9pCrX+pg6c3bW5bm27Bwryc+00LKdBt9zezItH0FIUjzbCuFwzYgfpJJ0tLel1MfGtWslbZIk7TFhYsa62ta/dbCqF8ybrqmTg3ExXnp9VDrqIxX1kYr6SEV9pApqfeQUdnzPvc+EkvFoN4/TJK1Ken960jIMoz8W03+2btU201XVtb1HLZ2JxrTvrV49+DzTeB2ldWNNZp4dAECFKORdz5+U1CfpDZJuS3r/SEmLC1iOsnPFa6/poc2bU9/cNHzP4cIMYacvEtf2/h3Pq6ukpuHnHQQAIHAKFnZ8z91s2c6Nkr5m2c4L5hL1yyUtMJekYxjd0ejQoJNBc3W19m9qGnZZ+n2xhrudBAAAQZS3sGPZzhITXAYGgiyxbCcu6Wbfcy8w710m6RuSHpDUIukpSSf4npttbp6K9mrSZIETqqt1SHOz+np7VFuXaMGprarS7g0NeuvkyWoZ5gagGnIl1k4uNAAAJSSfl57vmcM6vZIuNf+Qg5eTJgs8vKVFX99ll5QBZHc9068/PxXR6lhcDyoqqXvY/WxOGj7OHDsAgEpSyDE7GIPksLNrQ2qTTNvmmD78ux5FRnkFOWEHAFBJgnHtcYC9nNSNtVta2PnXsuiog44kvW0/Mi4AoHLwW6/EvZLUspMedp5ZHR187hwY1tsPHH7m5GTzWkPae+bw43oAAAgiwk4J29jfr42RiCSpoapKc2pTrxd/elWiWedt+4d1wj78OAEASMdvxxKW3IW1S329qpIuF4/F4nq2LdGyc8Cc3Ftrtqzu10PXrFfniv48lrZ44vG4QiEu6BtAfaSiPlJRH6moj1TlWB91E6r0rt9mui3nDoSdErYyKeykTxa4bGNcW8ziyY0hzZ2U+6DjJYu71LG0L38FLQlZ7zhSgaiPVNRHKuojFfWRqrzqo7pu5PIyQLmEre5LBJK5dXUpy5LH6xw4t2pUkwRu74zmsBYAAMFAy04JW9WbuJv53LTxOs+vSYzX2X/26AYcR/sSKfjoT0/Vwjc2jqucxbZh7RpNmTGr2MUoGdRHKuojFfWRivpIFdT6IOyUsNVJYWdOWsvOio2JsLPbtNE10CWHndqmKtU0lHcDX7g+VPafIZ+oj1TURyrqIxX1kSqo9RG8TxQgbUndWLPTws7KTYmwM2/y6CYJjPYmwk64jgkGAQDBRtgpUVsiEW2J7hhbUxcKaWo4tRFu5aZEYJnXOrofYyQp7FTXEnYAAMFG2ClRyYOT59TVpQxA7u6XOrbuCCzhKmnWhFG27PQTdgAAlYOwU6JSxuukDU5esyXxY5szKaTqKrqxAADIhLBTorINTl69JRFQ5o+yC0vp3ViEHQBAwBF2SlA0HteTW7cOvh4adhI/tnmTRv8jTL4aK0w3FgAg4Ag7Jehbq1bpn1u2DL7euzF1Hpy25LDTOvqwkhx2GLMDAAg6wk6J2RqN6rb16wdfn9LaqoOamlLWWb05EVDmTR5Dy05v4rJ1urEAAEFH2Ckxy3t6NBBFFtTV6YsLFgy5FURyN9Zox+zEY3FFk+7/ScsOACDoCDslZlnSzT93a2hIudP5gJQxO6OdUDCtC2s099QCAKAcEXZKzPKkq7AWpA1MlqSunrg29+wIKHVhaXrz+MIOAABBR9gpMcktOwvr64csT75NxNxJIVWNco6dSB+XnQMAKgthp8Qkh50Fw4adRFiZO6bByVx2DgCoLISdEhKNx7UyuRtrhJYdJhQEAGBkhJ0S8u+uLvXFd4SRKeGwWqqrh6yzYuPY73YuxuwAACpQOId1UAC/XbdO31q1avD1cON1lNaNNX8s3VjMngwAqDC07JSAeDyuW9auTXkvfdbkAcndWGObUJBuLABAZSHslICXt2/X2v7ETH9vnTxZ586YMey6qWN2Rh9WIn1JsyfTsgMAqAB0YxVRPB7X35ZE9duOjZJpyFnQO0FTX5mtX70Sk9Sbsn5/TNpiLtZqqJGmNI1vzE6Ylh0AQAUg7BTJip4efW1pux5a3qva1h4NDEV+6r8NenRF34jbz5tcNabZj1O6sWjZAQBUAMJOkXxz1So9EdmihjmJ9+Ixqae9Kdtmg47adeiVWrlIufS8ll5MAEDwEXaK5OXt24e8t7Bvos48umHEbZu1Rece0zym49KNBQCoNISdIojH49oYiQy+3vjITH371CadPG+CanLomupo71fzGINKlNtFAAAqTN7DjmU7iyTdJOkYSYt8z12WtnyGpG9IOlFSg6TnJV3pe+6D+S5LqeqKRtVvJg+M9YfU39ait81uzinojFeEMTsAgAqT10Eblu04kh6TtDzLandImiHpIPP4kKTFlu3MzmdZSllyq06sJ6yFrSHVhgsTPJhBGQBQafI9QrVV0psk3TzcQst2JpiWnMt8z233PbfHtPI0SToiz2UpWRuT5tSJ9VZrt+ljG2w8Fik3AqUbCwBQAfLajeV77o3aEWrmZVi+RdL5aW/vYh7b8lmWUpbcshPtqdbu0wp3VRQtOwCASpNz2LFsJywp4yVAvud2jvbgpqXnJkl3+577WKb1IpF+dbQHJwut2LZt8Hmst1ozmzeroz2SdZtk46mPbZsTx+ne3qmO9q4x7aeUBO38GC/qIxX1kYr6SEV9pCrX+pg6M/tImNG07Bwryc+00LKdBtMtlRPLdhZIWixpnaSzshYyXDPiByknvW1t0uYdISPWE9bBu03R1Jm5d2V1tLeNuT7CVWsl7bjsffL0Vk2dmdu8PqVsPPURRNRHKuojFfWRivpIFdT6yDns+J57n6S89HtYtnOYCTqupEt8z+3PYbPA6OhLfNxob7X2mFG4biyuxgIAVJqCz7Nj2c5+ku6RdLXvud8r9PFLwcptibAzJRwe85w5YxHtTx6gzAzKAIDgK+hvO8t2qiX9StKPKzXoSFJ7b2LczILm2oIeO+XeWFyNBQCoAHlt2bFsZ4mkBUkhaollO3FJN/uee4GkN0g6RNJ+lu18Km3zgXUCrzPSP1jze06uKeix6cYCAFSafF96vucIy/+Rr3E/5aw7FB18fuC0Arfs9BN2AACVhXtj7QQvLO/WA492SfGhy2KSFkSqJVVLMWn+9IhWbO1WfJh1M+nqjGnbpG4Ne4AR9HXFBp8zqSAAoBIQdvLs30u36clL16mxL3OQOFktg89fUIdeGNOR1o1pq2S07AAAKgGX4+SZt3ijarMEnVJRP6lKtU38+AEAwUfLTp7FXusfzJDrZ0W1PcOc02FVabfmOk2pH/19sfp6e1RbVz/4erQ3Sw/XVWnPk1tUVaCbjwIAUEyEnTyKRqOasirxeo8Lp+jtb5yU9+PsmOFyRt73CwBAENGPkUdPvNij5q4dVdpXG9cJr28ZcRsAALBzBapl55HntuqRJ7oUj43+KqV82PJa3+At3NfPiamxbvRdVAAAIL8CE3aeeHGbnrlivab0Fm8cytSk533zA1O1AACUtcB0Yy2+rUN1RQw66fY6PMPIZAAAUFCBaH7o749qytPRwey2apeYIo3FK0/TPvX69AlTilcAAAAwqOzCzq/v7tAr921RKHEvTVX1xzWvc8f4mJ76mD7yrflqbSm7jwYAAHaCskoE23qi6rphixZuy9z7tmafEEEHAAAMKqsxO0tX9aoxS9CJhOM6/O2TC1omAABQ2sqqCWTthv7B510TYoqeljow53UHN+mNBzC3DQAASCirsLNxc2KgTndLXJ9538yilgcAAJS+surG2pwUdvrrSucycwAAULrKKux0b40NPo/WZ10VAABAKrew07M1Ovg8Vk/LDgAAGFlZhZ3+7kTLTpxuLAAAkIOyCjvRpLBT3VhWRQcAAEVSXomhJ3E38zBhBwAA5KCsEkNVUtipayqrogMAgCIpq8RQ3ZsIO40t1UUtCwAAKA9lFXZqehLPJ0woq/kQAQBAkZRX2OlNPG+dSMsOAAAYWVmFnbrexOXm01priloWAABQHsoq7NT2JMLOrCmEHQAAMLKyCTvdvVHV9u0IO7FQXDMn0Y0FAABGVjZhp21D4iagfXVx1dQQdgAAwMjKJuys6egbfN7HTUABAECOyibsdHSmtuwAAADkomzCTufm5LBT1KIAAIAykteZ+SzbWSTpJknHSFrke+6yLOseJelhSV/xPfeqkfbdtSWqFvM8Us8dzwEAQG7y1rJj2Y4j6TFJy3NYt8GEoq257n97V+KO51FadgAAQI7y2Y3VKulNkm7OYd1rJb0o6b+57rxna3TwebyhbHrfAABAkeUtNfiee6PvuUtGWs+ynaMlvVfSRaPZf6QzEXaqJxJ2AABAbnIas2PZTlhSc6blvud25rifRtN9danvuWss28m5oFVbYoPZrLGVm4ACAIDc5JoajpXkZ1po2U6D77k9mZYnuVbSC77n3pJ7EaVIpF+1WxOXmzc196mjvW00uwiUSKS/oj9/OuojFfWRivpIRX2koj5SlWt9TJ05O+vynMKO77n3SRrXJVCm++psSQeMdttwuEaNSWN2dtttsqbOnDie4pS1jva2EX+wlYT6SEV9pKI+UlEfqaiPVEGtj0L2B50vaYKk55K6ryZKer1lO6f6nntIpg2j0ZiauhJZa5+FDQUoLgAACIJChp1PSPpC2nu3SXpU0nXZNlzTGVdN/46w01cb1+xW7osFAAByk7ewY9nOEkkLkq7wWmLZTlzSzb7nXuB77iZJm9K26ZW0xffc9mz7fnVNYrxOd3Nc1dWEHQAAkJu8hR3fc/ccwzbH5rLemvUxNZrn3c3cFwsAAOSuLCas6dyYCDh9LdwqAgAA5K4swk5PZyLsxFrKosgAAKBElEVyiHclwk54EuN1AABA7soi7FQn3S60eQqzJwMAgNyVRdipSwo7U6bVFLMoAACgzJRF2KnpSzxvnUTLDgAAyF2ZhJ3EFViTWhizAwAAclceYac/8bx1AmEHAADkrjzCTlLLzrTJdGMBAIDclUfYMffFiofiam2iZQcAAOSuLMLOgP4aqaaGsAMAAHJXXmGnlvtiAQCA0SmzsFPsEgAAgHJTZmGHlh0AADA6ZRV2IkyeDAAARikUj9NaAgAAgqusWnYAAABGi7ADAAACjbADAAACjbADAAACjbADAAACjbADAAACjbADAAACLVzsAmRi2U6DpOslnSFpgqTnJX3a99z7i122QrBsZ5mkOZKiaYsO8D13aSXUj2U7iyTdJOkYSYt8z12WtCzr5w9i/YxQHxV1vli2M0PSNySdKKnBfJ4rfc99UBV4fuRQHxVzfli2s7+kayW9QVKtpCWSrvE993ZV5rkxUn1UxLlRyi07P5Z0vKTjJE2VdKukuy3b2b3YBSugC3zPrU/7t9QsC3T9WLbjSHpM0vIMq4z0+QNVPznUhyrsfLlD0gxJB5nHhyQttmxntlleUedHDvWhSjg/LNtplvSApJclLZI0TdLtkv5o2c4+ZrWKOTdyrA9VwrlRki07lu20SjpH0um+575g3v6WZTvvkXSxpMuLXMSiqpD6aZX0JknzJJ2bvGCkz2/ZzjUBrJ+M9TGSoJ0vlu0M/HV5ve+57ea9b0j6jKQjLNt5sJLOj5HqQ9KfR9g+SOdHg/ncv/U9t1s7Pt8PJH1V0n6W7bRX0rkxUn1IeiHbxkE6N0oy7Eg6xJTtX2nvP2H+81aKMyzbuULSbEkvSbrK99zFlVA/vufeqB3/2eYNs3ikzx+4+hmhPgZUxPnie+4WSeenvb2LeWyrtPMjh/oYEPjzw/fc9ZJ+PvDasp2pkj4raZWk+yvw3BipPgYE/two1W6s6eZxQ9r7HZJmFqE8xfCMpKWm+XCepDsl3WnZzhuonxE/fyXWT8WeL6Zl4yZJd/ue+1ilnx/D1Icq8fywbKdX0nozxs3yPbejks+NDPWhSjk3SrVlJ5OQpIq4c6nvuaemvXW1ZTunSbpQkp9hs4qpnwxG+vyBrZ9KPV8s21kgabGkdZLOGmH1wJ8fmeqjEs8P33PrTEvGxyQ9YtlOtpaIwJ8bw9WH77lLK+XcKNWWnXbzOC3t/elJyyrRy5JmUT8jfv5Kr58BgT5fLNs5zDSn/1PSib7ndplFFXl+ZKmPTAJ9fmjHL/gO33O/KGmtGWNSkefGgGHqI5PAnRulGnaelNRnLpVLdqT5jxxolu0ssmznJ5btTEpbtK/pT63o+snh81dU/VTi+WLZzn6S7pF0re+5F/ue25+0uOLOj2z1UUnnh2U7b7NsZ6VlO01pi0KSIpV2boxUH5V0bpRkN5bvuZst27lR0tcs23nBXG57uaQF5jK4oGuXdIqkCZbtXGJOtk9K2l3SOyu9fkb6/BVYPxV1vli2Uy3pV+Zn/b305ZV2foxUHxV2fjxm5pL5gWU7l0vaLukiSbtJ+nOlnRsj1UclnRslGXaMy8wkWQ9IapH0lKQTfM/NNs9IIPieu92ynePN53/JpPBnJB3re+4Ss1qg68eynSXmP9RA6+MSy3bikm72PfeCHD5/oOpnpPqosPPlDeYqkf0s2/lU2rJKPD9GrI9KOT98z91g2Y5lJtFban7RvyjJSRqsXTHnRi71USnnRigeL6sxRgAAAKNSqmN2AAAA8oKwAwAAAo2wAwAAAo2wAwAAAo2wAwAAAo2wAwAAAq2U59kBypJlO1dJ+tLAnYF9zx0yv4NlO8skPeh77vsLWK5fSnqf77mhQh1zLCzb+aa5i3dY0t6+564qdplUpPpLOpcW+Z67rFDHBYKGlh1g53m9+aWNHFm2s7eZofUBSSeauytXBMt2qizb6bRs59ikt2+QdJiktiIWDSh7tOwAO88fzDTrf/I9d1OxC7OzWbZT53tu7zh3M9083uV7blndeycPDpQ0MfkN33PbCDrA+BF2gJ3ncjM1+1ckfTTbipbtPChpL99zZw7z/kLfcxea17+R9BZJb5T0c/NX/yZJ10v6nqTPSvqwpAmSHpX0Qd9zV6btc6Gkn5h9RCTdLemjyYHMsp3XmXIfmTTF/Hd9z/1V0jq/lPR2SadJ+qV5e1GWz/hG0yXzerPPlyX9TNL3fc+ND3QTmdV/YdnOL7J134xURst2bpF0hqTpvuduTtv2UUmzTd3GLds5RdKnzQ0Qw2Zq/et8z/1Dls8zUN4G33N70t9P7u6ybOcoSV8wt3VokvSqpJ/6nvsjpXZXSdIDlu3I99zQcN1Ylu0sMNP/W5ImSVoj6XZJX/A9d4tZ5wLTKrSfpA9KOltSo5nq/+O+5/4nqWwfN+ssMufD8+az35HpswPlhm4sYCcxY02ukXSxZTsH5mm3EUk1km4y/06VtEzSdyTdKGlvSeeaX6zHS/rhMPv4k6SHzLbflnRmUliRZTuHSnrY3AfnXLPefyT90rKdD6ftKyTp6+Z4Z2YqtGU7R0v6m2m5ON/cfPARSd+VdLVZ7SpJF5vnX87WfZNjGX9nQtDJadvOk3S4pN+aoHOcCQsbJb1b0jtMgPi9ZTsn5vhzyciynb0k+ZKaJb1f0tsk/UvSDy3bucisdoP5zDJ1cFiGfbWau01bkj4n6QRzQ8aLJd1l2c5AwIqYx++b7/kzTaDZR9Kdlu2Ezf4+LOlbkn4j6SRJ75K0UtKfTUADAoGWHWDn+pakD0j6kaSj87C/uKRW0ypwi3b8wopLus/cEHIfMyD6fst2TjetN+lu8T33O+b5/eaX/wct25nje+5qSV+V1Cnprb7nbjPr+aZF4UuW7dzge+7AL9MJkn49UJYsviypS9Kbfc/tStrnTEmfsGznG77nLjM3PJWkZb7n/jvL/kYso6R7TYB5h/llPuB0E9IG3psn6S+mNabT1Om/zbZnSbpnhM82kl3Mz+fTvue+aPb/dxPCzpb0f77ntplB65K0JMtn/4ikOZKO8j33EfPeg5bt9Jtz7ThJ95vzRJLW+p778YGNTUj8lKS9JD0n6a2SnvU99+tJ69xngmPfOD83UDJo2QF2It9z+yRdKukoy3bem8dd35v0fIV5vD/tyq8VppsjXXr3xIPm8RDLdmrML8wHk0LEANeMqdkn7f2/Ziuo2efRkv6WFHQG3G26Vw7Nto9h9jdiGX3P7Zf0Z0knWrbTmLTOGZKe8T33Oe34Gf3a99yTB4KOea9T0gYThMbF91zP99xTB4KOeS8qafkY9n+cpHVJQWfA3eYxPdzemfZ64FxpNY+rzd3SP2rZzgRTtpjvudf5nvuvUZYNKFm07AA7me+5nmU7d0m6zrKd24f5hT+Wfa5PetlvHtenrdZvWjDSrUx7vdY8TpM0VVKdpLMs2zkrw+FnS3om6fW6EYo71XQnpR9XSd1Us0bYR/r+ci3j70z3zYmma2aupCMkfWZgRct2mk3L09slzTVlHTDuPwhNl9GVpitpkaT6pMXLR7m7OZKGuxQ/Uz2uTXs90CI38Lk+L2kPST+Q9B3Ldh6XtFjSL3zPHennCpQNWnaAwrhU0mQzLqXY0uf9CSW9P7Dsd5IOzvDv0bTt+5XdwD5rc1hnNOXPpYwPSGo3XVkyXViSdGvS/n5ufj5/NN06h5h9bBhFmbL5qglT/zRjiw41+392DPuKj7Ies9ar77kbfc89XtIBZiB0vxln9qIZawQEAi07QAH4nvuqZTvXS/qMZTs3DrNKLMMfHzN2QnFmpbWyDBxjrZnXpkdSve+5T+XpeAP7nDPMsrnmcfUY9jdiGX3PjVm2c5ukc00Ly+mS/j5whZplO3WS3inpj77nXjGwnWU7tWZAcTYx85j+c0v/mZ0j6Qnfcy9IftOynZYcPmu6VZIOtmwnlNZlOZZ6HOR7tTSVtQAAG3FJREFU7rMmfF1r2c4+JixeYcabAWWPlh2gcK413Q0/GGbZJkmTLdsZ7OIwg2132wnleFva6zebX9z/NgOPH5D0Vst2pievZNnOByzb+VzSFT85Mft8UJI1MC4kySlmoHG2wcjD7W80ZbzVXAXmmEHcyYOVG8wffemtOBeZrrLqLEUZuFR/dtLxW8yl9cma0/dvLnVfmLb/gfCS7Zi+GW9zXNr7p5jH+7Jsm8KynRrLdr5t2Y6dcgDPfUHSa6ZbEwgEWnaAAvE9d7tlO5dLum2Y7oUHTFfL1Zbt/MC0gnzTzB0zlhaATGKSzrdsp0HSfyUdYy6H/kPSGI0vSvqHme/lanMZ9nHmUuefDnf7ixxcZS4V/6tlO9+QtN1c6XSSpEuT56nJUc5l9D33Uct2lptL5COmu2pgWadlO09LOtOynYfNfk42g7AflbS/ZTuW6YJK94CkT0i6yrKdL5gr064143Bak9Z7UJJt5r55UdKbTDj5kyTHsp23m6kA1pj1P2jZzuQMweWnZh6l31q281kz7cDR5tL/233PfSzXCvQ9t9+ynUWSbrFs54tmDp4aE4YPNNMZAIFAyw5QQL7n/tH8EktvHfk/SdeZAPCS+aV2raSnzS+gfKiRtNXMpXKiGYh6maRfS7owqYz/Nr+QV5j5X+4121xpxraMmu+5j5vJELdJutnMa7O/udz7e2PY32jL+DtzCfg9vuduTFt2thnM/HMz63WLmW/nmyYc3jRcK4fvuYvNZdxHSvqfpN+b7T0lrhqTuVz8XrO/O8znPs3McbTW/Oz3MvMQ3WGWXZ8+m7I55mYTbu4z58tfTVfTddnmOcriPWZ+po+bff3JfJ4zkyeQBMpdKB4fyx9pAAAA5YGWHQAAEGiEHQAAEGiEHQAAEGiEHQAAEGiEHQAAEGiEHQAAEGiEHQAAEGiEHQAAEGiEHQAAEGiEHQAAEGiEHQAAEGiEHQAAEGjhkVZ46qmnZjU1NX02HA5PDIVChCMAAFAS4vF4LBKJbN62bdvXDjrooDWZ1st61/Onnnpq1qRJk26aP3/+tKqqKm6PDgAASkosFgutWLFifWdn5wcyBZ6sLTVNTU2fJegAAIBSVVVVFZ8/f/60pqamKzOuk20H4XB4IkEHAACUsqqqqng4HJ6QcXm2jRmjAwAAykG2zEKYAQAAgUbYAQAAgTbipedB8uWvfmPhPx55bEq2dT580fnLnNNO3pBtnZWrVteed+FH909/v6GhPjpzxoyet5/6trX2idamTMetrq6OT5w4oX/vvfbo+sC572lbMH9eX/J+urq2Vt1y6+9nPv7Ek5PWd2yoC1dXx6dPn9Z77DFHb3j36c66cDi3H9tLL79Sf/W11+/a3r623vfcJ3PaCFl99NJP7blk6cvNmZYfd+wb11/56U+sOPOc8/bfb999tnz+s59cXtgSBs+/n/xv8+//6M5cvnxlY1dXV7ixqTGyaMGC7jPf/c41rzvkoG257OOxJ/7d8oWrrtnjFzf88Nl5c+f05bDJsH5ywy9m337n3bPuXfwn/j+NQywW0+133j3lvvsfmrpq9eqG/v5I1YQJLf1777lH17tOd9bus/ee2wfWde9YPOXH/3fjwl/f+JNnZs2a2Z/L/s8857z9j3nT0Rs+dOF5bWMt48D3/CUfvvC1U08+aeNY9zMWH730U3tWVVXHv//try/dmcexbOfQM9759rYLz39fxku2y+EYuaiosHPZxz688kMXnbdq8PUnr9xr4YL53R+/5OIVA++1tLREc93fee87Z8Vb3vymwVCzcWNnzV/vu7/1O9//8S5btmxZcea73rl+YNmUKa193//21/8nSZH+SOjVZcsbfnbjr+Zd+YWr9/jVjT95biDArF/fEf7EFZ/fMxaNhs569+lt+++3z9ZoNBr695P/bfn9be6cRx97YvI3v/7VpfX1dVkHji/27m396c9uWjB50sQxf7FjeLvvtuvWq75wxSvDLWuob4gVvkTB9Y9HHptw9TXX7X7cMW9cf+57zmxrbZ0caWtbU3vrH/4084tfvmbP67929f/23Wfv7TnsCiUiGo3qyi9+Zdf/vbik5Z1vP3XNZR/70PK6urrYq68ua/j9H92Zn/zM5/e+8orLXz76yCO2FLushXLO+y/c76MfvnD5Ea9/XZckfeWqz70cCoWKXaxAqaiwM2FCS3SCWgZfV1VVxWtra2PTp02LjGV/TU2N0eRtp0+bFtlrz91XL1u+omGxd++M5LBTFaqKJ687e/as/t7e3lVfv/67u7308qsNe++1x3ZJuv47P1jQ29tb9dMffueF1smTBoPXLosW9h72ukO6Lrnsin1uuPGXsz/2kYtWZyvbr2/53dxLPnzha6tWt9X/7g9/mjOWz4fhhcPh+FjPGYyOd48/deaM6T2f/fRlg3+QzJk9q+/AA/Z75ZLLrtjz2ef/10zYKS83//b3M556+tmJ11375RcPPGC/7oH358+b23fUkYdv+eiln9pzsXfvtEoJO52bN1evW99Rl/ze5EmTcv6jG7mpqLCTi4Hm7q995YtLXnfowVsH3j/p1DMOOfXkk9pzaRadP29uz9KlL7eMtF48rpAkNdTXxyRp+YqVtU89/eykC847d3ly0BmwaOGC3rdab17n/+2BaRdf8IG22trajK07133ty0sWLpjfe+Mvb5k5UjmAUhWJRELRWCwUjUZVXV09+H5tbW38/370nRcHXq9ctbr2Rz/9+bz//W9JS19/f9X/t3ffcU2cfQDAf0nIDgkzEGYYARVXcdeFo7Wl7mq1rbWvWrXaWhWtOOu2KopaUao460ClWqmj7gXuqoDslRAgQEKY2ePu/QNCE2RJ7Vvkfb6fD58Pubvkucsld797fr974uTE1UwYN7p4ZPAIi/RDQUEhdcu2XfxcoZBFp9MNwSPek86Y9kWxaf6jx0+tjx6Pds0T5zOIRBLu7cVXfjXti4KuXQJUgLwRl69c5/bp3bPcPNAxIZPJ+J5d29KbStUbjUY4cPgX3u278Q4VFZVkFotp6NXjnYp5c2cVMBiMup5VDMNg5+5It7tx9+21Wh0xoFOH6iUh34kcHR0MUJvm3xd1xC07V8jU6/VEJ66jdvIn4yXvDx9a0dJtKSsrt4qIjHJLSHzJUanVJAd7O92I94dJv/hskhQAIFcoos7+ZmHn5UtCsi9dueaYlp5pTSaTsUED+skXzJtTUCgpqiuJWLVmo5+9na3u1PFDL83TWEJRHnXW3AWdV68MzTrz63nn7JwcFofN0c+cMVXswuPptu+M4BdKimjOTlztogXfCE3Bf3GJlByxd797anqGtUajIdnb2WlHBn8gnTRxnKy57QIAmDMvpAONRjPuCNuUZT49ZMkKX5VKbfVzRHj667bRUBr4RUISc8ny1R3Wr1mRaerZunn7Lufk6V9dJJJiOo1GNfYI7F7x3dzZBWx2yzMv9aEC5X+ARFJEtbe3azJ9lCsUUU/HnOO9061LBZ/voYWanW6N4zj069u70Suavr17VqrVGtLLlFRGU6/P96x5TQR5m/UM7F4plcqoCxYv87sX/4Ct1mhe6dtXqdXEJctW+8tKSykrly3O3r1ja2rPwO6VuyJ+9rp15x7HfNn9B4+6j/roA2nEzrCUYUMGl56KOed6/eZtGwCAtPRM+poNWwROXK5m2+YNaZs3rM6gkMnYitXr/YuKS8j/y+1urwoKJZTy8gpKj3e6NXqMa64mcc/PB1xjL/zhPGHc6KLI3dtTvp45Xfz4yTPbdZvCvMyXu3Mv3gEAYOumdRmhi+fn5OQImWHhP3lCbSC0as1Ggd6gJ2zZuDojcvf2lL69e5Zv37nHOyMzi9aSbcFxHJatWueblp5pvfC7ucLI3dtTRrw/THr85Bn3E6diuLXbggMAHDhyzH1o0CD53p+2pUz8eEzRlWs3nU6ciuG68Jx1K5YuygYA+D5kXs6eXdvSGng/cACAo8eiXceODi6J2BmWwuU6aPb8fMBz776D7jOnT83fEbYxDcdxiNx/yMP0vHUbt3oXSopoa1cty9q/d1fymFHBJQePHPO4F/+g0bFozA3o368sLT3Tuqqquu4qo6qqmpSWlsEeOKBf2ZtooyH3Hz623rJtl6+/wFfxU/iW1EULvhUmp6Sx12zY7NWCpzeqVT07vKXVPf5Oo29S0WbrNlMoqNZoCL9f+MMhISnZZvqXn4vN55XK5dSRYye9AzVXJgSD0UgQ+PooFofMqytglZeVkwEAeM5OjQZKPJ6zDgBAJpNTAKBFxZltTY/nz9vM5+dZYOBrf34yMrOsTfuyvn17d6a4uvDaVJ3U4eGiNvN+T7vBf633e+LHY0uLS6TUP67ecFq/KUxgRSLhXl6eyt69elSMGz2ylMNhG2/dvmdTKpdTflqxJduUDv52zszCFwlJ7N8v/MEdGjSo0vR6QYMGyEe8V3PlPnf2DMmjJ3/a3r4bb/fesCEV585f4DIYdOOKpYvyyGQyDgCwPDRE9NmXM7teunzV/qvpU4ubWNV/RUhIiMuOHTt4LVl28uTJpdHR0RYF859++qnnqVOnHFry/IULFxaFh4db9GzHxcUxBg4c2OJeL5mslAwAwHV0aNV3RKvVEa7duM0dNnSwbML4MaVQ2+NdUlJCOXr8lHtRcQmZ5+ykBwBgMhiGBfPmFAAACHy9NbnCvOJTZ866VVcriCwWE9sRtimdwWBgHA7bCAAw/T9Tis6ev+Dy57MXbH8/gaa5dXmekMjMFYqYS79fkD1wQL+q2nWRpqSkWV++cp37+eSJUlPdTeA73SpMN614erhLn79IZN+Le2D/xWeTpBw22wAAwGazjfb2do2mx/v06lFu+ix/OOK90u07I7ynfzmkoHevHgoAgMGD+svPnb9Q91lYsyo0x4pkBXZ2tgYAgAnjx5T+ei6W9/TZC/agAe82myIcPjSo/OixaPd78Q84ph7S23fjbDAcJ7w/bEjZm2ijITFnzzu7uvDUSxbNz4fafafVasU1JR85NIGvT7P7piEojfU3RO4/xN9/4Iin6bFOrydy2Gz9lE8/KTCv1wEAsLW10W3ZsCYTAAAHHKSyUvKly1cd58xb1GnzhjWZAl9vjRWJhEPtFUNjMKyml5ZIJKCRrf8l3l6eytBFC4QNzXPiOrapQOdtRyAQ4LtvZhd+8fmk4nvxDzgvEpLYySlp7BPRMW6xFy47r/theVZGVjaDRqNipkDHRODro3zy9Jmt+bRuXTsrzB97urup8gsldKjpbWX4eHspTYEOAICdna2B6+igzRXlNdmTirSMqZcCq3eMOxF9hht92rK2cPXK0KxePQMt9leuUETTarXELgEdLaZ3DuikwHEcMjKzGDxnp0oAAH8/gcUyAh8vNYZhkCfOp3UO6KgSFxRST56K4eUXFDJ0Oj0RcBwwDIOqakWLzovp6ZlMqAlkLNrx9xMonr1ItFGpVHWZk86dLNeX7+mhvnbjVrOlDuZ8fb3rPt82nJoAyU8gqAs02Wy2Qa3W1PXCyOXl5MNHj7vkivKYGo2WBDgOOr2eqGjh9jlxHfV+At/qBw8f25iCnfsPHtl07OBXZUoF/t02GiIU5THf7dvHIv3cp1ePagKBAEkvU1go2PkXTJ44vnDwwP4VAAB6g56wdOVav65dAiqnTplcUn9ZEpGEm9JVUHs10rtnoGLG7HmdDh057vLjhh9yHWuvdgoKJFTzZc0VFZdQAACcuFxdoaSIMnvuggDTPD+BryI8bGNWQ89D3hwymYI1tn+Qf4atjY1xzMjgsjEjg8swDINbt+/Z/LR3H3/vvoPuri48DYPBeOWKmMViGtWavw7+AACc2pOECZVKxXRaHREAQK3WkJhMxivFzkwmw6hWq0n1pyOvz9mJqwMAkBQVUwGg2jR9ZPAH8v79+lYCAJTIZOSVqzf4GzHslZSlUqkkAQCw6901a6rlUCr/CjCsrVkWy9BqayM1Gg1RIikir98UJnBzdVGHLl6Q4+hgrycSiDBnXkgAtJBSVfOZ4LDZFu2wattVKJV168JiMS2WodNoRp1O/1plJKb1BwCA2h4jGo1aN838zVIoFMTV634UUChkbP43X4tcXJy1JCIJlv2wTvA6bQ7s36/82IlTbhqNlmAw6AnJqWmcWTP+k/cm26hPo9GSbt+Nc4iLf2AxTAyO41BRUdnqmKVVT2xLqaM3jUB4tcfEYDCA0Wh85Ytna2ujNz/pTZ0yuWBP5AGvJ0+fyU1di820BR7ubqocoYgJANC7Z2A1gUCAuAcPOXy+h7Sh5zx5+ozDYjENnQM6qnAch4idYammeVSzD35b1prUEdJ6r5s6aktUKhURCARg0P+6pZ9IJMLwYUEVickppXfuxjkKfL2VSqXqlWNZdbWCRKfTjPWnmT9WKJVWphMGnUEzKhTKV15HoVSR7Gxt22SPXXh4uKR+aul1REdH59VPbb2O10lhAQA4OjoY3Fxd1I+f/GkzsTYNBTVBqNGUTiJZkRrttWaxagKJSrM6EgCAysoqK6gXVCiVKotlqqprnsNg0I33Hz7maLVa4tofluU4cR31NcsriYYGjvONYTIZRgCAisoqkvkNJdXV1VYEAgFYLBam1ZYRa6ZZ9nSoNRoSlUr5x+64Skh8ySqvqCBv2bgmPfCdbnXlDpp6wX9zhg8dXH7o6HGP+w8fsTUaLREAYNiQweWtbaOh86tWq7UI+ug0mrFXz8DyzydPfCVtzLFht/ouWFSgXI81qy4qr9thmVnZ9KZSSyZjRgaX+Xh7KSIiozx1Ol2LvjQFhRK6rQ1HDwDg4GBv6NenV9lvsRd5JVLZKwWRojwx9dqNW9wPR7wnJZFIYGVlBXy+h9b0Z8pVI0h7IJOVWn08aWr3o8dONnhHoUwqo9pwOLoO/n5KrVZLTE1Lp5vPz8jKZnnz+RZ1bS9TUi0GhMzLy2e4ubqoAQB8vb2Vubkipl6vr/vuymSlVlKpjCoQ+LyV9XFt0djRH5UkJiVz7tyN5zQ0XyjKa7RA2NvLU0Oj0Ywvk1MsUkCJScksAoEAnTr41wVfmdnZTPNlMrNyGCQSCed7emhVtb0ytjacupPnpT+u2dX817IKgYCOHZQAAM+fJ1isS2paBovn7KQxD9BT09It1kUoyqO7uvAs0zEtOMe0lFKlJgIA2JhtX1z8Q7ZCobTCW7h9UDM+nKGjv1/V46fPOA8fPbHt1qVLhbU1C2ttGywm04hhGKjU6rrYIysn1+J76+3FVxaXlFDNz23u7q5ag9FI+Du35KNgpx6+p7uWTqcZb92Js9NqdYSi4hLyoSMnXGk0Wove5HlzZ4lLpDLakWPRFgdoDMcIUpnMyvSXmpZB37Jtl4c4v4AxfsyourTX/HlzxDYcjn5+SGiH2IuX7fLE+RShKI8acy7WYfHSVf7+foLqaVM/a7JQUqfT1bWlqf1QmR5X1bsiQl6fwWCw2Jfmf3J5GUoNvyGOjg6GoUMGy36LvcSLiIxyTU5JYxRKiiiJL5MZYeG73Z8nJNl8MmFc0dCgQRVcrqN2+649/OcvEplZ2Tm08F173CSSIvrH42q/W7Unkpu37jrcuHXXRijKo0ZERrmWyuWU4cOC5AAAE8aPlqo1GtLGLds9MzKzaIlJyYwNm7d5MRh046jgD5ocVR1puTGjguVDggbKtobv8oncf9AlNS2DXigpovz57AUrLHy3+49bd/h26uhfZR64mFAoFPzDEcOlt+7EOZyLvWgvEompV67dsP0t9iKvf78+clMtCQ4AVZXV5IjIKNfsHCHtzr14zpVrN7k9A7tXMBgMLKC25ufILyedxfkFlJhzsQ5Pn72wcXRw0OYK8xgyWWmz3+Pu3booBb4+ioNHjrvff/jYOidXSD145Lhz0ssUztjRH1kco5/++cI29sJle5FITD15+lfH5JQ0TtCggXKoScEZAACe/PmcnZKa1qIL6+Z06uivIhKJ+OmY35zyxPmUK9du2J6KOccT+PooCgokdImkqMV3Fw4c0K886WUK52VyKntI0IC6WprWtNGhg58SAOD3C5ftMQyDxKRkRlz8QzvzZT4eP7o4IzPbek9klEtOrpCanpFF37h5O39R6IoOZeUVrT5/oQNzPQwGA/tu7mzR4WMn3cZ9MqW7E9dR+/Ws6eIdu/Z4Ycbm451OHf3VQ4MGyc7/fpH3/rAhZaY0l1xeRvn8y1ndTMtZs1gGd3dX1fo1KzL6mKW87GxtjBE7t6ZHnz7LPR97yTnq4C9UEpGIOzs7aSZNHC/5eOyoUiKx6Rj1eUISa9WajX7m00xtD3i3r3z1ylBR694dBGrG52CZ70tzbGtrw9nTvyT+79eqffo+ZF6+j4+X6sbNOw637txzUKs1JA7bWu/p6aHatG5V3VhYWzetzYzYG+W2ev1mgcFgILq68NRLv1+Q3a9v72qoDVABAL6Z81XeoSMn3IQiEZPBYBimTpmcb7prRODro1m7amnm4V9OuC38fnknEomE+wl8q7duWpvR1F0yyOtbviREfPX6rao/rlx3vH7zjqNapSaxWCyDjzdfuWj+N7nDhg6uaGwE4a9nTpNYWVnhZ2LOueyLOkzmsNmGoMEDSufMmlE30KrRaCR8+P5waUVVldXi0JUddHo9oWuXgKpFC7/Ng5ohDZQTx4+RXL1+i3vh8hXnrp0DKpeHhggvXr5qf/rMOdfdkVHuM6dPbXLgVgCAjetWZu/eu989LHy3t0ajIXG5jtqvZ0575SeHpnw2sfD2nXi7/QePepDJZGxk8IjiTyaMlQEA+Hh7afv27ll25dpN7v2Hj+1OHo16+XffX3c3V93M6VPFZ86e58U/eGTvJ/BRLFuyUJicksbc8/MBzw2bt3vv/WlbRktea9jQweVRh37xoFDI2OCBA+rubGxNGz0DuyvHjx1VFHMulnf85Bk3gcBHMXP61ILlP6z3N9aeX/v361Mdunh+9snTv7pcvHzVmUKhYD4+XootG9dmNDT+XEsRmooiRSLRMT6f37G1L44gCIIg/69Mv69lPmAe8s8RiURpfD7/i4bmoTQWgiAIgiDtGgp2EARBEARp11DNDoIgCIL8A9zdXHXXL//21g790J6gnh0EQRAEQdo1FOwgCIIgCNKuNRns4Dj+VozIiyAIgiDI/7emYpYmgx2DwVCJNfD7JAiCIAiCIG0FhmEEg8HQ6C+tNxnsKJXKH8VisQwFPAiCIAiCtEUYhhHEYrFMqVRuamyZJgcVBABISEjgMZnM5VZWVmwCgYBqfBAEQRAEaRNwHMcMBkOVUqnc1L1796LGlms22EEQBEEQBHmboZ4aBEEQBEHaNRTsIAiCIAjSrqFgB0EQBEGQdg0FOwiCIAiCtGso2EEQBEEQpF37Ly7ewSFfTQQNAAAAAElFTkSuQmCC\n"
     },
     "metadata": {
      "bento_obj_id": "140087860416960",
      "needs_background": "light"
     }
    }
   ]
  },
  {
   "cell_type": "code",
   "metadata": {
    "originalKey": "81817f68-6383-4446-abc2-7ac698325684",
    "collapsed": false,
    "requestMsgId": "f058303d-23d4-4c3a-b5e5-0042b9f1cc05",
    "executionStartTime": 1674921757397,
    "executionStopTime": 1674921757407
   },
   "source": [
    ""
   ],
   "execution_count": 11,
   "outputs": []
  }
 ]
}
