{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "63570767",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "GPU is not available.\n"
     ]
    }
   ],
   "source": [
    "import torch\n",
    "import torch.nn as nn # All neural network modules, nn.Linear, nn.Conv2d, BatchNorm, Loss functions\n",
    "import torch.optim as optim # For all Optimization algorithms, SGD, Adam, etc.\n",
    "import torch.nn.functional as F # All functions that don't have any parameters\n",
    "from torch.autograd import Variable\n",
    "\n",
    "from sklearn.metrics.pairwise import rbf_kernel\n",
    "from sklearn.linear_model import LogisticRegression\n",
    "from sklearn.ensemble import RandomForestClassifier\n",
    "from sklearn.preprocessing import PolynomialFeatures\n",
    "from sklearn.linear_model import LinearRegression\n",
    "from sklearn.ensemble import RandomForestRegressor\n",
    "from sklearn.ensemble import GradientBoostingRegressor\n",
    "from quantile_forest import RandomForestQuantileRegressor\n",
    "\n",
    "\n",
    "import cvxpy as cp\n",
    "import numpy as np\n",
    "from numpy import linalg\n",
    "import pandas as pd\n",
    "\n",
    "from scipy.linalg import sqrtm\n",
    "\n",
    "\n",
    "from matplotlib import pyplot as plt\n",
    "\n",
    "import seaborn as sns\n",
    "import matplotlib.patches as patches\n",
    "import matplotlib.lines as lines\n",
    "\n",
    "import warnings\n",
    "random_seed = 42\n",
    "\n",
    "\n",
    "if torch.cuda.is_available():\n",
    "    print(\"GPU is available.\")\n",
    "    device = torch.device('cuda')\n",
    "else:\n",
    "    print(\"GPU is not available.\")\n",
    "    device = torch.device('cpu')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "6d50e812",
   "metadata": {},
   "outputs": [],
   "source": [
    "class NN1(nn.Module):\n",
    "    def __init__(self, input_size, output_size):\n",
    "        super(NN1, self).__init__()\n",
    "        self.fc1 = nn.Linear(input_size, 10)\n",
    "        self.fc2 = nn.Linear(10, output_size)\n",
    "\n",
    "    def forward(self, x):\n",
    "        x = F.relu(self.fc1(x))\n",
    "        x = self.fc2(x)\n",
    "        return x\n",
    "    \n",
    "# 2-layer NN\n",
    "class NN2(nn.Module):\n",
    "    def __init__(self, input_size, output_size):\n",
    "        super(NN2, self).__init__()\n",
    "        self.fc1 = nn.Linear(input_size, 50)\n",
    "        self.fc2 = nn.Linear(50, 50)\n",
    "        self.fc3 = nn.Linear(50, output_size)\n",
    "\n",
    "    def forward(self, x):\n",
    "        x = F.relu(self.fc1(x))\n",
    "        x = F.relu(self.fc2(x))\n",
    "        x = self.fc3(x)\n",
    "        return x "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "f72d6105",
   "metadata": {},
   "outputs": [],
   "source": [
    "class QuantileLoss(nn.Module):\n",
    "    def __init__(self, quantile):\n",
    "        super().__init__()\n",
    "        self.quantile = quantile\n",
    "\n",
    "    def forward(self, preds, target):\n",
    "        assert not target.requires_grad\n",
    "        assert preds.size(0) == target.size(0)\n",
    "        residuals = target - preds\n",
    "        return torch.max((self.quantile - 1) * residuals, self.quantile * residuals).mean()\n",
    "        \n",
    "        \n",
    "def est_quantile(est_type,quantile,X_pre,Y_pre,X_opt,X_adj,X_t):\n",
    "    \"\"\"\n",
    "        est_type: \n",
    "        \"NN1\": 1-layer NN;              \"NN2\": 2-layer NN; \n",
    "        \"qrf\": quantile regression forest;    \"gb\": gradient boosting\n",
    "        \n",
    "        quantile: the quantile we are estimating\n",
    "        (X_pre,Y_pre): training data\n",
    "        X_opt,X_adj,X_t: data used to predict\n",
    "        output: quantile estimator Q and the prediction Q(X) \n",
    "    \"\"\"\n",
    "    \n",
    "    if est_type == \"NN1\":\n",
    "        torch.manual_seed(42)\n",
    "        torch.cuda.manual_seed_all(42)\n",
    "        \n",
    "        # Convert numpy arrays to torch tensors\n",
    "        X = torch.from_numpy(X_pre).float().to(device)\n",
    "        Y = torch.from_numpy(Y_pre).float().to(device)\n",
    "        \n",
    "        model = NN1(input_size=5, output_size=1).to(device)\n",
    "        learning_rate = 0.001\n",
    "        \n",
    "        # Set loss function and optimizer\n",
    "        criterion = QuantileLoss(quantile).to(device)\n",
    "        optimizer = torch.optim.Adam(model.parameters(), lr=0.001)\n",
    "        \n",
    "        # Training loop\n",
    "        for epoch in range(1000):\n",
    "            optimizer.zero_grad()   # zero the gradient buffers\n",
    "            output = model(X)\n",
    "            loss = criterion(output, Y)\n",
    "            loss.backward()\n",
    "            optimizer.step()    # Does the update\n",
    "            \n",
    "        # Predict\n",
    "        X_opt = torch.from_numpy(X_opt).float().to(device)\n",
    "        X_adj = torch.from_numpy(X_adj).float().to(device)\n",
    "        X_t = torch.from_numpy(X_t).float().to(device)\n",
    "\n",
    "        Q_opt = model(X_opt)\n",
    "        Q_opt = Q_opt.detach().cpu().numpy().reshape(-1,1)\n",
    "        Q_adj = model(X_adj)\n",
    "        Q_adj = Q_adj.detach().cpu().numpy().reshape(-1,1)\n",
    "        Q_t = model(X_t)\n",
    "        Q_t = Q_t.detach().cpu().numpy().reshape(-1,1)\n",
    "        return model, Q_opt, Q_adj, Q_t\n",
    "    \n",
    "    if est_type == \"NN2\":\n",
    "        torch.manual_seed(42)\n",
    "        torch.cuda.manual_seed_all(42) \n",
    "        \n",
    "        # Convert numpy arrays to torch tensors\n",
    "        X = torch.from_numpy(X_pre).float().to(device)\n",
    "        Y = torch.from_numpy(Y_pre).float().to(device)\n",
    "        \n",
    "        model = NN2(input_size=5, output_size=1).to(device)\n",
    "        learning_rate = 0.001\n",
    "        \n",
    "        # Set loss function and optimizer\n",
    "        criterion = QuantileLoss(quantile).to(device)\n",
    "        optimizer = torch.optim.Adam(model.parameters(), lr=0.001)\n",
    "        \n",
    "        # Training loop\n",
    "        for epoch in range(1000):\n",
    "            optimizer.zero_grad()   # zero the gradient buffers\n",
    "            output = model(X)\n",
    "            loss = criterion(output, Y)\n",
    "            loss.backward()\n",
    "            optimizer.step()    # Does the update\n",
    "            \n",
    "        # Predict\n",
    "        X_opt = torch.from_numpy(X_opt).float().to(device)\n",
    "        X_adj = torch.from_numpy(X_adj).float().to(device)\n",
    "        X_t = torch.from_numpy(X_t).float().to(device)\n",
    "\n",
    "        Q_opt = model(X_opt)\n",
    "        Q_opt = Q_opt.detach().cpu().numpy().reshape(-1,1)\n",
    "        Q_adj = model(X_adj)\n",
    "        Q_adj = Q_adj.detach().cpu().numpy().reshape(-1,1)\n",
    "        Q_t = model(X_t)\n",
    "        Q_t = Q_t.detach().cpu().numpy().reshape(-1,1)\n",
    "        return model, Q_opt, Q_adj, Q_t\n",
    "    \n",
    "    if est_type == \"qrf\":\n",
    "        model = RandomForestQuantileRegressor(n_estimators = 500, random_state=random_seed)\n",
    "        model.fit(X_pre, Y_pre)\n",
    "        Q_opt = model.predict(X_opt,quantiles = [quantile]).reshape(-1,1)\n",
    "        Q_adj = model.predict(X_adj,quantiles = [quantile]).reshape(-1,1)\n",
    "        Q_t = model.predict(X_t,quantiles = [quantile]).reshape(-1,1)\n",
    "        return model, Q_opt, Q_adj, Q_t\n",
    "    \n",
    "    \n",
    "    if est_type == \"gb\":\n",
    "        model = GradientBoostingRegressor(n_estimators=300,random_state=random_seed,loss = \"quantile\", alpha = quantile)\n",
    "        model.fit(X_pre, Y_pre)\n",
    "        Q_opt = model.predict(X_opt).reshape(-1,1)\n",
    "        Q_adj = model.predict(X_adj).reshape(-1,1)\n",
    "        Q_t = model.predict(X_t).reshape(-1,1)\n",
    "        return model, Q_opt, Q_adj, Q_t\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "2bc6dfaf",
   "metadata": {},
   "outputs": [],
   "source": [
    "data = pd.read_csv('airfoil_self_noise.csv', header = None,sep='\\t')\n",
    "Data = data.to_numpy()\n",
    "\n",
    "Data[:,0] = np.log(Data[:,0]) # log transform\n",
    "Data[:,4] = np.log(Data[:,4]) # log transform\n",
    "\n",
    "beta = np.array([-1, 0, 0, 0, 1])\n",
    "\n",
    "def mydata(Data, beta, prop = 0.75):\n",
    "    \n",
    "    # random split the data from source into two parts: training and testing data\n",
    "    np.random.shuffle(Data)\n",
    "    train_idx = int(Data.shape[0] * prop)\n",
    "    Dtrain, Dtest = np.split(Data, [train_idx])\n",
    "    \n",
    "    # creat the shift data set\n",
    "    probabilities = np.exp(Dtest[:,0:5].dot(beta)) # Calculate probabilities proportional to w(x) = exp(x^T * beta)\n",
    "    probabilities /= np.sum(probabilities) # Normalize probabilities\n",
    "    \n",
    "    sample_indices = np.random.choice(len(Dtest), size=len(Dtest), replace=True, p=probabilities)\n",
    "    Dshift = Dtest[sample_indices]\n",
    "    \n",
    "    return Dtrain, Dtest, Dshift"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0e3fed2a",
   "metadata": {},
   "outputs": [],
   "source": [
    "mc_iteration = 200\n",
    "bandwidth = np.zeros(mc_iteration)\n",
    "coverage = np.zeros(mc_iteration)\n",
    "\n",
    "for j in range(mc_iteration):\n",
    "    Dtrain, Dtest, Dshift = mydata(Data, beta)\n",
    "\n",
    "    Y_train = Dtrain[:,5]\n",
    "    X_train = Dtrain[:,0:5]\n",
    "\n",
    "    X_test = Dshift[:,0:5]\n",
    "    Y_test = Dshift[:,5]\n",
    "\n",
    "\n",
    "    pre_idx = int(Dtrain.shape[0] * 0.5)\n",
    "    Dtrain_pre, Dtrain_opt = np.split(Dtrain, [pre_idx])\n",
    "    adj_idx = int(Dtrain_opt.shape[0] * 0.5)\n",
    "    Dtrain_opt, Dtrain_adj = np.split(Dtrain_opt, [adj_idx])\n",
    "\n",
    "    X_pre = Dtrain_pre[:,0:5]\n",
    "    Y_pre = Dtrain_pre[:,5]\n",
    "\n",
    "    X_opt = Dtrain_opt[:,0:5]\n",
    "    Y_opt = Dtrain_opt[:,5]\n",
    "\n",
    "    X_adj = Dtrain_adj[:,0:5]\n",
    "    Y_adj = Dtrain_adj[:,5]\n",
    "    \n",
    "    mean_model = RandomForestRegressor(n_estimators = 1000, random_state = 42)\n",
    "    mean_model.fit(X_pre, Y_pre)\n",
    "\n",
    "    mean_est_pre = mean_model.predict(X_pre)\n",
    "    mean_est_opt = mean_model.predict(X_opt)\n",
    "    mean_est_adj = mean_model.predict(X_adj)\n",
    "\n",
    "    Y_centered_squared_pre = (Y_pre - mean_est_pre)**2 \n",
    "    Y_centered_squared_opt = (Y_opt - mean_est_opt)**2 \n",
    "    Y_centered_squared_adj = (Y_adj - mean_est_adj)**2 \n",
    "    \n",
    "    quantile = [0.95,0.95,0.95,0.95]\n",
    "\n",
    "\n",
    "    m1,Q1_opt,Q1_adj,Q1_t = est_quantile(\"NN1\",quantile[0],X_pre,Y_centered_squared_pre,X_opt,X_adj,X_test) # quantile def\n",
    "    m2,Q2_opt,Q2_adj,Q2_t = est_quantile(\"NN2\",quantile[1],X_pre,Y_centered_squared_pre,X_opt,X_adj,X_test)\n",
    "    m3,Q3_opt,Q3_adj,Q3_t = est_quantile(\"qrf\",quantile[2],X_pre,Y_centered_squared_pre,X_opt,X_adj,X_test)\n",
    "    m4,Q4_opt,Q4_adj,Q4_t = est_quantile(\"gb\",quantile[3],X_pre,Y_centered_squared_pre,X_opt,X_adj,X_test)\n",
    "    \n",
    "\n",
    "    ######## Variance estimator ###########\n",
    "    \n",
    "    rf_var_model = RandomForestRegressor(n_estimators = 1000, random_state = 42)\n",
    "    rf_var_model.fit(X_pre, Y_centered_squared_pre)\n",
    "    var_hat_pre = rf_var_model.predict(X_pre)\n",
    "    var_hat_adj = rf_var_model.predict(X_adj)\n",
    "    var_hat_opt = rf_var_model.predict(X_opt)\n",
    "    var_hat_test = rf_var_model.predict(X_test)\n",
    "    \n",
    "    E_opt = np.vstack((np.matrix.flatten(Q1_opt), np.matrix.flatten(Q2_opt), np.matrix.flatten(Q3_opt), np.matrix.flatten(Q4_opt), var_hat_opt))\n",
    "    E_adj = np.vstack((np.matrix.flatten(Q1_adj), np.matrix.flatten(Q2_adj), np.matrix.flatten(Q3_adj), np.matrix.flatten(Q4_adj), var_hat_adj))\n",
    "    E_test = np.vstack((np.matrix.flatten(Q1_t), np.matrix.flatten(Q2_t), np.matrix.flatten(Q3_t), np.matrix.flatten(Q4_t), var_hat_test))\n",
    "    \n",
    "    n_opt = X_opt.shape[0]\n",
    "    n_adj = X_adj.shape[0]\n",
    "    n_test = X_test.shape[0]\n",
    "\n",
    "    cons_opt = np.ones(n_opt).reshape(1,-1)\n",
    "    cons_adj = np.ones(n_adj).reshape(1,-1)\n",
    "    cons_test = np.ones(n_test).reshape(1,-1)\n",
    "\n",
    "    A_opt = np.vstack((E_opt,cons_opt))\n",
    "    A_adj = np.vstack((E_adj,cons_adj))\n",
    "    A_test = np.vstack((E_test,cons_test))\n",
    "\n",
    "    weight = cp.Variable(A_opt.shape[0])\n",
    "\n",
    "\n",
    "    constraints = [weight>=0]+[weight @ A_opt >= Y_centered_squared_opt]  ### Change the objective \n",
    "    prob = cp.Problem(cp.Minimize(cp.sum(weight @ A_test)), constraints)\n",
    "    prob.solve()\n",
    "    optimal_weight = weight.value\n",
    "\n",
    "    f_hat_init_opt = optimal_weight @  A_opt\n",
    "    f_hat_init_adj = optimal_weight @  A_adj\n",
    "    f_hat_init_test = optimal_weight @ A_test\n",
    "    \n",
    "    domain_cov = np.vstack((X_pre, X_test))\n",
    "    domain_idx =  np.vstack((np.zeros(X_pre.shape[0]).reshape(-1,1), np.ones(X_test.shape[0]).reshape(-1,1)))\n",
    "    domain_idx = np.matrix.flatten(domain_idx)\n",
    "\n",
    "    logistic_model = LogisticRegression(penalty='l2', solver='lbfgs', max_iter=1000,random_state=0) \n",
    "    logistic_model.fit(domain_cov, domain_idx)\n",
    "\n",
    "    prob_hat = logistic_model.predict_proba(X_adj)[:,0]\n",
    "    prob_hat = np.clip(prob_hat, 0.01, 0.99)\n",
    "    est_ratio = prob_hat/(1-prob_hat)\n",
    "    \n",
    "    delta = 1.01\n",
    "    alpha = 0.05\n",
    "    stepsize = 0.005\n",
    "    prop_outside = np.mean(est_ratio * ((Y_centered_squared_adj/f_hat_init_adj) > delta))\n",
    "    while prop_outside <= alpha:\n",
    "        delta = delta - stepsize\n",
    "        prop_outside = np.mean(est_ratio * ((Y_centered_squared_adj/f_hat_init_adj) > delta))\n",
    "        \n",
    "    mean_est_test = mean_model.predict(X_test)\n",
    "    Y_centered_squared_test = (Y_test - mean_est_test)**2 \n",
    "    coverage[j] = np.mean(Y_centered_squared_test < (delta * f_hat_init_test))\n",
    "    bandwidth[j] = 2 * np.mean((delta * f_hat_init_test) ** 0.5)\n",
    "    print(j, \" th iteration is done!\")\n",
    "    print(\"The value of delta is\",delta)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8140da4b",
   "metadata": {},
   "outputs": [],
   "source": [
    "sns.histplot(np.array(coverage).reshape(-1, 1), kde = True, bins = 30)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2c39858d",
   "metadata": {},
   "outputs": [],
   "source": [
    "sns.histplot(np.array(bandwidth).reshape(-1, 1), kde = True,bins = 30)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "b3add921",
   "metadata": {},
   "outputs": [],
   "source": [
    "def run_utopia(mc_iter):\n",
    "    Dtrain, Dtest, Dshift = mydata(Data, beta)\n",
    "    Y_train = Dtrain[:,5]\n",
    "    X_train = Dtrain[:,0:5]\n",
    "\n",
    "    X_test = Dshift[:,0:5]\n",
    "    Y_test = Dshift[:,5]\n",
    "\n",
    "\n",
    "    pre_idx = int(Dtrain.shape[0] * 0.5)\n",
    "    Dtrain_pre, Dtrain_opt = np.split(Dtrain, [pre_idx])\n",
    "    adj_idx = int(Dtrain_opt.shape[0] * 0.5)\n",
    "    Dtrain_opt, Dtrain_adj = np.split(Dtrain_opt, [adj_idx])\n",
    "\n",
    "    X_pre = Dtrain_pre[:,0:5]\n",
    "    Y_pre = Dtrain_pre[:,5]\n",
    "\n",
    "    X_opt = Dtrain_opt[:,0:5]\n",
    "    Y_opt = Dtrain_opt[:,5]\n",
    "\n",
    "    X_adj = Dtrain_adj[:,0:5]\n",
    "    Y_adj = Dtrain_adj[:,5]\n",
    "    \n",
    "    mean_model = RandomForestRegressor(n_estimators = 1000, random_state = 42)\n",
    "    mean_model.fit(X_pre, Y_pre)\n",
    "\n",
    "    mean_est_pre = mean_model.predict(X_pre)\n",
    "    mean_est_opt = mean_model.predict(X_opt)\n",
    "    mean_est_adj = mean_model.predict(X_adj)\n",
    "\n",
    "    Y_centered_squared_pre = (Y_pre - mean_est_pre)**2 \n",
    "    Y_centered_squared_opt = (Y_opt - mean_est_opt)**2 \n",
    "    Y_centered_squared_adj = (Y_adj - mean_est_adj)**2 \n",
    "    \n",
    "    quantile = [0.85,0.95,0.9,0.9]\n",
    "\n",
    "\n",
    "    m1,Q1_opt,Q1_adj,Q1_t = est_quantile(\"NN1\",quantile[0],X_pre,Y_centered_squared_pre,X_opt,X_adj,X_test) # quantile def\n",
    "    m2,Q2_opt,Q2_adj,Q2_t = est_quantile(\"NN2\",quantile[1],X_pre,Y_centered_squared_pre,X_opt,X_adj,X_test)\n",
    "    m3,Q3_opt,Q3_adj,Q3_t = est_quantile(\"qrf\",quantile[2],X_pre,Y_centered_squared_pre,X_opt,X_adj,X_test)\n",
    "    m4,Q4_opt,Q4_adj,Q4_t = est_quantile(\"gb\",quantile[3],X_pre,Y_centered_squared_pre,X_opt,X_adj,X_test)\n",
    "    \n",
    "\n",
    "    ######## Variance estimator ###########\n",
    "    \n",
    "    rf_var_model = RandomForestRegressor(n_estimators = 1000, random_state = 42)\n",
    "    rf_var_model.fit(X_pre, Y_centered_squared_pre)\n",
    "    var_hat_pre = rf_var_model.predict(X_pre)\n",
    "    var_hat_adj = rf_var_model.predict(X_adj)\n",
    "    var_hat_opt = rf_var_model.predict(X_opt)\n",
    "    var_hat_test = rf_var_model.predict(X_test)\n",
    "    \n",
    "    E_opt = np.vstack((np.matrix.flatten(Q1_opt), np.matrix.flatten(Q2_opt), np.matrix.flatten(Q3_opt), np.matrix.flatten(Q4_opt), var_hat_opt))\n",
    "    E_adj = np.vstack((np.matrix.flatten(Q1_adj), np.matrix.flatten(Q2_adj), np.matrix.flatten(Q3_adj), np.matrix.flatten(Q4_adj), var_hat_adj))\n",
    "    E_test = np.vstack((np.matrix.flatten(Q1_t), np.matrix.flatten(Q2_t), np.matrix.flatten(Q3_t), np.matrix.flatten(Q4_t), var_hat_test))\n",
    "    \n",
    "    n_opt = X_opt.shape[0]\n",
    "    n_adj = X_adj.shape[0]\n",
    "    n_test = X_test.shape[0]\n",
    "\n",
    "    cons_opt = np.ones(n_opt).reshape(1,-1)\n",
    "    cons_adj = np.ones(n_adj).reshape(1,-1)\n",
    "    cons_test = np.ones(n_test).reshape(1,-1)\n",
    "\n",
    "    A_opt = np.vstack((E_opt,cons_opt))\n",
    "    A_adj = np.vstack((E_adj,cons_adj))\n",
    "    A_test = np.vstack((E_test,cons_test))\n",
    "\n",
    "    weight = cp.Variable(A_opt.shape[0])\n",
    "\n",
    "\n",
    "    constraints = [weight>=0]+[weight @ A_opt >= Y_centered_squared_opt]  ### Change the objective \n",
    "    prob = cp.Problem(cp.Minimize(cp.sum(weight @ A_test)), constraints)\n",
    "    prob.solve()\n",
    "    optimal_weight = weight.value\n",
    "\n",
    "    f_hat_init_opt = optimal_weight @  A_opt\n",
    "    f_hat_init_adj = optimal_weight @  A_adj\n",
    "    f_hat_init_test = optimal_weight @ A_test\n",
    "    \n",
    "    domain_cov = np.vstack((X_pre, X_test))\n",
    "    domain_idx =  np.vstack((np.zeros(X_pre.shape[0]).reshape(-1,1), np.ones(X_test.shape[0]).reshape(-1,1)))\n",
    "    domain_idx = np.matrix.flatten(domain_idx)\n",
    "\n",
    "    logistic_model = LogisticRegression(penalty='l2', solver='lbfgs', max_iter=1000,random_state=0) \n",
    "    logistic_model.fit(domain_cov, domain_idx)\n",
    "\n",
    "    prob_hat = logistic_model.predict_proba(X_adj)[:,0]\n",
    "    prob_hat = np.clip(prob_hat, 0.01, 0.99)\n",
    "    est_ratio = prob_hat/(1-prob_hat) #### Multiplying by the constant or normalization #####\n",
    "    #est_ratio /= est_ratio.mean()\n",
    "    est_ratio = est_ratio * (X_pre.shape[0]/X_test.shape[0])\n",
    "    \n",
    "    \n",
    "    delta = 1.01\n",
    "    alpha = 0.05\n",
    "    stepsize = 0.005\n",
    "    prop_outside = np.mean(est_ratio * ((Y_centered_squared_adj/f_hat_init_adj) > delta))\n",
    "    while prop_outside <= alpha:\n",
    "        delta = delta - stepsize\n",
    "        prop_outside = np.mean(est_ratio * ((Y_centered_squared_adj/f_hat_init_adj) > delta))\n",
    "        \n",
    "    mean_est_test = mean_model.predict(X_test)\n",
    "    Y_centered_squared_test = (Y_test - mean_est_test)**2 \n",
    "    cov = np.mean(Y_centered_squared_test < (delta * f_hat_init_test))\n",
    "    bw = 2 * np.mean((delta * f_hat_init_test) ** 0.5)\n",
    "    return cov, bw"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "cc25add8",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/var/folders/8h/91p3j8752c13k8tln_mp4lbw00dncy/T/ipykernel_57253/673238192.py:106: RuntimeWarning: invalid value encountered in sqrt\n"
     ]
    }
   ],
   "source": [
    "from joblib import Parallel, delayed\n",
    "results = Parallel(n_jobs=10)(delayed(run_utopia)(mc_iter) for mc_iter in range(200))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "d05b41b5",
   "metadata": {},
   "outputs": [],
   "source": [
    "coverage = np.array(results)[:,0]\n",
    "bandwidth = np.array(results)[:,1]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "2141a568",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.9610505319148936"
      ]
     },
     "execution_count": 16,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "coverage.mean()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "f773dc48",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Text(0.5, 0, 'Coverage--Our method')"
      ]
     },
     "execution_count": 17,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAjMAAAGwCAYAAABcnuQpAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy88F64QAAAACXBIWXMAAA9hAAAPYQGoP6dpAABLOElEQVR4nO3deXwU9f0/8Nfskc3m2tz3wRVIwg3hVAS0HNEiSFVUtOBP/UpFhPLzKKVqtJZ4VESr2GIVsAXRr3j+RA7FcMgdCFfCHdhALjZ3ssnm2M/vj5iVJQmQzSazk7yej84j2ZnPzL53hzGvznzmM5IQQoCIiIhIoVRyF0BERETUFgwzREREpGgMM0RERKRoDDNERESkaAwzREREpGgMM0RERKRoDDNERESkaBq5C2hvVqsVOTk58Pb2hiRJcpdDREREN0AIgfLycoSHh0Oluva5l04fZnJychAVFSV3GUREROSA7OxsREZGXrNNpw8z3t7eABq+DB8fH5mrISIiohtRVlaGqKgo29/xa+n0Yabx0pKPjw/DDBERkcLcSBcRdgAmIiIiRWOYISIiIkVjmCEiIiJFY5ghIiIiRWOYISIiIkVjmCEiIiJFY5ghIiIiRWOYISIiIkVjmCEiIiJFY5ghIiIiRWOYISIiIkVjmCEiIiJFY5ghIiIiRWOYISIiIkXTyF0AERERyctoNMJkMjm0bmBgIKKjo51cUeswzBAREXVhRqMRcfHxqDKbHVpf7+GBE5mZsgYahhkiIqIuzGQyocpsxszn3kBIdM9WrZtvPIs1rz0Dk8nEMENERETyConuicjYvnKX4RB2ACYiIiJFY5ghIiIiRWOYISIiIkVjmCEiIiJFY5ghIiIiRWOYISIiIkVjmCEiIiJFY5ghIiIiRWOYISIiIkVjmCEiIiJFc5kwk5KSAkmSsGDBAts8IQSSk5MRHh4OvV6PcePG4fjx4/IVSURERC7HJcLM/v37sWLFCgwYMMBu/uuvv46lS5fi3Xffxf79+xEaGooJEyagvLxcpkqJiIjI1cgeZioqKjBz5kx88MEH8PPzs80XQmDZsmVYvHgxpk+fjn79+mH16tUwm81Yu3Zti9uzWCwoKyuzm4iIiKjzkj3MzJ07F3fccQd+85vf2M3PyspCXl4eJk6caJun0+kwduxY7Nq1q8XtpaSkwGAw2KaoqKh2q52IiIjkJ2uYWbduHQ4ePIiUlJQmy/Ly8gAAISEhdvNDQkJsy5qzaNEilJaW2qbs7GznFk1EREQuRSPXG2dnZ2P+/PnYvHkz3N3dW2wnSZLdayFEk3lX0ul00Ol0TquTiIiIXJtsZ2bS0tJQUFCAoUOHQqPRQKPRYNu2bXjnnXeg0WhsZ2SuPgtTUFDQ5GwNERERdV2yhZnbbrsNR48eRXp6um1KTEzEzJkzkZ6ejh49eiA0NBRbtmyxrVNTU4Nt27Zh9OjRcpVNRERELka2y0ze3t7o16+f3TxPT08EBATY5i9YsABLlixBbGwsYmNjsWTJEnh4eOCBBx6Qo2QiIiJyQbKFmRvx7LPPoqqqCk888QSKi4sxYsQIbN68Gd7e3nKXRkRERC7CpcJMamqq3WtJkpCcnIzk5GRZ6iEiIiLXJ/s4M0RERERtwTBDREREisYwQ0RERIrGMENERESKxjBDREREisYwQ0RERIrGMENERESKxjBDREREisYwQ0RERIrGMENERESKxjBDREREisYwQ0RERIrGMENERESKxjBDREREisYwQ0RERIrGMENERESKxjBDREREisYwQ0RERIrGMENERESKxjBDREREisYwQ0RERIrGMENERESKxjBDREREisYwQ0RERIrGMENERESKxjBDREREisYwQ0RERIrGMENERESKxjBDREREisYwQ0RERIrGMENERESKJmuYef/99zFgwAD4+PjAx8cHo0aNwvfff29bPnv2bEiSZDeNHDlSxoqJiIjI1WjkfPPIyEi8+uqr6NWrFwBg9erVmDp1Kg4dOoS+ffsCACZPnoyVK1fa1nFzc5OlViIiInJNsoaZKVOm2L3+29/+hvfffx979uyxhRmdTofQ0NAb3qbFYoHFYrG9Lisrc06xRERE5JJcps9MfX091q1bh8rKSowaNco2PzU1FcHBwejduzcee+wxFBQUXHM7KSkpMBgMtikqKqq9SyciIiIZyR5mjh49Ci8vL+h0OsyZMwdffvklEhISAABJSUlYs2YNtm7dijfffBP79+/Hrbfeanfm5WqLFi1CaWmpbcrOzu6oj0JEREQykPUyEwD06dMH6enpKCkpwfr16zFr1ixs27YNCQkJmDFjhq1dv379kJiYiJiYGHz33XeYPn16s9vT6XTQ6XQdVT4RERHJTPYw4+bmZusAnJiYiP379+Ptt9/Gv/71ryZtw8LCEBMTg9OnT3d0mUREROSiZL/MdDUhRIuXkQoLC5GdnY2wsLAOroqIiIhclaxnZv785z8jKSkJUVFRKC8vx7p165CamoqNGzeioqICycnJ+N3vfoewsDCcP38ef/7znxEYGIi77rpLzrKJiIjIhcgaZvLz8/HQQw8hNzcXBoMBAwYMwMaNGzFhwgRUVVXh6NGj+Pjjj1FSUoKwsDCMHz8en376Kby9veUsm4iIiFyIrGHmww8/bHGZXq/Hpk2bOrAaIiIiUiKX6zNDRERE1BoMM0RERKRoDDNERESkaAwzREREpGgMM0RERKRoDDNERESkaAwzREREpGgMM0RERKRoDDNERESkaAwzREREpGgMM0RERKRoDDNERESkaAwzREREpGgMM0RERKRoDDNERESkaAwzREREpGgMM0RERKRoDDNERESkaAwzREREpGgMM0RERKRoDDNERESkaAwzREREpGgMM0RERKRoDDNERESkaAwzREREpGgauQsgIiLqTIxGI0wmk0PrBgYGIjo62skVdX4MM0RERE5iNBoRFx+PKrPZofX1Hh44kZnJQNNKDDNEREROYjKZUGU2Y+ZzbyAkumer1s03nsWa156ByWRimGklhhkiIiInC4nuicjYvnKX0WWwAzAREREpGsMMERERKZqsYeb999/HgAED4OPjAx8fH4waNQrff/+9bbkQAsnJyQgPD4der8e4ceNw/PhxGSsmIiIiVyNrmImMjMSrr76KAwcO4MCBA7j11lsxdepUW2B5/fXXsXTpUrz77rvYv38/QkNDMWHCBJSXl8tZNhEREbkQWcPMlClTcPvtt6N3797o3bs3/va3v8HLywt79uyBEALLli3D4sWLMX36dPTr1w+rV6+G2WzG2rVrW9ymxWJBWVmZ3URERESdl8v0mamvr8e6detQWVmJUaNGISsrC3l5eZg4caKtjU6nw9ixY7Fr164Wt5OSkgKDwWCboqKiOqJ8IiIikonsYebo0aPw8vKCTqfDnDlz8OWXXyIhIQF5eXkAgJCQELv2ISEhtmXNWbRoEUpLS21TdnZ2u9ZPRERE8pJ9nJk+ffogPT0dJSUlWL9+PWbNmoVt27bZlkuSZNdeCNFk3pV0Oh10Ol271UtERESuRfYzM25ubujVqxcSExORkpKCgQMH4u2330ZoaCgANDkLU1BQ0ORsDREREXVdsoeZqwkhYLFY0L17d4SGhmLLli22ZTU1Ndi2bRtGjx4tY4VERETkSmS9zPTnP/8ZSUlJiIqKQnl5OdatW4fU1FRs3LgRkiRhwYIFWLJkCWJjYxEbG4slS5bAw8MDDzzwgJxlExERkQuRNczk5+fjoYceQm5uLgwGAwYMGICNGzdiwoQJAIBnn30WVVVVeOKJJ1BcXIwRI0Zg8+bN8Pb2lrNsIiIiciGyhpkPP/zwmsslSUJycjKSk5M7piAiIiJSHJfrM0NERETUGgwzREREpGgMM0RERKRoDDNERESkaAwzREREpGgMM0RERKRoDDNERESkaAwzREREpGgMM0RERKRoDDNERESkaAwzREREpGgMM0RERKRoDDNERESkaAwzREREpGgMM0RERKRoDDNERESkaAwzREREpGgMM0RERKRoDDNERESkaAwzREREpGgMM0RERKRoDDNERESkaAwzREREpGgMM0RERKRoDDNERESkaAwzREREpGgMM0RERKRoDDNERESkaAwzREREpGgMM0RERKRoDDNERESkaLKGmZSUFAwbNgze3t4IDg7GtGnTcPLkSbs2s2fPhiRJdtPIkSNlqpiIiIhcjaxhZtu2bZg7dy727NmDLVu2oK6uDhMnTkRlZaVdu8mTJyM3N9c2bdiwQaaKiYiIyNVo5HzzjRs32r1euXIlgoODkZaWhltuucU2X6fTITQ09Ia2abFYYLFYbK/LysqcUywRERG5JJfqM1NaWgoA8Pf3t5ufmpqK4OBg9O7dG4899hgKCgpa3EZKSgoMBoNtioqKateaiYiISF4uE2aEEFi4cCFuvvlm9OvXzzY/KSkJa9aswdatW/Hmm29i//79uPXWW+3Ovlxp0aJFKC0ttU3Z2dkd9RGIiIhIBrJeZrrSk08+iSNHjmDnzp1282fMmGH7vV+/fkhMTERMTAy+++47TJ8+vcl2dDoddDpdu9dLRERErsElwsy8efPwzTffYPv27YiMjLxm27CwMMTExOD06dMdVB0RERG5MlnDjBAC8+bNw5dffonU1FR07979uusUFhYiOzsbYWFhHVAhERERuTpZ+8zMnTsX//3vf7F27Vp4e3sjLy8PeXl5qKqqAgBUVFTg6aefxu7du3H+/HmkpqZiypQpCAwMxF133SVn6UREROQiZD0z8/777wMAxo0bZzd/5cqVmD17NtRqNY4ePYqPP/4YJSUlCAsLw/jx4/Hpp5/C29tbhoqJiIjI1TgUZnr06IH9+/cjICDAbn5JSQmGDBmCc+fO3dB2hBDXXK7X67Fp0yZHSiQiIqIuwqHLTOfPn0d9fX2T+RaLBZcuXWpzUUREREQ3qlVnZr755hvb75s2bYLBYLC9rq+vx48//ohu3bo5rTgiIiKi62lVmJk2bRoAQJIkzJo1y26ZVqtFt27d8OabbzqtOCIiIqLraVWYsVqtAIDu3btj//79CAwMbJeiiIiIiG6UQx2As7KynF0HERERkUMcvjX7xx9/xI8//oiCggLbGZtGH330UZsLIyIiIroRDoWZl156CS+//DISExMRFhYGSZKcXRcRERHRDXEozPzzn//EqlWr8NBDDzm7HiIiIqJWcWicmZqaGowePdrZtRARERG1mkNh5tFHH8XatWudXQsRERFRqzl0mam6uhorVqzADz/8gAEDBkCr1dotX7p0qVOKIyIiIroeh8LMkSNHMGjQIADAsWPH7JaxMzARERF1JIfCzE8//eTsOoiIiIgc4lCfGSIiIiJX4dCZmfHjx1/zctLWrVsdLoiIiIioNRwKM439ZRrV1tYiPT0dx44da/IASiIiIqL25FCYeeutt5qdn5ycjIqKijYVRERERNQaTu0z8+CDD/K5TERERNShnBpmdu/eDXd3d2dukoiIiOiaHLrMNH36dLvXQgjk5ubiwIEDeP75551SGBEREdGNcCjMGAwGu9cqlQp9+vTByy+/jIkTJzqlMCIiIqIb4VCYWblypbPrICIiInKIQ2GmUVpaGjIzMyFJEhISEjB48GBn1UVERER0QxwKMwUFBbjvvvuQmpoKX19fCCFQWlqK8ePHY926dQgKCnJ2nURERETNcuhupnnz5qGsrAzHjx9HUVERiouLcezYMZSVleGpp55ydo1ERERELXLozMzGjRvxww8/ID4+3jYvISEB7733HjsAExERycBoNMJkMrV6vczMzHaopmM5FGasViu0Wm2T+VqtFlartc1FERER0Y0zGo2Ii49Hldns8DaUPIK/Q2Hm1ltvxfz58/HJJ58gPDwcAHDp0iX88Y9/xG233ebUAomIiOjaTCYTqsxmzHzuDYRE92zVupn7tuH71W+jurq6naprfw6FmXfffRdTp05Ft27d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\n",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "sns.histplot(np.array(coverage).reshape(-1, 1), kde = True, bins = 30, legend = False)\n",
    "plt.xlabel('Coverage--Our method')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "fd96dda1",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Text(0.5, 0, 'Bandwidth--Our method')"
      ]
     },
     "execution_count": 20,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAjMAAAGwCAYAAABcnuQpAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy88F64QAAAACXBIWXMAAA9hAAAPYQGoP6dpAABUv0lEQVR4nO3dd3yTdeIH8M+TpEl3S3dLJ9CWlrJBkA0y78CBniCicC78CTgQVORQ9O7AcSDc4Tg9RTzBcSqIC2SWvQplFiili07SlaYjSZPn90chWmabpn3ypJ/365UXZDzJ5wkQPv3m+zxfQRRFEUREREQypZA6ABEREVFzsMwQERGRrLHMEBERkayxzBAREZGsscwQERGRrLHMEBERkayxzBAREZGsqaQO0NIsFgvy8/Ph5eUFQRCkjkNERESNIIoiKisrERYWBoXi5mMvTl9m8vPzERERIXUMIiIiskFubi7Cw8Nv+hinLzNeXl4A6t8Mb29vidMQERFRY+h0OkRERFj/H78Zpy8zV75a8vb2ZpkhIiKSmcZMEeEEYCIiIpI1lhkiIiKSNZYZIiIikjWWGSIiIpI1lhkiIiKSNZYZIiIikjWWGSIiIpI1lhkiIiKSNZYZIiIikjWWGSIiIpI1lhkiIiKSNZYZIiIikjWWGSIiIpI1lhkiIiKSNZXUAYhaWk5ODrRarU3bBgQEIDIy0s6JiIjInlhmyKnl5OSgc0ICaqqrbdrezd0dZ9LSWGiIiBwYyww5Na1Wi5rqajz44tsIjuzYpG2LcjKw5s150Gq1LDNERA6MZYbahODIjgiP7SJ1DCIiagGcAExERESyxjJDREREssYyQ0RERLLGMkNERESyxjJDREREssYyQ0RERLLGMkNERESyxjJDREREssYyQ0RERLLGMkNERESyxjJDREREssYyQ0RERLLGMkNERESyxjJDREREssYyQ0RERLLGMkNERESyxjJDREREssYyQ0RERLLGMkNERESyJmmZWbJkCfr27QsvLy8EBQXh7rvvxtmzZxs8Zvr06RAEocGlf//+EiUmIiIiRyNpmUlOTsbMmTOxf/9+bN68GXV1dRg9ejSqqqoaPG7s2LEoKCiwXn7++WeJEhMREZGjUUn54hs3bmxwfdWqVQgKCkJKSgqGDBlivV2j0SAkJKS14xEREZEMONScmYqKCgCAn59fg9t37NiBoKAgxMXF4fHHH0dxcfENn8NgMECn0zW4EBERkfNymDIjiiLmzJmDQYMGISkpyXr7uHHjsGbNGmzbtg1Lly7FoUOHMGLECBgMhus+z5IlS+Dj42O9REREtNYuEBERkQQk/Zrp92bNmoXjx49j9+7dDW6fNGmS9fdJSUno06cPoqKi8NNPP2HixInXPM/8+fMxZ84c63WdTsdCQ0RE5MQcoszMnj0bGzZswM6dOxEeHn7Tx4aGhiIqKgrp6enXvV+j0UCj0bRETCIiInJAkpYZURQxe/ZsrFu3Djt27EBMTMwttykpKUFubi5CQ0NbISERERE5OknnzMycOROff/451q5dCy8vLxQWFqKwsBA1NTUAAL1ej7lz52Lfvn3IysrCjh07MGHCBAQEBOCee+6RMjoRERE5CElHZt5//30AwLBhwxrcvmrVKkyfPh1KpRInTpzAZ599hvLycoSGhmL48OH46quv4OXlJUFiIiIicjSSf810M25ubti0aVMrpSEiIiI5cphDs4mIiIhswTJDREREssYyQ0RERLLGMkNERESyxjJDREREssYyQ0RERLLGMkNERESyxjJDREREssYyQ0RERLLGMkNERESyxjJDREREssYyQ0RERLLGMkNERESyxjJDREREssYyQ0RERLLGMkNERESyxjJDREREssYyQ0RERLLGMkNERESyxjJDREREssYyQ0RERLLGMkNERESyxjJDREREssYyQ0RERLLGMkNERESyxjJDREREssYyQ0RERLLGMkNERESyxjJDREREssYyQ0RERLLGMkNERESyppI6AJGjS0tLs2m7gIAAREZG2jkNERFdjWWG6AZ0pZcAAFOnTrVpezd3d5xJS2OhISJqYSwzRDdQo9cBAP44YwHiu/Vu0rZFORlY8+Y8aLValhkiohbGMkN0C/5hUQiP7SJ1DCIiugFOACYiIiJZY5khIiIiWWOZISIiIlljmSEiIiJZY5khIiIiWWOZISIiIlljmSEiIiJZY5khIiIiWWOZISIiIlljmSEiIiJZY5khIiIiWWOZISIiIlljmSEiIiJZY5khIiIiWWOZISIiIlljmSEiIiJZY5khIiIiWWOZISIiIlljmSEiIiJZk7TMLFmyBH379oWXlxeCgoJw99134+zZsw0eI4oiFi1ahLCwMLi5uWHYsGE4deqURImJiIjI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62Oc+XPwVvvv7J+Djfnzv1pzPwOFyy9MREREJEucM0NERESyxjJDREREssYyQ0RERLLGMkNERESyxjJDREREssYyQ0RERLLGMkNERESyxjJDREREssYyQ9QGDRs2DM8++6xN2y5atAg9evS46WOmT5+Ou++++6aP2bFjBwRBQHl5uU052pqsrCwIgoDU1FS7P3d0dDSWL19u9+clai0sM0QSmD59OgRBsF78/f0xduxYHD9+XOpotzR37lxs3bq1Sds0pzw1ltlsxjvvvINu3brB1dUVvr6+GDduHPbs2dOir9sSGlMGieg3LDNEEhk7diwKCgpQUFCArVu3QqVSYfz48VLHuiVPT0+HW+FXFEVMnjwZr7/+Op5++mmkpaUhOTkZERERGDZsGNavX9+s5zcajfYJSkQtgmWGSCIajQYhISEICQlBjx498OKLLyI3NxeXLl2yPubFF19EXFwc3N3d0aFDByxcuBAmk8l6/5WvfP773/8iOjoaPj4+mDx5MiorK62PqaqqwsMPPwxPT0+EhoZi6dKlDXL861//QteuXa3X169fD0EQ8O6771pvGzNmDObPn9/gNa8wm82YM2cOfH194e/vjxdeeAG/X/Jt+vTpSE5OxooVK6wjUVlZWdb7U1JS0KdPH7i7u2PAgAE4e/Zsk9/Lr7/+Gt988w0+++wzPPbYY4iJiUH37t3x4Ycf4s4778Rjjz2Gqqoqa56rRz2effZZDBs2zHp92LBhmDVrFubMmYOAgACMGjXquq975bkWL16M4OBg+Pr64rXXXkNdXR3mzZsHPz8/hIeH45NPPmmwXV5eHiZNmoR27drB398fd911l/U9WbRoEVavXo3vv//e+n79fjXxCxcuYPjw4XB3d0f37t2xb9++Bs/97bffokuXLtBoNIiOjr7mz7u4uBgTJkyAm5sbYmJisGbNmia800SOiWWGyAHo9XqsWbMGnTp1ajDq4eXlhU8//RSnT5/GihUr8NFHH+Gdd95psG1GRgbWr1+PH3/8ET/++COSk5PxxhtvWO+fN28etm/fjnXr1uHXX3/Fjh07kJKSYr1/2LBhOHXqFLRaLQAgOTkZAQEBSE5OBgDU1dVh7969GDp06HWzL126FJ988gk+/vhj7N69G6WlpVi3bp31/hUrVuD222/H448/bh2JioiIsN6/YMECLF26FIcPH4ZK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\n",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "sns.histplot(np.array(bandwidth).reshape(-1, 1), kde = True,bins = 30, legend = False)\n",
    "plt.xlabel('Bandwidth--Our method')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "d0e31d8e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "14.350432456969925"
      ]
     },
     "execution_count": 25,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.nanmean(bandwidth[coverage > 0.95]) "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "644d53af",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "2.2235266267294307"
      ]
     },
     "execution_count": 32,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.sqrt(np.ma.masked_invalid(bandwidth[coverage >= 0.95]).var())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "dd85cc68",
   "metadata": {},
   "outputs": [],
   "source": [
    "np.savetxt('Coverage_Ours_200.csv', coverage, delimiter=',')\n",
    "np.savetxt('Bandwidth_Ours_200.csv', bandwidth, delimiter=',')"
   ]
  }
 ],
 "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.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
