{
  "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
      }
    },
    "toc": {
      "base_numbering": 1,
      "nav_menu": {},
      "number_sections": true,
      "sideBar": true,
      "skip_h1_title": false,
      "title_cell": "Table of Contents",
      "title_sidebar": "Contents",
      "toc_cell": false,
      "toc_position": {},
      "toc_section_display": true,
      "toc_window_display": false
    },
    "varInspector": {
      "cols": {
        "lenName": 16,
        "lenType": 16,
        "lenVar": 40
      },
      "kernels_config": {
        "python": {
          "delete_cmd_postfix": "",
          "delete_cmd_prefix": "del ",
          "library": "var_list.py",
          "varRefreshCmd": "print(var_dic_list())"
        },
        "r": {
          "delete_cmd_postfix": ") ",
          "delete_cmd_prefix": "rm(",
          "library": "var_list.r",
          "varRefreshCmd": "cat(var_dic_list()) "
        }
      },
      "position": {
        "height": "423.4px",
        "left": "858.8px",
        "right": "20px",
        "top": "120px",
        "width": "350px"
      },
      "types_to_exclude": [
        "module",
        "function",
        "builtin_function_or_method",
        "instance",
        "_Feature"
      ],
      "window_display": false
    },
    "captumWidgetMessage": {},
    "last_server_session_id": "7d34e865-49dc-488a-a0bc-0b0a171cbf3f",
    "last_kernel_id": "76f15f75-b06c-41b9-ab13-b7249a2b2e22",
    "last_base_url": "https://bento.edge.x2p.facebook.net/",
    "last_msg_id": "ca397f66-c4b80ded698f4d013e75775e_242",
    "outputWidgetContext": {}
  },
  "nbformat": 4,
  "nbformat_minor": 2,
  "cells": [
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "4be3ab76-0dde-4daa-81b7-68df50547590",
        "showInput": false
      },
      "source": [
        "# Simultaneous multi-objective multi-fidelity optimization  "
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "8fb7e5e3-8176-4b7f-b9b6-9a20bfb7ad37",
        "showInput": false
      },
      "source": [
        "In this tutorial notebook we demonstrate how to perform multi-objective multi-fidelity (MOMF) optimization in BoTorch as described in [1]. The main concept in MOMF is a \"fidelity objective\" that is optimized along with the problem objectives. This fidelity objective can be thought of as a trust objective that rewards the optimization when going to higher fidelity. This emulates a real-world scenario where high fidelity data may sometimes yield similar values for other objectives but is still considered more trustworthy. Thus the MOMF explicitly optimizes for getting more trustworthy data while taking into account the higher computational costs associated with it.\n",
        "\n",
        "We will optimize a synthetic function that is a modified multi-fidelity Branin-Currin. This is a 3 x 2 dimensional problem with one of the input dimension being the fidelity. For the MOMF, this results in a 3 x 3 optimization since it also takes the fidelity objective into account. In this case the fidelity objective is a linear function of fidelity, $ f(s)=s$, where $s$ is the fidelity. The MOMF algorithm can accept any discrete or continuous cost functions as an input. In this example, we choose an exponential dependency of the form $C(s)=\\exp(s)$. The goal of the optimization is to find the Pareto front, which is a trade-off solution set for Multi-objective problems, at the highest fidelity. \n",
        "\n",
        "In the second part of this tutorial, we compare the method with a multi-objective only optimization using qEHVI [2] with q set to 1 (note that MOMF also supports q>1 if the underlying MO acquisition function supports it). The MO-only optimization runs only at the highest fidelity while MOMF makes use of lower fidelity data to estimate the Pareto front at the highest fidelity.\n",
        "\n",
        "[1] [Irshad, Faran, Stefan Karsch, and Andreas Döpp. \"Expected hypervolume improvement for simultaneous multi-objective and multi-fidelity optimization.\" arXiv preprint arXiv:2112.13901 (2021).](https://arxiv.org/abs/2112.13901)\n",
        "\n",
        "[2] [S. Daulton, M. Balandat, and E. Bakshy. Differentiable Expected Hypervolume Improvement for Parallel Multi-Objective Bayesian Optimization. Advances in Neural Information Processing Systems 33, 2020.](https://arxiv.org/abs/2006.05078)"
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "b8c55b46-3934-4389-a48b-9a4a1d0f7008",
        "code_folding": [],
        "hidden_ranges": [],
        "collapsed": false,
        "requestMsgId": "d0d48506-6d24-4a4b-a27a-49366b07e0f2",
        "customOutput": null,
        "executionStartTime": 1677807565825,
        "executionStopTime": 1677807567435
      },
      "source": [
        "import os\n",
        "from typing import Any, Callable, Dict, Optional, Tuple\n",
        "\n",
        "import matplotlib.pyplot as plt\n",
        "import numpy as np\n",
        "import torch"
      ],
      "execution_count": 1,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "I0302 173926.276 _utils_internal.py:179] NCCL_DEBUG env var is set to None\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "I0302 173926.277 _utils_internal.py:197] NCCL_DEBUG is forced to WARN from None\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "313624cb-6e9b-4419-aa46-53bc5dfdbae6",
        "showInput": false
      },
      "source": [
        "### Set dtype and device \n",
        "Setting up the global variable that determine the device to run the optimization. The optimization is much faster when it runs on GPU."
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "9080711e-1447-4d2a-b2cd-7ebcc65d3821",
        "code_folding": [],
        "hidden_ranges": [],
        "collapsed": false,
        "requestMsgId": "89df896b-bc40-4885-93eb-e1c5a2a8d1a7",
        "customOutput": null,
        "executionStartTime": 1677807567557,
        "executionStopTime": 1677807567566
      },
      "source": [
        "tkwargs = {  # Tkwargs is a dictionary contaning data about data type and data device\n",
        "    \"dtype\": torch.double,\n",
        "    \"device\": torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\"),\n",
        "}\n",
        "SMOKE_TEST = os.environ.get(\"SMOKE_TEST\")"
      ],
      "execution_count": 2,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "6ab54ce1-7768-4460-b965-369fa1c4f664",
        "showInput": false
      },
      "source": [
        "### Define the problem and optimization settings"
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "eb357e02-e811-425c-a4f9-2c00d15c2b0f",
        "collapsed": false,
        "requestMsgId": "cc866d6c-1875-4d42-b4ff-114864b138ff",
        "customOutput": null,
        "executionStartTime": 1677807567751,
        "executionStopTime": 1677807568470,
        "code_folding": [],
        "hidden_ranges": []
      },
      "source": [
        "from botorch.test_functions.multi_objective_multi_fidelity import MOMFBraninCurrin\n",
        "\n",
        "BC = MOMFBraninCurrin(negate=True)\n",
        "dim_x = BC.dim  # Input Dimension for MO-only optimization\n",
        "dim_yMO = BC.num_objectives  # Output Dimension for MO-only optimization\n",
        "dim_xMO = dim_x - 1  # Input Dimension for MOMF optimization\n",
        "dim_y = dim_yMO + 1  # Output Dimesnion for MOMF optimization\n",
        "\n",
        "ref_pointMO = [0] * dim_xMO  # Reference point for MO only Hypervolume calculation\n",
        "ref_point = [0] * dim_x  # Reference point for MOMF Hypervolume calculation\n",
        "\n",
        "BATCH_SIZE = 2  # For batch optimization, BATCH_SIZE should be greater than 1\n",
        "n_BATCH = 5 if SMOKE_TEST else 30  # Number of iterations within one optimization loop\n",
        "n_INIT = 5  # Number of initial points for MOMF\n",
        "n_INITMO = 1  # Number of initial points for MO-only optimization\n",
        "# Number of Monte Carlo samples, used to \n",
        "MC_SAMPLES = 8 if SMOKE_TEST else 128\n",
        "# Number of restart points for multi-start optimization\n",
        "NUM_RESTARTS = 2 if SMOKE_TEST else 10\n",
        "# Number of raw samples for initial point selection heuristic\n",
        "RAW_SAMPLES = 4 if SMOKE_TEST else 512\n",
        "\n",
        "# Bounds for MO-only optimization\n",
        "standard_boundsMO = torch.tensor([[0.0] * dim_xMO, [1.0] * dim_xMO], **tkwargs)\n",
        "# Bounds for MOMF optimization\n",
        "standard_bounds = torch.tensor([[0.0] * dim_x, [1.0] * dim_x], **tkwargs)"
      ],
      "execution_count": 3,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "ccfef565-dd93-43f1-b2ae-590939bf3e23",
        "showInput": false
      },
      "source": [
        "### Problem Setup \n",
        "The problem as described before is a modified multi-fidelity version of Branin-Currin (BC) function that results in a 3 x 2 problem. A simple fidelity objective is also defined here which is a linear function of the input fidelity. We also design a wrapper function around the BC that takes care of interfacing torch with numpy and appends the fidelity objective with the BC functions.\n",
        ""
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "1287a6f8-a45e-4f29-b879-f6fc96451ae3",
        "collapsed": false,
        "requestMsgId": "fc4ea43c-236b-442f-94e6-e74b87bb2084",
        "customOutput": null,
        "executionStartTime": 1677807568632,
        "executionStopTime": 1677807568636
      },
      "source": [
        "def fid_obj(X: torch.Tensor) -> torch.Tensor:\n",
        "    \"\"\"\n",
        "    A Fidelity Objective that can be thought of as a trust objective.\n",
        "    Higher Fidelity simulations are rewarded as being more\n",
        "    trustworthy. Here we consider just a linear fidelity objective.\n",
        "    \"\"\"\n",
        "    fid_obj = 1 * X[..., -1]\n",
        "    return fid_obj\n",
        "\n",
        "\n",
        "def get_objective(x: torch.Tensor) -> torch.Tensor:\n",
        "    \"\"\"Wrapper around the Objective function to take care of fid_obj stacking\"\"\"\n",
        "    y = BC(x)  # The Branin-Currin is called\n",
        "    fid = fid_obj(x)  # Getting the fidelity objective values\n",
        "    fid_out = fid.unsqueeze(-1)\n",
        "    # Concatenating objective values with fid_objective\n",
        "    y_out = torch.cat([y, fid_out], -1)\n",
        "    return y_out"
      ],
      "execution_count": 4,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "6a44ae07-fe2e-49e1-9825-e4cc13bff750",
        "showInput": false
      },
      "source": [
        "### Helper functions to define Cost \n",
        "\n",
        "The cost_func function returns an exponential cost from the fidelity. The cost_callable is a wrapper around it that takes care of the input output shapes. This is given as a callable function to MOMF that internally divides the hypervolume by cost."
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "b4cddf01-d91f-44a9-86be-fbe1a0f5d25d",
        "collapsed": false,
        "requestMsgId": "7379c6f5-277b-49b1-a009-b39b8c3d8704",
        "customOutput": null,
        "executionStartTime": 1677807568776,
        "executionStopTime": 1677807568807
      },
      "source": [
        "def cost_func(x):\n",
        "    \"\"\"A simple exponential cost function.\"\"\"\n",
        "    exp_arg = torch.tensor(4, **tkwargs)\n",
        "    val = torch.exp(exp_arg * x)\n",
        "    return val\n",
        "\n",
        "\n",
        "# Displaying the min and max costs for this optimization\n",
        "print(f\"Min Cost: {cost_func(0)}\")\n",
        "print(f\"Max Cost: {cost_func(1)}\")\n",
        "\n",
        "\n",
        "def cost_callable(X: torch.Tensor) -> torch.Tensor:\n",
        "    r\"\"\"Wrapper for the cost function that takes care of shaping\n",
        "    input and output arrays for interfacing with cost_func.\n",
        "    This is passed as a callable function to MOMF.\n",
        "\n",
        "    Args:\n",
        "        X: A `batch_shape x q x d`-dim Tensor\n",
        "    Returns:\n",
        "        Cost `batch_shape x q x m`-dim Tensor of cost generated\n",
        "        from fidelity dimension using cost_func.\n",
        "    \"\"\"\n",
        "\n",
        "    cost = cost_func(torch.flatten(X)).reshape(X.shape)\n",
        "    cost = cost[..., [-1]]\n",
        "    return cost"
      ],
      "execution_count": 5,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Min Cost: 1.0\nMax Cost: 54.598150033144236\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "2f3f70a4-3654-4e65-ab6c-69b1e16ab3d6",
        "showInput": false
      },
      "source": [
        "### Model Initialization \n",
        "We use a multi-output SingleTaskGP to model the problem with a homoskedastic Gaussian likelihood with an inferred noise level. \n",
        "The model is initialized with 5 random points where the fidelity dimension of the initial points is sampled from a probability distribution : $p(s)=C(s)^{-1}$ "
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "e4c13423-3541-4786-ae34-f7182f739c4b",
        "collapsed": false,
        "requestMsgId": "bbf454ec-3c11-4a83-a34e-f0a0dbd740b0",
        "customOutput": null,
        "executionStartTime": 1677807568972,
        "executionStopTime": 1677807568982
      },
      "source": [
        "from botorch.models.gp_regression import SingleTaskGP\n",
        "from botorch.models.transforms.outcome import Standardize\n",
        "from gpytorch.mlls.exact_marginal_log_likelihood import ExactMarginalLogLikelihood\n",
        "\n",
        "\n",
        "def gen_init_data(dim_x: int, n_points: int) -> Tuple[torch.Tensor, torch.Tensor]:\n",
        "    \"\"\"\n",
        "    Generates training data with Fidelity dimension sampled\n",
        "    from a probability distribution that depends on Cost function\n",
        "    \"\"\"\n",
        "    train_x = torch.rand(n_points, dim_x, **tkwargs)\n",
        "    # Array from which fidelity values are sampled\n",
        "    fid_samples = torch.linspace(0, 1, 101, **tkwargs)\n",
        "    # Probability calculated from the Cost function\n",
        "    prob = 1 / cost_func(fid_samples)\n",
        "    # Normalizing\n",
        "    prob = prob / torch.sum(prob)\n",
        "    # Generating indices to choose fidelity samples\n",
        "    idx = prob.multinomial(num_samples=n_points, replacement=True)\n",
        "    train_x[:, -1] = fid_samples[idx]\n",
        "    # Calls the objective wrapper to generate train_obj\n",
        "    train_obj = get_objective(train_x)\n",
        "    return train_x, train_obj\n",
        "\n",
        "\n",
        "def initialize_model(\n",
        "    train_x: torch.Tensor, train_obj: torch.Tensor, state_dict: Optional[Dict[str, Any]]=None\n",
        ") -> Tuple[ExactMarginalLogLikelihood, SingleTaskGP]:\n",
        "    \"\"\"Initializes a SingleTaskGP with Matern 5/2 Kernel and returns the model and its MLL.\"\"\"\n",
        "    model = SingleTaskGP(\n",
        "        train_x, train_obj, outcome_transform=Standardize(m=train_obj.shape[-1])\n",
        "    )\n",
        "    mll = ExactMarginalLogLikelihood(model.likelihood, model)\n",
        "    if state_dict is not None:\n",
        "        model.load_state_dict(state_dict=state_dict)\n",
        "    return mll, model"
      ],
      "execution_count": 6,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "dcc1666f-f387-40b0-8137-25b133d22dd2",
        "showInput": false
      },
      "source": [
        "### Helper function to optimize acquisition function \n",
        "This is a helper function that initializes, optimizes the acquisition function MOMF and returns the new_x and new_obj. The problem is called from within this helper function.\n",
        "\n",
        "A simple initialization heuristic is used to select the 20 restart initial locations from a set of 1024 random points. Multi-start optimization of the acquisition function is performed using LBFGS-B with exact gradients computed via auto-differentiation."
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "2ab73c8f-09b9-43f2-9f20-3bf2d914dce7",
        "collapsed": false,
        "requestMsgId": "b290d7b7-4b1c-4520-b673-d9429da6adfb",
        "customOutput": null,
        "executionStartTime": 1677807569140,
        "executionStopTime": 1677807569178,
        "code_folding": [],
        "hidden_ranges": []
      },
      "source": [
        "from botorch.acquisition.multi_objective.multi_fidelity import MOMF\n",
        "from botorch.optim.optimize import optimize_acqf\n",
        "from botorch.sampling.normal import SobolQMCNormalSampler\n",
        "from botorch.utils.multi_objective.box_decompositions.non_dominated import (\n",
        "    FastNondominatedPartitioning,\n",
        ")\n",
        "from botorch.utils.transforms import unnormalize\n",
        "\n",
        "\n",
        "def optimize_MOMF_and_get_obs(\n",
        "    model: SingleTaskGP,\n",
        "    train_obj: torch.Tensor,\n",
        "    sampler: SobolQMCNormalSampler,\n",
        "    ref_point: torch.Tensor,\n",
        "    standard_bounds: torch.Tensor,\n",
        "    BATCH_SIZE: int,\n",
        "    cost_call: Callable[[torch.Tensor], torch.Tensor],\n",
        "):\n",
        "    \"\"\"\n",
        "    Wrapper to call MOMF and optimizes it in a sequential greedy\n",
        "    fashion returning a new candidate and evaluation\n",
        "    \"\"\"\n",
        "    partitioning = FastNondominatedPartitioning(\n",
        "        ref_point=torch.tensor(ref_point, **tkwargs), Y=train_obj\n",
        "    )\n",
        "    acq_func = MOMF(\n",
        "        model=model,\n",
        "        ref_point=ref_point,  # use known reference point\n",
        "        partitioning=partitioning,\n",
        "        sampler=sampler,\n",
        "        cost_call=cost_call,\n",
        "    )\n",
        "    # Optimization\n",
        "    candidates, _ = optimize_acqf(\n",
        "        acq_function=acq_func,\n",
        "        bounds=standard_bounds,\n",
        "        q=BATCH_SIZE,\n",
        "        num_restarts=NUM_RESTARTS,\n",
        "        raw_samples=RAW_SAMPLES,  # used for intialization heuristic\n",
        "        options={\"batch_limit\": 5, \"maxiter\": 200, \"nonnegative\": True},\n",
        "        sequential=True,\n",
        "    )\n",
        "    # observe new values\n",
        "    new_x = unnormalize(candidates.detach(), bounds=standard_bounds)\n",
        "    new_obj = get_objective(new_x)\n",
        "    return new_x, new_obj"
      ],
      "execution_count": 7,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "bdb86f2e-6774-4b00-af78-d081948812a5",
        "showInput": false
      },
      "source": [
        "### Running MOMF optimization \n",
        "\n",
        "We run 30 iterations to optimize the multi-fidelity versions of the Branin-Currin functions. The optimization loop works in the following sequence. \n",
        "\n",
        "1. At the start an initial data of 5 random points is generated and a model initialized using this data.\n",
        "2. The models are used to generate an acquisition function that is optimized to select new input parameters\n",
        "3. The objective function is evaluated at the suggested new_x and returns a new_obj.\n",
        "4. The models are updated with the new points and then are used again to make the next prediction.\n",
        ""
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "2ab80417-6e93-459f-91cd-846ded130e3f",
        "showInput": true,
        "customInput": null,
        "collapsed": false,
        "requestMsgId": "28cea5de-0493-4b66-8bfe-0d078b2e5fbc",
        "executionStartTime": 1677807569294,
        "executionStopTime": 1677807569302,
        "customOutput": null
      },
      "source": [
        "from botorch import fit_gpytorch_mll\n",
        "from tqdm import tqdm, trange"
      ],
      "execution_count": 8,
      "outputs": []
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "312928d7-97ec-4aa6-8fad-ab60a4d143db",
        "collapsed": false,
        "requestMsgId": "1473f416-f608-4c4e-9b9b-b60fb8ac48f1",
        "customOutput": null,
        "executionStartTime": 1677807569434,
        "executionStopTime": 1677807722319,
        "code_folding": [],
        "hidden_ranges": []
      },
      "source": [
        "%%time\n",
        "\n",
        "# Intializing train_x to zero\n",
        "train_x = torch.zeros(n_INIT + n_BATCH * BATCH_SIZE, dim_x, **tkwargs)\n",
        "# Intializing train_obj to zero\n",
        "train_obj = torch.zeros(n_INIT + n_BATCH * BATCH_SIZE, dim_y, **tkwargs)\n",
        "torch.manual_seed(0)\n",
        "train_x[:n_INIT, :], train_obj[:n_INIT, :] = gen_init_data(dim_x, n_INIT)\n",
        "\n",
        "# Generate Sampler\n",
        "momf_sampler = SobolQMCNormalSampler(sample_shape=torch.Size([MC_SAMPLES]))\n",
        "\n",
        "# run N_BATCH rounds of BayesOpt after the initial random batch\n",
        "\n",
        "for iteration in tqdm(range(0, n_BATCH)):\n",
        "    # Updating indices used to store new observations\n",
        "    lower_index = n_INIT + iteration * BATCH_SIZE\n",
        "    upper_index = n_INIT + iteration * BATCH_SIZE + BATCH_SIZE\n",
        "\n",
        "    # reinitialize the models so they are ready for fitting on next iteration\n",
        "    mll, model = initialize_model(train_x[:lower_index, :], train_obj[:lower_index, :])\n",
        "\n",
        "    fit_gpytorch_mll(mll=mll)  # Fit the model\n",
        "    \n",
        "    # optimize acquisition functions and get new observations\n",
        "    new_x, new_obj = optimize_MOMF_and_get_obs(\n",
        "        model=model,\n",
        "        train_obj=train_obj[:upper_index, :],\n",
        "        sampler=momf_sampler,\n",
        "        ref_point=ref_point,\n",
        "        standard_bounds=standard_bounds,\n",
        "        BATCH_SIZE=BATCH_SIZE,\n",
        "        cost_call=cost_callable\n",
        "    )\n",
        "    # Updating train_x and train_obj\n",
        "    train_x[lower_index:upper_index, :] = new_x\n",
        "    train_obj[lower_index:upper_index, :] = new_obj"
      ],
      "execution_count": 9,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r  0%|          | 0/30 [00:00<?, ?it/s]",
            "\r  3%|▎         | 1/30 [00:02<01:14,  2.58s/it]",
            "\r  7%|▋         | 2/30 [00:05<01:11,  2.56s/it]",
            "\r 10%|█         | 3/30 [00:08<01:14,  2.78s/it]",
            "\r 13%|█▎        | 4/30 [00:11<01:17,  2.96s/it]",
            "\r 17%|█▋        | 5/30 [00:17<01:38,  3.95s/it]",
            "\r 20%|██        | 6/30 [00:21<01:41,  4.24s/it]",
            "\r 23%|██▎       | 7/30 [00:27<01:47,  4.67s/it]",
            "\r 27%|██▋       | 8/30 [00:32<01:47,  4.87s/it]",
            "\r 30%|███       | 9/30 [00:38<01:49,  5.19s/it]",
            "\r 33%|███▎      | 10/30 [00:44<01:48,  5.42s/it]",
            "\r 37%|███▋      | 11/30 [00:50<01:47,  5.65s/it]",
            "\r 40%|████      | 12/30 [00:57<01:49,  6.11s/it]",
            "\r 43%|████▎     | 13/30 [01:03<01:41,  5.97s/it]",
            "\r 47%|████▋     | 14/30 [01:08<01:30,  5.64s/it]",
            "\r 50%|█████     | 15/30 [01:12<01:18,  5.26s/it]",
            "\r 53%|█████▎    | 16/30 [01:18<01:15,  5.36s/it]",
            "\r 57%|█████▋    | 17/30 [01:23<01:07,  5.17s/it]",
            "\r 60%|██████    | 18/30 [01:27<00:59,  4.95s/it]",
            "\r 63%|██████▎   | 19/30 [01:32<00:55,  5.02s/it]",
            "\r 67%|██████▋   | 20/30 [01:38<00:51,  5.11s/it]",
            "\r 70%|███████   | 21/30 [01:43<00:46,  5.20s/it]",
            "\r 73%|███████▎  | 22/30 [01:49<00:42,  5.37s/it]",
            "\r 77%|███████▋  | 23/30 [01:54<00:36,  5.22s/it]",
            "\r 80%|████████  | 24/30 [01:58<00:30,  5.04s/it]",
            "\r 83%|████████▎ | 25/30 [02:03<00:24,  4.93s/it]",
            "\r 87%|████████▋ | 26/30 [02:08<00:20,  5.12s/it]",
            "\r 90%|█████████ | 27/30 [02:15<00:16,  5.54s/it]",
            "\r 93%|█████████▎| 28/30 [02:21<00:11,  5.53s/it]",
            "\r 97%|█████████▋| 29/30 [02:26<00:05,  5.45s/it]",
            "\r100%|██████████| 30/30 [02:32<00:00,  5.78s/it]",
            "\r100%|██████████| 30/30 [02:32<00:00,  5.09s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "CPU times: user 1h 40min 17s, sys: 1min 34s, total: 1h 41min 52s\nWall time: 2min 32s\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "bc189d13-6ad9-4687-8e67-c506dcc1dad1",
        "showInput": false
      },
      "source": [
        "### Result:  Evaluating the Pareto front at the highest fidelity from MOMF\n",
        "\n",
        "After the optimization we are interested in evaluating the final Pareto front. For this we train a GP model with the data acquired by the MOMF optimization. After this we generate $10^4$ random test points between between [0,1] with the fidelity dimension set to 1 to approximate the Pareto front. Two helper functions are defined to achieve this objective where one function generates the test data and the other extracts the Pareto front at the highest fidelity for a given training and testing data.\n",
        "\n",
        "**Note: This works reasonably well only for lower dimensional search spaces**"
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "c8cf162e-9d0b-4fde-9486-3b949f3296ee",
        "collapsed": false,
        "requestMsgId": "9976fec7-0b59-41d9-a1b3-eeacdf4978eb",
        "customOutput": null,
        "executionStartTime": 1677807722448,
        "executionStopTime": 1677807722456
      },
      "source": [
        "from botorch.utils.multi_objective.pareto import is_non_dominated\n",
        "from botorch.utils.multi_objective.box_decompositions import DominatedPartitioning\n",
        "\n",
        "\n",
        "def gen_test_points(n_points: int, dim_x: int) -> torch.Tensor:\n",
        "    \"\"\"\n",
        "    Function to generate random points with fidelity dimension set to 1.\n",
        "    Used to evaluate Pareto front from MOMF\n",
        "    \"\"\"\n",
        "    test_x = torch.rand(size=(n_points, dim_x), **tkwargs)\n",
        "    test_x[:, -1] = 1\n",
        "    return test_x\n",
        "\n",
        "\n",
        "def get_pareto(\n",
        "    train_x: torch.Tensor, train_obj: torch.Tensor, test_x: torch.Tensor\n",
        ") -> Tuple[float, torch.Tensor]:\n",
        "    \"\"\"\n",
        "    Function that takes in training and testing data with a reference point.\n",
        "    \n",
        "    It computes the posterior mean at the testing points based on the model.\n",
        "    From these points, the non-dominated set is calculated and used to compute\n",
        "    the hypervolume.\n",
        "    \"\"\"\n",
        "    mll, model = initialize_model(train_x, train_obj)\n",
        "    fit_gpytorch_mll(mll)\n",
        "    with torch.no_grad():\n",
        "        # Compute posterior mean over outputs at testing data\n",
        "        means = model.posterior(test_x).mean\n",
        "    # Calculating Non-dominated points\n",
        "    pareto_mask = is_non_dominated(means)\n",
        "    pareto_front = means[pareto_mask]\n",
        "    # Computing Hypervolume\n",
        "    box_decomp = DominatedPartitioning(\n",
        "        torch.tensor(ref_pointMO,**tkwargs), pareto_front\n",
        "    )\n",
        "    hyper_volume = box_decomp.compute_hypervolume().item()\n",
        "    return hyper_volume, pareto_front"
      ],
      "execution_count": 10,
      "outputs": []
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "b908df5b-4183-4c3f-95c2-bfdf8739ce79",
        "collapsed": false,
        "requestMsgId": "414e876f-4de6-4df2-90fd-29b2d5ee8b65",
        "customOutput": null,
        "executionStartTime": 1677807722727,
        "executionStopTime": 1677807725035
      },
      "source": [
        "# Using the above two functions to generate the final Pareto front.\n",
        "n_points = 10**4\n",
        "test_x = gen_test_points(n_points, dim_x)\n",
        "\n",
        "hypervolume, final_PF = get_pareto(train_x, train_obj[:, :-1], test_x)\n",
        "hypervolume"
      ],
      "execution_count": 11,
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": "0.48020996821155426"
          },
          "metadata": {
            "bento_obj_id": "140557423574896"
          },
          "execution_count": 11
        }
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "ea24bf8b-a2cf-4a74-afe4-d1b09b6ab70f",
        "showInput": false
      },
      "source": [
        "Plotting the final Pareto front."
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "db102a2d-5072-4dea-95e3-8b9129095409",
        "collapsed": false,
        "requestMsgId": "725e3ee7-fe6a-49a1-a0c2-5d778b40faef",
        "customOutput": null,
        "executionStartTime": 1677807725344,
        "executionStopTime": 1677807725900
      },
      "source": [
        "fig, axes = plt.subplots(1, 1, figsize=(4, 4), dpi=200)\n",
        "axes.plot(\n",
        "    final_PF[:, 0].detach().cpu().numpy(),\n",
        "    final_PF[:, 1].detach().cpu().numpy(),\n",
        "    \"o\",\n",
        "    markersize=3.5,\n",
        "    label=\"MOMF\",\n",
        ")\n",
        "axes.set_title(\"Branin-Currin Pareto Front\", fontsize=\"12\")\n",
        "axes.set_xlabel(\"Branin\", fontsize=\"10\")\n",
        "axes.set_ylabel(\"Currin\", fontsize=\"10\")\n",
        "axes.set_xlim(0, 1)\n",
        "axes.set_ylim(0, 1)\n",
        "axes.tick_params(labelsize=10)\n",
        "axes.legend(loc=\"lower right\", fontsize=\"7\", frameon=True, ncol=1)\n",
        "plt.tight_layout()"
      ],
      "execution_count": 12,
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": "<Figure size 800x800 with 1 Axes>",
            "image/png": "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\n"
          },
          "metadata": {
            "bento_obj_id": "140557428341344",
            "needs_background": "light"
          }
        }
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "9333f719-5117-4e76-bd9b-6b1262c0da29",
        "showInput": false
      },
      "source": [
        "# Comparison of MOMF with single-fidelity multi-objective optimization using qEHVI \n",
        "\n",
        "In this section, we draw a comparison of the MOMF with qEHVI. This section parallels the last, again running 30 iterations and defining helper functions for qEHVI similar to those used for MOMF. \n",
        "\n",
        "**Note: Most of the material for qEHVI example has been taken from [3]**\n",
        "\n",
        "[3] [Constrained, Parallel, Multi-Objective BO in BoTorch with qNEHVI, and qParEGO](https://botorch.org/tutorials/constrained_multi_objective_bo)"
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "40e77498-2648-43a0-9fae-f0fb2dfd3ff6",
        "showInput": false
      },
      "source": [
        "Here we define a wrapper function for single-fidelity multi-objective qEHVI optimization that appends a column of ones (representing highest fidelity) around the Branin-Currin function."
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "1a8503dd-3376-4141-9615-a8c7d4b19267",
        "collapsed": false,
        "requestMsgId": "81ea55da-f905-4996-91e3-79bb910058de",
        "customOutput": null,
        "executionStartTime": 1677807726181,
        "executionStopTime": 1677807726184
      },
      "source": [
        "def get_obj_MO(X: torch.Tensor) -> torch.Tensor:\n",
        "    \"\"\"\n",
        "    Get the Branin-Currin objective at X.\n",
        "\n",
        "    Since MO-only optimization only evaluates at the highest fidelity,\n",
        "    a column of ones is added to the input to be consistent with the\n",
        "    objective function definition.\n",
        "    \"\"\"\n",
        "    h_fid = torch.ones(*X.shape[:-1], 1, **tkwargs)\n",
        "    X = torch.cat([X, h_fid], dim=-1)\n",
        "    y = BC(X)\n",
        "    return y"
      ],
      "execution_count": 13,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "8c1f9c30-cd00-4f50-ad94-d7e65530a10d",
        "showInput": false
      },
      "source": [
        "### Data Initialization qEHVI \n",
        "\n",
        "For qEHVI, we initialize with one point to keep the initial cost low. We do not aim to make the initial costs the same, but for all cases the initial costs of the MOMF are lower when the fidelity is drawn in a probabilistic fashion."
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "adafd8d7-29fd-4cca-b71a-4ade50983f5c",
        "collapsed": false,
        "requestMsgId": "5d2462fe-1ba1-4d92-8a92-59e2e59681e2",
        "customOutput": null,
        "executionStartTime": 1677807726289,
        "executionStopTime": 1677807726297
      },
      "source": [
        "def gen_init_data_MO(dim_x: int, points: int) -> Tuple[torch.Tensor, torch.Tensor]:\n",
        "    \"\"\"Generate random training data.\"\"\"\n",
        "    train_x = torch.rand(size=(points, dim_x), **tkwargs)\n",
        "    train_obj = get_obj_MO(train_x)\n",
        "    return train_x, train_obj"
      ],
      "execution_count": 14,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "bf63446e-85c6-45b7-a9a2-b6dcc5176ed3",
        "showInput": false
      },
      "source": [
        "### Helper function to optimize acquisition function \n",
        "This is a helper function that initializes and optimizes the acquisition function qEHVI and returns the new_x and new_obj. The problem is called from within this helper function.\n",
        "\n",
        "A simple initialization heuristic is used to select the 20 restart initial locations from a set of 1024 random points. Multi-start optimization of the acquisition function is performed using LBFGS-B with exact gradients computed via auto-differentiation."
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "f3d43939-a492-46d2-9d11-58f27f7dc86c",
        "collapsed": false,
        "requestMsgId": "056dfc13-9587-432e-9007-150dd5fe23a1",
        "code_folding": [],
        "hidden_ranges": [],
        "customOutput": null,
        "executionStartTime": 1677807726461,
        "executionStopTime": 1677807726477
      },
      "source": [
        "from botorch.acquisition.multi_objective.monte_carlo import (\n",
        "    qExpectedHypervolumeImprovement,\n",
        ")\n",
        "\n",
        "\n",
        "def optimize_MO_and_get_obs(\n",
        "    model: SingleTaskGP,\n",
        "    train_obj: torch.Tensor,\n",
        "    sampler: SobolQMCNormalSampler,\n",
        "    ref_point: torch.Tensor,\n",
        "    standard_bounds: torch.Tensor,\n",
        "    BATCH_SIZE_MO: int\n",
        ") -> Tuple[torch.Tensor, torch.Tensor]:\n",
        "    \"\"\"\n",
        "    Optimize the qEHVI acquisition function and return a new candidate and observation.\n",
        "    \"\"\"\n",
        "    # partition non-dominated space into disjoint rectangles\n",
        "    partitioning = FastNondominatedPartitioning(\n",
        "        ref_point=torch.tensor(ref_point, **tkwargs), Y=train_obj\n",
        "    )\n",
        "    acq_func = qExpectedHypervolumeImprovement(\n",
        "        model=model,\n",
        "        ref_point=ref_point,  # use a known reference point\n",
        "        partitioning=partitioning,\n",
        "        sampler=sampler,\n",
        "    )\n",
        "    # optimize\n",
        "    candidates, _ = optimize_acqf(\n",
        "        acq_function=acq_func,\n",
        "        bounds=standard_bounds,\n",
        "        q=BATCH_SIZE_MO,\n",
        "        num_restarts=NUM_RESTARTS,\n",
        "        raw_samples=RAW_SAMPLES,  # used for intialization heuristic\n",
        "        options={\"batch_limit\": 5, \"maxiter\": 200, \"nonnegative\": True},\n",
        "        sequential=True,\n",
        "    )\n",
        "    # observe new values\n",
        "    new_x = unnormalize(candidates.detach(), bounds=standard_bounds)\n",
        "    new_obj = get_obj_MO(new_x)\n",
        "    return new_x, new_obj"
      ],
      "execution_count": 15,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "d724499c-1b5a-4aed-a0eb-a3c714c43d53",
        "showInput": false
      },
      "source": [
        "### Running qEHVI optimization for multiple Trials\n",
        "\n",
        "We run 30 iterations to optimize the Branin-Currin functions using qEHVI. The Bayesian loop works almost the same as the one we ran early with MOMF:\n",
        "\n",
        "1. An initial data is generated and a model initialized using this data.\n",
        "2. The models are used to generated an acquisition function that gives us a suggestion for new input parameters\n",
        "3. The objective function is evaluated at the suggested new_x and returns a new_obj.\n",
        "4. The models are updated with the new points and then are used again to make the next prediction."
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "a164ba4e-94f4-45ac-aece-433790bd1429",
        "showInput": true,
        "customInput": null,
        "collapsed": false,
        "requestMsgId": "2f6b294a-808f-4d30-95ac-0f0d9ec2787e",
        "executionStartTime": 1677807726594,
        "executionStopTime": 1677807837327,
        "customOutput": null
      },
      "source": [
        "# Intializing train_x to zero\n",
        "train_xMO = torch.zeros(\n",
        "    n_INITMO + n_BATCH * BATCH_SIZE, dim_xMO, **tkwargs\n",
        ")\n",
        "# Intializing train_obj to zero\n",
        "train_objMO = (\n",
        "    torch.zeros(n_INITMO + n_BATCH * BATCH_SIZE, dim_yMO, **tkwargs)\n",
        ")\n",
        "torch.manual_seed(0)\n",
        "train_xMO[:n_INITMO, :], train_objMO[:n_INITMO, :] = gen_init_data_MO(\n",
        "    dim_xMO, n_INITMO\n",
        ")\n",
        "mll_MO, model_MO = initialize_model(\n",
        "    train_xMO[:n_INITMO, :], train_objMO[:n_INITMO, :]\n",
        ")\n",
        "\n",
        "# Generate Sampler\n",
        "sampler = SobolQMCNormalSampler(sample_shape=torch.Size([MC_SAMPLES]))\n",
        "\n",
        "for iteration in tqdm(range(0, n_BATCH)):\n",
        "    # run N_BATCH rounds of BayesOpt after the initial random batch\n",
        "    fit_gpytorch_mll(mll=mll_MO)  # Fit the model\n",
        "\n",
        "    # Updating indices used to store new observations\n",
        "    lower_index = n_INITMO + iteration * BATCH_SIZE\n",
        "    upper_index = n_INITMO + iteration * BATCH_SIZE + BATCH_SIZE\n",
        "    # optimize acquisition functions and get new observations\n",
        "    new_x, new_obj = optimize_MO_and_get_obs(\n",
        "        model=model_MO,\n",
        "        train_obj=train_objMO[:upper_index, :],\n",
        "        sampler=sampler,\n",
        "        ref_point=ref_pointMO,\n",
        "        standard_bounds=standard_boundsMO,\n",
        "        BATCH_SIZE_MO=BATCH_SIZE,\n",
        "    )\n",
        "    # Updating train_x and train_obj\n",
        "    train_xMO[lower_index:upper_index, :] = new_x\n",
        "    train_objMO[lower_index:upper_index, :] = new_obj\n",
        "    # reinitialize the models so they are ready for fitting on next iteration\n",
        "    mll_MO, model_MO = initialize_model(train_xMO[:upper_index, :], train_objMO[:upper_index, :])"
      ],
      "execution_count": 16,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r  0%|          | 0/30 [00:00<?, ?it/s]",
            "\r  3%|▎         | 1/30 [00:01<00:42,  1.46s/it]",
            "\r  7%|▋         | 2/30 [00:03<00:43,  1.55s/it]",
            "\r 10%|█         | 3/30 [00:04<00:45,  1.70s/it]",
            "\r 13%|█▎        | 4/30 [00:07<00:50,  1.96s/it]",
            "\r 17%|█▋        | 5/30 [00:09<00:54,  2.20s/it]",
            "\r 20%|██        | 6/30 [00:13<01:03,  2.65s/it]",
            "\r 23%|██▎       | 7/30 [00:17<01:12,  3.15s/it]",
            "\r 27%|██▋       | 8/30 [00:20<01:04,  2.91s/it]",
            "\r 30%|███       | 9/30 [00:23<01:05,  3.13s/it]",
            "\r 33%|███▎      | 10/30 [00:28<01:10,  3.52s/it]",
            "\r 37%|███▋      | 11/30 [00:33<01:16,  4.01s/it]",
            "\r 40%|████      | 12/30 [00:37<01:12,  4.00s/it]",
            "\r 43%|████▎     | 13/30 [00:41<01:10,  4.14s/it]",
            "\r 47%|████▋     | 14/30 [00:45<01:03,  3.94s/it]",
            "\r 50%|█████     | 15/30 [00:48<00:54,  3.66s/it]",
            "\r 53%|█████▎    | 16/30 [00:53<00:57,  4.13s/it]",
            "\r 57%|█████▋    | 17/30 [00:58<00:57,  4.44s/it]",
            "\r 60%|██████    | 18/30 [01:02<00:51,  4.26s/it]",
            "\r 63%|██████▎   | 19/30 [01:07<00:49,  4.52s/it]",
            "\r 67%|██████▋   | 20/30 [01:11<00:44,  4.43s/it]",
            "\r 70%|███████   | 21/30 [01:16<00:40,  4.47s/it]",
            "\r 73%|███████▎  | 22/30 [01:21<00:37,  4.63s/it]",
            "\r 77%|███████▋  | 23/30 [01:25<00:32,  4.58s/it]",
            "\r 80%|████████  | 24/30 [01:29<00:25,  4.27s/it]",
            "\r 83%|████████▎ | 25/30 [01:32<00:20,  4.01s/it]",
            "\r 87%|████████▋ | 26/30 [01:36<00:16,  4.02s/it]",
            "\r 90%|█████████ | 27/30 [01:39<00:11,  3.74s/it]",
            "\r 93%|█████████▎| 28/30 [01:43<00:07,  3.65s/it]",
            "\r 97%|█████████▋| 29/30 [01:46<00:03,  3.65s/it]",
            "\r100%|██████████| 30/30 [01:50<00:00,  3.71s/it]",
            "\r100%|██████████| 30/30 [01:50<00:00,  3.69s/it]",
            "\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "2fa1b7b7-293f-4692-9c98-8ec8f0f015ef",
        "showInput": false
      },
      "source": [
        "### Evaluating the hypervolume for each iteration\n",
        "\n",
        "As before, we are interested in the Pareto front at the highest fidelity after the MOMF. In this section we only show the evolution of the mean hypervolume for both MOMF and qEHVI. The calculation for the MOMF is done in a similar fashion as before, but now for each iteration."
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "3bfff3cf-2b08-4ea6-a79a-648caf262de3",
        "showInput": false
      },
      "source": [
        "The following code loops over the number of trials and the number of iterations within a single trial. It works in the following steps: \n",
        "1. At the start of each iteration a model is trained on the MOMF data.\n",
        "2. We generate a posterior mean from the model and calculated the dominated set from these 10000 points.\n",
        "3. This dominated set is used to estimate the hypervolume and then the next iteration is started where the model now takes n+1 data points\n",
        "\n",
        "**The test points used here are already generated before for evaluation of MOMF**\n",
        ""
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "920723cf-8abf-4d22-a404-0db53ee1ccf6",
        "collapsed": false,
        "requestMsgId": "fe41acaa-141f-457b-8183-4c01e3fd4c4b",
        "executionStopTime": 1677807921495,
        "customOutput": null,
        "executionStartTime": 1677807837917,
        "code_folding": [],
        "hidden_ranges": []
      },
      "source": [
        "%%time\n",
        "# Generating the final Pareto front by fitting a GP to MOMF data and evaluating the\n",
        "# GP posterior at the highest fidelity with 10000 random points\n",
        "\n",
        "# Array that contains hypervolume for n_TRIALS\n",
        "hv_MOMF = np.zeros(train_obj.shape[0])\n",
        "# Loop to get evolution of hypervolume during MOMF optimization.\n",
        "for i in trange(n_INIT, train_obj.shape[0]):\n",
        "    hv_MOMF[i], _ = get_pareto(train_x[:i, :], train_obj[:i, :-1], test_x)"
      ],
      "execution_count": 17,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r  0%|          | 0/60 [00:00<?, ?it/s]",
            "\r  2%|▏         | 1/60 [00:00<00:45,  1.29it/s]",
            "\r  3%|▎         | 2/60 [00:01<00:41,  1.40it/s]",
            "\r  5%|▌         | 3/60 [00:01<00:35,  1.60it/s]",
            "\r  7%|▋         | 4/60 [00:02<00:30,  1.84it/s]",
            "\r  8%|▊         | 5/60 [00:03<00:31,  1.75it/s]",
            "\r 10%|█         | 6/60 [00:03<00:31,  1.73it/s]",
            "\r 12%|█▏        | 7/60 [00:04<00:31,  1.66it/s]",
            "\r 13%|█▎        | 8/60 [00:04<00:24,  2.11it/s]",
            "\r 15%|█▌        | 9/60 [00:05<00:36,  1.41it/s]",
            "\r 17%|█▋        | 10/60 [00:06<00:39,  1.25it/s]",
            "\r 18%|█▊        | 11/60 [00:07<00:40,  1.20it/s]",
            "\r 20%|██        | 12/60 [00:08<00:48,  1.01s/it]",
            "\r 22%|██▏       | 13/60 [00:09<00:36,  1.27it/s]",
            "\r 23%|██▎       | 14/60 [00:10<00:45,  1.02it/s]",
            "\r 25%|██▌       | 15/60 [00:12<00:51,  1.14s/it]",
            "/mnt/xarfuse/uid-28646/3942ec4c-seed-nspid4026533215_cgpid2605254-ns-4026533212/linear_operator/utils/cholesky.py:40: NumericalWarning: A not p.d., added jitter of 1.0e-08 to the diagonal\n  warnings.warn(\n/mnt/xarfuse/uid-28646/3942ec4c-seed-nspid4026533215_cgpid2605254-ns-4026533212/linear_operator/utils/cholesky.py:40: NumericalWarning: A not p.d., added jitter of 1.0e-07 to the diagonal\n  warnings.warn(\n/mnt/xarfuse/uid-28646/3942ec4c-seed-nspid4026533215_cgpid2605254-ns-4026533212/linear_operator/utils/cholesky.py:40: NumericalWarning: A not p.d., added jitter of 1.0e-06 to the diagonal\n  warnings.warn(\n/mnt/xarfuse/uid-28646/3942ec4c-seed-nspid4026533215_cgpid2605254-ns-4026533212/linear_operator/utils/cholesky.py:40: NumericalWarning: A not p.d., added jitter of 1.0e-05 to the diagonal\n  warnings.warn(\n/mnt/xarfuse/uid-28646/3942ec4c-seed-nspid4026533215_cgpid2605254-ns-4026533212/linear_operator/utils/cholesky.py:40: NumericalWarning: A not p.d., added jitter of 1.0e-04 to the diagonal\n  warnings.warn(\n/mnt/xarfuse/uid-28646/3942ec4c-seed-nspid4026533215_cgpid2605254-ns-4026533212/linear_operator/utils/cholesky.py:40: NumericalWarning: A not p.d., added jitter of 1.0e-03 to the diagonal\n  warnings.warn(\n\r 27%|██▋       | 16/60 [00:13<00:51,  1.18s/it]",
            "\r 28%|██▊       | 17/60 [00:15<00:56,  1.32s/it]",
            "\r 30%|███       | 18/60 [00:16<00:56,  1.36s/it]",
            "\r 32%|███▏      | 19/60 [00:18<00:59,  1.44s/it]",
            "\r 33%|███▎      | 20/60 [00:20<01:05,  1.63s/it]",
            "\r 35%|███▌      | 21/60 [00:21<01:02,  1.59s/it]",
            "/mnt/xarfuse/uid-28646/3942ec4c-seed-nspid4026533215_cgpid2605254-ns-4026533212/linear_operator/utils/cholesky.py:40: NumericalWarning: A not p.d., added jitter of 1.0e-08 to the diagonal\n  warnings.warn(\n/mnt/xarfuse/uid-28646/3942ec4c-seed-nspid4026533215_cgpid2605254-ns-4026533212/linear_operator/utils/cholesky.py:40: NumericalWarning: A not p.d., added jitter of 1.0e-07 to the diagonal\n  warnings.warn(\n/mnt/xarfuse/uid-28646/3942ec4c-seed-nspid4026533215_cgpid2605254-ns-4026533212/linear_operator/utils/cholesky.py:40: NumericalWarning: A not p.d., added jitter of 1.0e-06 to the diagonal\n  warnings.warn(\n/mnt/xarfuse/uid-28646/3942ec4c-seed-nspid4026533215_cgpid2605254-ns-4026533212/linear_operator/utils/cholesky.py:40: NumericalWarning: A not p.d., added jitter of 1.0e-05 to the diagonal\n  warnings.warn(\n/mnt/xarfuse/uid-28646/3942ec4c-seed-nspid4026533215_cgpid2605254-ns-4026533212/linear_operator/utils/cholesky.py:40: NumericalWarning: A not p.d., added jitter of 1.0e-04 to the diagonal\n  warnings.warn(\n/mnt/xarfuse/uid-28646/3942ec4c-seed-nspid4026533215_cgpid2605254-ns-4026533212/linear_operator/utils/cholesky.py:40: NumericalWarning: A not p.d., added jitter of 1.0e-03 to the diagonal\n  warnings.warn(\n\r 37%|███▋      | 22/60 [00:23<01:06,  1.75s/it]",
            "\r 38%|███▊      | 23/60 [00:25<01:07,  1.81s/it]",
            "\r 40%|████      | 24/60 [00:26<00:57,  1.60s/it]",
            "\r 42%|████▏     | 25/60 [00:28<00:55,  1.58s/it]",
            "\r 43%|████▎     | 26/60 [00:30<00:56,  1.66s/it]",
            "\r 45%|████▌     | 27/60 [00:31<00:51,  1.57s/it]",
            "\r 47%|████▋     | 28/60 [00:33<00:49,  1.53s/it]",
            "\r 48%|████▊     | 29/60 [00:34<00:47,  1.52s/it]",
            "\r 50%|█████     | 30/60 [00:36<00:45,  1.50s/it]",
            "\r 52%|█████▏    | 31/60 [00:37<00:41,  1.44s/it]",
            "\r 53%|█████▎    | 32/60 [00:39<00:41,  1.49s/it]",
            "\r 55%|█████▌    | 33/60 [00:40<00:39,  1.48s/it]",
            "\r 57%|█████▋    | 34/60 [00:41<00:37,  1.44s/it]",
            "\r 58%|█████▊    | 35/60 [00:42<00:27,  1.08s/it]",
            "\r 60%|██████    | 36/60 [00:42<00:20,  1.20it/s]",
            "\r 62%|██████▏   | 37/60 [00:44<00:27,  1.18s/it]",
            "\r 63%|██████▎   | 38/60 [00:46<00:30,  1.39s/it]",
            "\r 65%|██████▌   | 39/60 [00:47<00:30,  1.47s/it]",
            "\r 67%|██████▋   | 40/60 [00:48<00:26,  1.34s/it]",
            "\r 68%|██████▊   | 41/60 [00:50<00:27,  1.46s/it]",
            "\r 70%|███████   | 42/60 [00:52<00:25,  1.44s/it]",
            "\r 72%|███████▏  | 43/60 [00:53<00:23,  1.39s/it]",
            "\r 73%|███████▎  | 44/60 [00:55<00:26,  1.66s/it]",
            "\r 75%|███████▌  | 45/60 [00:59<00:35,  2.36s/it]",
            "\r 77%|███████▋  | 46/60 [01:00<00:28,  2.06s/it]",
            "\r 78%|███████▊  | 47/60 [01:02<00:26,  2.02s/it]",
            "\r 80%|████████  | 48/60 [01:04<00:23,  1.95s/it]",
            "\r 82%|████████▏ | 49/60 [01:05<00:18,  1.67s/it]",
            "\r 83%|████████▎ | 50/60 [01:07<00:16,  1.63s/it]",
            "\r 85%|████████▌ | 51/60 [01:08<00:14,  1.66s/it]",
            "\r 87%|████████▋ | 52/60 [01:10<00:13,  1.65s/it]",
            "\r 88%|████████▊ | 53/60 [01:12<00:11,  1.61s/it]",
            "\r 90%|█████████ | 54/60 [01:14<00:10,  1.71s/it]",
            "\r 92%|█████████▏| 55/60 [01:16<00:09,  1.80s/it]",
            "\r 93%|█████████▎| 56/60 [01:17<00:06,  1.75s/it]",
            "\r 95%|█████████▌| 57/60 [01:19<00:05,  1.68s/it]",
            "\r 97%|█████████▋| 58/60 [01:20<00:03,  1.64s/it]",
            "\r 98%|█████████▊| 59/60 [01:22<00:01,  1.58s/it]",
            "\r100%|██████████| 60/60 [01:23<00:00,  1.52s/it]",
            "\r100%|██████████| 60/60 [01:23<00:00,  1.39s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "CPU times: user 54min 36s, sys: 59.1 s, total: 55min 35s\nWall time: 1min 23s\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "a7ea66e7-1000-4649-a109-23c05fa34186",
        "showInput": false
      },
      "source": [
        "For the qEHVI the hypervolume calculation is more straightforward. We calculated at each iteration the set of non-dominated points from the training data and use that to estimate the hypervolume."
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "44b79dfa-0f7b-47a9-bf4a-ad79de86581d",
        "collapsed": false,
        "requestMsgId": "bd220648-baad-4ce6-a362-b1f18f2dc41a",
        "executionStopTime": 1677807921819,
        "customOutput": null,
        "executionStartTime": 1677807921806
      },
      "source": [
        "# Since the MO data is already at highest fidelity,\n",
        "# we can calculate the evolution of hypervolume directly from data\n",
        "\n",
        "hv_MO = np.zeros(train_objMO.shape[0])\n",
        "# Loop to get evolution of hypervolume per iteration\n",
        "for i in range(train_objMO.shape[0]):\n",
        "    # Calculating Non-dominated points\n",
        "    pareto_maskMO = is_non_dominated(train_objMO[:i, :])\n",
        "    # Used to calculate hypervolume\n",
        "    box_decomp = DominatedPartitioning(\n",
        "        torch.tensor(ref_pointMO, **tkwargs),\n",
        "        train_objMO[:i, :][pareto_maskMO],\n",
        "    )\n",
        "    hv_MO[i] = box_decomp.compute_hypervolume().item()"
      ],
      "execution_count": 18,
      "outputs": []
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "a0e48588-bd5a-47d6-a882-1ab2c78214ce",
        "collapsed": false,
        "requestMsgId": "a45e5ea4-a81f-42b9-89b7-5270f5a0fa44",
        "executionStopTime": 1677807921995,
        "customOutput": null,
        "executionStartTime": 1677807921978
      },
      "source": [
        "# Cost calculation for both MOMF and MO\n",
        "cost_single = cost_func(train_x[:, -1])\n",
        "cost_MOMF = torch.cumsum(cost_single, axis=0)\n",
        "# Generating ones equal to number of iterations for MO only-optimization\n",
        "ones = torch.ones(train_objMO.shape[0], **tkwargs)\n",
        "cost_singleMO = cost_func(ones)\n",
        "cost_MO = torch.cumsum(cost_singleMO, axis=0)"
      ],
      "execution_count": 19,
      "outputs": []
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "92bd7261-4af0-446d-9860-66da9b65ec5c",
        "collapsed": false,
        "requestMsgId": "02e08476-08cd-484a-8e20-32b7e67a6435",
        "executionStopTime": 1677807922126,
        "customOutput": null,
        "executionStartTime": 1677807922120
      },
      "source": [
        "# Converting to Numpy arrays for plotting\n",
        "cost_MOMF = cost_MOMF.cpu().numpy()\n",
        "cost_MO = cost_MO.cpu().numpy()\n",
        "# The approximate max hypervolume taken by evaluating the BC function offline with random 50000 points\n",
        "true_hv = 0.5235514158034145"
      ],
      "execution_count": 20,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "241e1bf5-6c82-4fb1-9f89-0271646d7860",
        "showInput": false
      },
      "source": [
        "For plotting purposes we calculate the cost of both MOMF and MO using the last dimension of the training input data."
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "originalKey": "69e38b84-efab-4d37-8240-840dfe04b976",
        "showInput": false
      },
      "source": [
        "### Results\n",
        "\n",
        "The following plot shows, for each iteration and for both MOMF and qEHVI, the hypervolume (as a percentage of the true hypervolume) and the cost (shown on a log scale)."
      ]
    },
    {
      "cell_type": "code",
      "metadata": {
        "originalKey": "4586e4ae-80d1-407f-bdd4-e7fef2749b84",
        "showInput": true,
        "customInput": null,
        "customOutput": null,
        "collapsed": false,
        "requestMsgId": "c0311205-50c7-4b31-a580-fa8f7840e72b",
        "executionStopTime": 1677807922713,
        "executionStartTime": 1677807922247
      },
      "source": [
        "fig, ax = plt.subplots(1, 1, figsize=(4, 4), dpi=200)\n",
        "ax.plot(\n",
        "    cost_MOMF[n_INIT :],\n",
        "    hv_MOMF[n_INIT : ] / true_hv * 100,\n",
        "    label=\"MOMF\"\n",
        ")\n",
        "ax.plot(\n",
        "    cost_MO[n_INITMO:],\n",
        "    hv_MO[n_INITMO:] / true_hv * 100,\n",
        "    label=\"EHVI\"\n",
        ")\n",
        "\n",
        "ax.set_title(\"Branin-Currin\", fontsize=\"12\")\n",
        "ax.set_xlabel(\"Total Cost Log Scale\", fontsize=\"10\")\n",
        "ax.set_ylabel(\"Hypervolume (%)\", fontsize=\"10\")\n",
        "ax.set_ylim(0, 100)\n",
        "ax.tick_params(labelsize=10)\n",
        "ax.legend(loc=\"lower right\", fontsize=\"7\", frameon=True, ncol=1)\n",
        "plt.xscale(\"log\")\n",
        "plt.tight_layout()"
      ],
      "execution_count": 21,
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": "<Figure size 800x800 with 1 Axes>",
            "image/png": "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\n"
          },
          "metadata": {
            "bento_obj_id": "140557461788848",
            "needs_background": "light"
          }
        }
      ]
    }
  ]
}
