{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a6acb2e2",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import time\n",
    "import numpy as np\n",
    "import warnings\n",
    "import matplotlib.pyplot as plt\n",
    "from botorch.models import SingleTaskGP\n",
    "from botorch.fit import fit_gpytorch_mll\n",
    "from botorch.acquisition import AcquisitionFunction\n",
    "from botorch.optim import optimize_acqf\n",
    "from gpytorch.mlls import ExactMarginalLogLikelihood\n",
    "from scipy.stats import truncnorm\n",
    "from botorch.models.transforms.outcome import Standardize\n",
    "from pyDOE import *\n",
    "\n",
    "warnings.filterwarnings('ignore')\n",
    "\n",
    "def forrester_1d(y):\n",
    "    \"\"\"The standard 1D Forrester function.\"\"\"\n",
    "    return (6 * y - 2)**2 * np.sin(12 * y - 4)\n",
    "\n",
    "def objective(x):\n",
    "    \"\"\"\n",
    "    High-fidelity 2D Forrester-like objective function (corrected).\n",
    "    x is a NumPy array where each row is an input vector [x1, x2].\n",
    "    \"\"\"\n",
    "    x1 = x[:, 0]\n",
    "    x2 = x[:, 1]\n",
    "\n",
    "    A_hf = 0.08\n",
    "    B_hf = 0.9\n",
    "    C_hf = 2.0\n",
    "\n",
    "    # A simple way to get a 1D input for f1D based on x1 and x2\n",
    "    # We can use the L2 norm of x scaled by A_hf, or a scaled combination.\n",
    "    norm_x = np.sqrt(x1**2 + x2**2)\n",
    "    f1d_input = A_hf * norm_x\n",
    "\n",
    "    f1d_term = forrester_1d(f1d_input)\n",
    "    linear_term = B_hf * (x1 + x2) + C_hf\n",
    "\n",
    "    f_h = f1d_term + linear_term\n",
    "    return f_h\n",
    "\n",
    "def to_tensor(X):\n",
    "    return torch.tensor(X, dtype=torch.double)\n",
    "\n",
    "def compute_probability_of_minimum_monte_carlo(X_test, posterior, X_train, Y_train, num_samples=100, t=5):\n",
    "    min_indices = []\n",
    "    threshold = torch.min(Y_train)\n",
    "    mu = posterior.mean.squeeze()\n",
    "    sigma = torch.sqrt(posterior.variance.squeeze())\n",
    "    minimizer_locations = []\n",
    "    all_samples = []\n",
    "    for _ in range(num_samples):\n",
    "        threshold_np = threshold.item()\n",
    "        a = -np.inf * np.ones_like(mu.detach().numpy())\n",
    "        b = (threshold_np - mu) / sigma\n",
    "        samples = truncnorm.rvs(a=a, b=b.detach().numpy(), loc=mu.detach().numpy(), scale=sigma.detach().numpy())    \n",
    "        all_samples.append(samples)\n",
    "        minimizer_index = np.argmin(samples)\n",
    "        min_indices.append(minimizer_index.item())\n",
    "        minimizer_locations.append(X_test[minimizer_index].numpy())\n",
    "    p_min_mc = np.bincount(min_indices, minlength=len(X_test)) / num_samples\n",
    "    p_min_mc = torch.tensor(p_min_mc)\n",
    "    p_min_mc_next = torch.zeros(X_test.shape[0])\n",
    "    all_samples = np.stack(all_samples)  \n",
    "    for i in range(X_test.shape[0]):\n",
    "        p_min_mc_fantasy_max_sum = 0\n",
    "        for j in range(t):\n",
    "            threshold_fantasy = all_samples[j][i]\n",
    "            X_train_f = torch.cat([X_train, X_test[i:i+1]], dim=0)\n",
    "            f_sample = torch.tensor(all_samples[j][i]).reshape(-1, 1)\n",
    "            Y_train_f = torch.cat([Y_train, f_sample], dim=0)\n",
    "            gp_model_fantasy = SingleTaskGP(X_train_f, Y_train_f, outcome_transform=Standardize(m=1))\n",
    "            mll_fantasy = ExactMarginalLogLikelihood(gp_model_fantasy.likelihood, gp_model_fantasy)\n",
    "            fit_gpytorch_mll(mll_fantasy)\n",
    "            posterior_fantasy = gp_model_fantasy.posterior(X_test)   \n",
    "            \n",
    "            min_indices_fantasy = []\n",
    "            mu_fantasy = posterior_fantasy.mean.squeeze()\n",
    "            sigma_fantasy = torch.sqrt(posterior_fantasy.variance.squeeze())\n",
    "            minimizer_locations_fantasy = []\n",
    "            for _ in range(num_samples):\n",
    "                threshold_fantasy_np = threshold_fantasy .item()\n",
    "                a_fantasy = -np.inf * np.ones_like(mu_fantasy.detach().numpy())\n",
    "                b_fantasy = (threshold_fantasy_np - mu_fantasy) / sigma_fantasy\n",
    "                samples_fantasy = truncnorm.rvs(a=a_fantasy, b=b_fantasy.detach().numpy(), loc=mu_fantasy.detach().numpy(), scale=sigma_fantasy.detach().numpy())    \n",
    "                minimizer_index_fantasy = np.argmin(samples_fantasy)\n",
    "                min_indices_fantasy.append(minimizer_index_fantasy.item())\n",
    "                minimizer_locations_fantasy.append(X_test[minimizer_index_fantasy].numpy())\n",
    "            p_min_mc_fantasy = np.bincount(min_indices_fantasy, minlength=len(X_test)) / num_samples\n",
    "            p_min_mc_fantasy = torch.tensor(p_min_mc_fantasy)\n",
    "            p_min_mc_fantasy_max_sum += max(p_min_mc_fantasy)\n",
    "        p_min_mc_fantasy_max_mean = p_min_mc_fantasy_max_sum / t\n",
    "        p_min_mc_next[i] = p_min_mc_fantasy_max_mean      \n",
    "        #print(p_min_mc, p_min_mc_next)\n",
    "    return p_min_mc + (1 - p_min_mc) * p_min_mc_next\n",
    "\n",
    "class NRFSUR(AcquisitionFunction):\n",
    "    def __init__(self, model, X_train, Y_train, num_samples=5):\n",
    "        super().__init__(model)\n",
    "        self.X_train = X_train\n",
    "        self.Y_train = Y_train\n",
    "        self.num_samples = num_samples\n",
    "\n",
    "    def forward(self, X_test):\n",
    "        self.X_test = X_test\n",
    "        posterior = self.model.posterior(self.X_test)\n",
    "        prob_min = compute_probability_of_minimum_monte_carlo(self.X_test, posterior, self.X_train, self.Y_train)\n",
    "        return prob_min\n",
    "\n",
    "# Setup\n",
    "bounds = torch.tensor([[25, 40], [25, 40]])\n",
    "# Bounds\n",
    "lower_bounds = np.array([25, 25])\n",
    "upper_bounds = np.array([40, 40])\n",
    "\n",
    "# Normalize to [0, 1]\n",
    "def normalize(X):\n",
    "    return (X - lower_bounds) / (upper_bounds - lower_bounds)\n",
    "\n",
    "# Unnormalize back to original domain\n",
    "def unnormalize(X_norm):\n",
    "    return X_norm * (upper_bounds - lower_bounds) + lower_bounds\n",
    "\n",
    "\n",
    "num_iterations = 60\n",
    "num_trials = 20\n",
    "all_min_NRFSUR = []\n",
    "N_dim = 2\n",
    "N_test = 100\n",
    "for trial in range(2,3):\n",
    "    np.random.seed(trial)\n",
    "    torch.manual_seed(trial)\n",
    "\n",
    "    x1 = np.linspace(25, 40, 100)\n",
    "    x2 = np.linspace(25, 40, 100)\n",
    "    X1, X2 = np.meshgrid(x1, x2)\n",
    "    '''\n",
    "    X1_normalized = (X1 - np.min(x1)) / (np.max(x1) - np.min(x1))\n",
    "    X2_normalized = (X2 - np.min(x2)) / (np.max(x2) - np.min(x2))\n",
    "    '''\n",
    "    X_test_np = np.vstack([X1.ravel(), X2.ravel()]).T\n",
    "    X_test = to_tensor(X_test_np)\n",
    "\n",
    "    train_indices = np.random.choice(X_test_np.shape[0], size=5, replace=False)\n",
    "    X_train_np = X_test_np[train_indices]\n",
    "    Y_train_np = objective(to_tensor(X_train_np)).unsqueeze(-1)\n",
    "    X_train = to_tensor(X_train_np)\n",
    "    Y_train = Y_train_np.clone()\n",
    "    min_NRFSUR = [Y_train.min().item()]\n",
    "    \n",
    "    for iteration in range(num_iterations):\n",
    "        start = time.perf_counter()\n",
    "\n",
    "        X_test_acq_normalized_np=lhs(N_dim,N_test)\n",
    "        X_test_acq_normalized=torch.tensor(X_test_acq_normalized_np)\n",
    "        X_test_acq = unnormalize(X_test_acq_normalized)\n",
    "        \n",
    "        X_train_normalized = normalize(X_train)\n",
    "        gp_model = SingleTaskGP(X_train_normalized, Y_train, outcome_transform=Standardize(m=1))\n",
    "        mll = ExactMarginalLogLikelihood(gp_model.likelihood, gp_model)\n",
    "        fit_gpytorch_mll(mll)\n",
    "        posterior = gp_model.posterior(X_test_acq_normalized)\n",
    "        \n",
    "        nrfs_acquisition_func = NRFSUR(gp_model, X_train, Y_train)\n",
    "        acq_values_nrfs = nrfs_acquisition_func(X_test_acq_normalized)\n",
    "        acq_values_nrfs_norm = (acq_values_nrfs - torch.min(acq_values_nrfs)) / (torch.max(acq_values_nrfs) - torch.min(acq_values_nrfs))\n",
    "        candidates_nrfs = X_test_acq[torch.argmax(acq_values_nrfs)]\n",
    "        next_point = candidates_nrfs.unsqueeze(0)\n",
    "        next_value = objective(next_point).unsqueeze(-1)\n",
    "\n",
    "        # Plot current training data\n",
    "        plt.figure(figsize=(7, 5))\n",
    "        Z = objective(X_test).reshape(100, 100).detach().numpy()\n",
    "        plt.contourf(X1, X2, Z, levels=50, cmap='viridis')\n",
    "        plt.colorbar(label=\"Objective Value\")\n",
    "        plt.scatter(X_train[:, 0], X_train[:, 1], c='white', edgecolors='black', s=60, label='Previous Samples')\n",
    "        plt.scatter(next_point[0, 0], next_point[0, 1], c='red', edgecolors='black', s=100, marker='*', label='New Sample')\n",
    "        plt.title(f\"Trial {trial+1}, Iteration {iteration+1}\")\n",
    "        plt.xlabel('$x_1$')\n",
    "        plt.ylabel('$x_2$')\n",
    "        plt.legend()\n",
    "        plt.tight_layout()\n",
    "        plt.show()\n",
    "        \n",
    "        print(f\"Trial {trial+1}, Iter {iteration+1}/{num_iterations}\")\n",
    "        print(f\"Next point: {next_point.squeeze().tolist()} | Acquisition: {torch.max(acq_values_nrfs).item():.4f} | Obj: {next_value.item():.4f}\")\n",
    "\n",
    "        X_train = torch.cat([X_train, next_point], dim=0)\n",
    "        Y_train = torch.cat([Y_train, next_value], dim=0)\n",
    "        min_NRFSUR.append(Y_train.min().item())\n",
    "        \n",
    "        '''\n",
    "        plt.figure(figsize=(7, 5))\n",
    "        plt.contourf(X1_normalized, X2_normalized, posterior.mean.squeeze().reshape(100, 100).detach().numpy(), levels=50, cmap='viridis')\n",
    "        plt.colorbar(label=\"Objective Value\")\n",
    "        plt.show()\n",
    "        '''\n",
    "        \n",
    "        end = time.perf_counter()\n",
    "        print(f\"Iteration time: {end - start:.2f}s\\n\")\n",
    "\n",
    "    all_min_NRFSUR.append(min_NRFSUR)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b938671f",
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "import torch\n",
    "import time\n",
    "import numpy as np\n",
    "import warnings\n",
    "import matplotlib.pyplot as plt\n",
    "from botorch.models import SingleTaskGP\n",
    "from botorch.fit import fit_gpytorch_mll\n",
    "from botorch.acquisition import AcquisitionFunction\n",
    "from botorch.optim import optimize_acqf\n",
    "from gpytorch.mlls import ExactMarginalLogLikelihood\n",
    "from scipy.stats import truncnorm\n",
    "from botorch.models.transforms.outcome import Standardize\n",
    "from pyDOE import *\n",
    "\n",
    "warnings.filterwarnings('ignore')\n",
    "\n",
    "def forrester_1d(y):\n",
    "    \"\"\"The standard 1D Forrester function.\"\"\"\n",
    "    return (6 * y - 2)**2 * np.sin(12 * y - 4)\n",
    "\n",
    "def objective(x):\n",
    "    \"\"\"\n",
    "    High-fidelity 2D Forrester-like objective function (corrected).\n",
    "    x is a NumPy array where each row is an input vector [x1, x2].\n",
    "    \"\"\"\n",
    "    x1 = x[:, 0]\n",
    "    x2 = x[:, 1]\n",
    "\n",
    "    A_hf = 0.08\n",
    "    B_hf = 0.9\n",
    "    C_hf = 2.0\n",
    "\n",
    "    # A simple way to get a 1D input for f1D based on x1 and x2\n",
    "    # We can use the L2 norm of x scaled by A_hf, or a scaled combination.\n",
    "    norm_x = np.sqrt(x1**2 + x2**2)\n",
    "    f1d_input = A_hf * norm_x\n",
    "\n",
    "    f1d_term = forrester_1d(f1d_input)\n",
    "    linear_term = B_hf * (x1 + x2) + C_hf\n",
    "\n",
    "    f_h = f1d_term + linear_term\n",
    "    return f_h\n",
    "\n",
    "def to_tensor(X):\n",
    "    return torch.tensor(X, dtype=torch.double)\n",
    "\n",
    "def compute_probability_of_minimum_monte_carlo(X_test, posterior, X_train, Y_train, num_samples=100, t=5):\n",
    "    min_indices = []\n",
    "    threshold = torch.min(Y_train)\n",
    "    mu = posterior.mean.squeeze()\n",
    "    sigma = torch.sqrt(posterior.variance.squeeze())\n",
    "    minimizer_locations = []\n",
    "    all_samples = []\n",
    "    for _ in range(num_samples):\n",
    "        threshold_np = threshold.item()\n",
    "        a = -np.inf * np.ones_like(mu.detach().numpy())\n",
    "        b = (threshold_np - mu) / sigma\n",
    "        samples = truncnorm.rvs(a=a, b=b.detach().numpy(), loc=mu.detach().numpy(), scale=sigma.detach().numpy())    \n",
    "        all_samples.append(samples)\n",
    "        minimizer_index = np.argmin(samples)\n",
    "        min_indices.append(minimizer_index.item())\n",
    "        minimizer_locations.append(X_test[minimizer_index].numpy())\n",
    "    p_min_mc = np.bincount(min_indices, minlength=len(X_test)) / num_samples\n",
    "    p_min_mc = torch.tensor(p_min_mc)\n",
    "    p_min_mc_next = torch.zeros(X_test.shape[0])\n",
    "    all_samples = np.stack(all_samples)  \n",
    "    for i in range(X_test.shape[0]):\n",
    "        p_min_mc_fantasy_max_sum = 0\n",
    "        for j in range(t):\n",
    "            threshold_fantasy = all_samples[j][i]\n",
    "            X_train_f = torch.cat([X_train, X_test[i:i+1]], dim=0)\n",
    "            f_sample = torch.tensor(all_samples[j][i]).reshape(-1, 1)\n",
    "            Y_train_f = torch.cat([Y_train, f_sample], dim=0)\n",
    "            gp_model_fantasy = SingleTaskGP(X_train_f, Y_train_f, outcome_transform=Standardize(m=1))\n",
    "            mll_fantasy = ExactMarginalLogLikelihood(gp_model_fantasy.likelihood, gp_model_fantasy)\n",
    "            fit_gpytorch_mll(mll_fantasy)\n",
    "            posterior_fantasy = gp_model_fantasy.posterior(X_test)   \n",
    "            \n",
    "            min_indices_fantasy = []\n",
    "            mu_fantasy = posterior_fantasy.mean.squeeze()\n",
    "            sigma_fantasy = torch.sqrt(posterior_fantasy.variance.squeeze())\n",
    "            minimizer_locations_fantasy = []\n",
    "            for _ in range(num_samples):\n",
    "                threshold_fantasy_np = threshold_fantasy .item()\n",
    "                a_fantasy = -np.inf * np.ones_like(mu_fantasy.detach().numpy())\n",
    "                b_fantasy = (threshold_fantasy_np - mu_fantasy) / sigma_fantasy\n",
    "                samples_fantasy = truncnorm.rvs(a=a_fantasy, b=b_fantasy.detach().numpy(), loc=mu_fantasy.detach().numpy(), scale=sigma_fantasy.detach().numpy())    \n",
    "                minimizer_index_fantasy = np.argmin(samples_fantasy)\n",
    "                min_indices_fantasy.append(minimizer_index_fantasy.item())\n",
    "                minimizer_locations_fantasy.append(X_test[minimizer_index_fantasy].numpy())\n",
    "            p_min_mc_fantasy = np.bincount(min_indices_fantasy, minlength=len(X_test)) / num_samples\n",
    "            p_min_mc_fantasy = torch.tensor(p_min_mc_fantasy)\n",
    "            p_min_mc_fantasy_max_sum += max(p_min_mc_fantasy)\n",
    "        p_min_mc_fantasy_max_mean = p_min_mc_fantasy_max_sum / t\n",
    "        p_min_mc_next[i] = p_min_mc_fantasy_max_mean      \n",
    "        #print(p_min_mc, p_min_mc_next)\n",
    "    return p_min_mc + (1 - p_min_mc) * p_min_mc_next\n",
    "\n",
    "class NRFSUR(AcquisitionFunction):\n",
    "    def __init__(self, model, X_train, Y_train, num_samples=5):\n",
    "        super().__init__(model)\n",
    "        self.X_train = X_train\n",
    "        self.Y_train = Y_train\n",
    "        self.num_samples = num_samples\n",
    "\n",
    "    def forward(self, X_test):\n",
    "        self.X_test = X_test\n",
    "        posterior = self.model.posterior(self.X_test)\n",
    "        prob_min = compute_probability_of_minimum_monte_carlo(self.X_test, posterior, self.X_train, self.Y_train)\n",
    "        return prob_min\n",
    "\n",
    "# Setup\n",
    "bounds = torch.tensor([[25, 40], [25, 40]])\n",
    "# Bounds\n",
    "lower_bounds = np.array([25, 25])\n",
    "upper_bounds = np.array([40, 40])\n",
    "\n",
    "# Normalize to [0, 1]\n",
    "def normalize(X):\n",
    "    return (X - lower_bounds) / (upper_bounds - lower_bounds)\n",
    "\n",
    "# Unnormalize back to original domain\n",
    "def unnormalize(X_norm):\n",
    "    return X_norm * (upper_bounds - lower_bounds) + lower_bounds\n",
    "\n",
    "\n",
    "num_iterations = 60\n",
    "num_trials = 20\n",
    "all_min_NRFSUR = []\n",
    "N_dim = 2\n",
    "N_test = 100\n",
    "for trial in range(num_trials):\n",
    "    np.random.seed(trial)\n",
    "    torch.manual_seed(trial)\n",
    "\n",
    "    x1 = np.linspace(25, 40, 100)\n",
    "    x2 = np.linspace(25, 40, 100)\n",
    "    X1, X2 = np.meshgrid(x1, x2)\n",
    "    '''\n",
    "    X1_normalized = (X1 - np.min(x1)) / (np.max(x1) - np.min(x1))\n",
    "    X2_normalized = (X2 - np.min(x2)) / (np.max(x2) - np.min(x2))\n",
    "    '''\n",
    "    X_test_np = np.vstack([X1.ravel(), X2.ravel()]).T\n",
    "    X_test = to_tensor(X_test_np)\n",
    "\n",
    "    train_indices = np.random.choice(X_test_np.shape[0], size=5, replace=False)\n",
    "    X_train_np = X_test_np[train_indices]\n",
    "    Y_train_np = objective(to_tensor(X_train_np)).unsqueeze(-1)\n",
    "    X_train = to_tensor(X_train_np)\n",
    "    Y_train = Y_train_np.clone()\n",
    "    min_NRFSUR = [Y_train.min().item()]\n",
    "    \n",
    "    for iteration in range(num_iterations):\n",
    "        start = time.perf_counter()\n",
    "\n",
    "        X_test_acq_normalized_np=lhs(N_dim,N_test)\n",
    "        X_test_acq_normalized=torch.tensor(X_test_acq_normalized_np)\n",
    "        X_test_acq = unnormalize(X_test_acq_normalized)\n",
    "        \n",
    "        X_train_normalized = normalize(X_train)\n",
    "        gp_model = SingleTaskGP(X_train_normalized, Y_train, outcome_transform=Standardize(m=1))\n",
    "        mll = ExactMarginalLogLikelihood(gp_model.likelihood, gp_model)\n",
    "        fit_gpytorch_mll(mll)\n",
    "        posterior = gp_model.posterior(X_test_acq_normalized)\n",
    "        \n",
    "        nrfs_acquisition_func = NRFSUR(gp_model, X_train, Y_train)\n",
    "        acq_values_nrfs = nrfs_acquisition_func(X_test_acq_normalized)\n",
    "        acq_values_nrfs_norm = (acq_values_nrfs - torch.min(acq_values_nrfs)) / (torch.max(acq_values_nrfs) - torch.min(acq_values_nrfs))\n",
    "        candidates_nrfs = X_test_acq[torch.argmax(acq_values_nrfs)]\n",
    "        next_point = candidates_nrfs.unsqueeze(0)\n",
    "        next_value = objective(next_point).unsqueeze(-1)\n",
    "        \n",
    "        print(f\"Trial {trial+1}, Iter {iteration+1}/{num_iterations}\")\n",
    "        print(f\"Next point: {next_point.squeeze().tolist()} | Acquisition: {torch.max(acq_values_nrfs).item():.4f} | Obj: {next_value.item():.4f}\")\n",
    "\n",
    "        X_train = torch.cat([X_train, next_point], dim=0)\n",
    "        Y_train = torch.cat([Y_train, next_value], dim=0)\n",
    "        min_NRFSUR.append(Y_train.min().item())\n",
    "        \n",
    "        '''\n",
    "        plt.figure(figsize=(7, 5))\n",
    "        plt.contourf(X1_normalized, X2_normalized, posterior.mean.squeeze().reshape(100, 100).detach().numpy(), levels=50, cmap='viridis')\n",
    "        plt.colorbar(label=\"Objective Value\")\n",
    "        plt.show()\n",
    "        '''\n",
    "        \n",
    "        end = time.perf_counter()\n",
    "        print(f\"Iteration time: {end - start:.2f}s\\n\")\n",
    "\n",
    "    all_min_NRFSUR.append(min_NRFSUR)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9793c4c3",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.save(all_min_NRFSUR, 'all_min_NRFSUR')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3025e7ce",
   "metadata": {},
   "outputs": [],
   "source": [
    "NRFSUR_mean = np.array(all_min_NRFSUR).mean(axis=0)\n",
    "NRFSUR_std = np.array(all_min_NRFSUR).std(axis=0)\n",
    "NRFSUR_lower_bound = NRFSUR_mean - NRFSUR_std/np.sqrt(20)\n",
    "NRFSUR_upper_bound = NRFSUR_mean + NRFSUR_std/np.sqrt(20)\n",
    "x_vals = list(range(len(NRFSUR_mean)))\n",
    "plt.plot(x_vals, NRFSUR_mean, label=\"NRFS\", color='green', linewidth=5)\n",
    "plt.plot(x_vals, NRFSUR_mean, label=\"NRFS\", color='green', linewidth=5)\n",
    "plt.plot(x_vals, NRFSUR_lower_bound, color='green', linewidth=1, linestyle=':')\n",
    "plt.plot(x_vals, NRFSUR_upper_bound, color='green', linewidth=1, linestyle=':')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4de8d4b1",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.10.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
