{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "49281bb8",
   "metadata": {},
   "outputs": [],
   "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",
    "\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",
    "import warnings"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "6c30a49c",
   "metadata": {},
   "outputs": [],
   "source": [
    "# read the data\n",
    "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",
    "# def mydata(Data, beta = np.array([-1, 0, 0, 0, 1])):\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] * 0.5)\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_size = int(0.25 * len(Dtest)) # Sample 25% of the points from Dtest with replacement\n",
    "#     sample_indices = np.random.choice(len(Dtest), size=sample_size, replace=True, p=probabilities)\n",
    "#     Dshift = Dtest[sample_indices]\n",
    "    \n",
    "#     return Dtrain, Dtest, Dshift\n",
    "\n",
    "\n",
    "def mydata(Data, beta = np.array([-1, 0, 0, 0, 1]), 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": 4,
   "id": "428f9627",
   "metadata": {},
   "outputs": [],
   "source": [
    "random_seed = 42\n",
    "\n",
    "def prob_est(Dtrain, Dshift, est_type = 'lr'):\n",
    "    X_S = Dtrain[:,0:5]\n",
    "    X_T = Dshift[:,0:5]\n",
    "    n_S = X_S.shape[0]\n",
    "    n_T = X_T.shape[0]\n",
    "    Y_S = np.zeros(n_S).reshape(-1,1)\n",
    "    Y_T = np.ones(n_T).reshape(-1,1)\n",
    "    X_train = np.vstack((X_S, X_T))\n",
    "    Y_train = np.vstack((Y_S, Y_T))\n",
    "  \n",
    "    if est_type == 'lr':\n",
    "        # fit logistic regression\n",
    "        with warnings.catch_warnings():\n",
    "            warnings.simplefilter(\"ignore\")\n",
    "            logistic_model = LogisticRegression(penalty='l2', solver='lbfgs', max_iter=1000,random_state=0) \n",
    "            logistic_model.fit(X_train, Y_train[:,0])\n",
    "\n",
    "        return logistic_model\n",
    "    \n",
    "    if est_type == 'rf':\n",
    "        # fit random forest\n",
    "        rf_model = RandomForestClassifier(n_estimators = 100, random_state=random_seed)\n",
    "        rf_model.fit(X_train, Y_train[:,0])\n",
    "\n",
    "        return rf_model\n",
    "\n",
    "    \n",
    "def ratio_est(model, X):\n",
    "    \n",
    "        prob_1 = model.predict_proba(X)[:,1]\n",
    "        prob_1 = np.clip(prob_1, 0.01, 0.99)\n",
    "        est_ratio = prob_1/(1-prob_1)\n",
    "        \n",
    "        return est_ratio\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "5067f88d",
   "metadata": {},
   "outputs": [],
   "source": [
    "def weighting_function(model, X_calib, x_test, x):\n",
    "    # Define weighting function here\n",
    "    # all the data should be array\n",
    "    ratio_calib = ratio_est(model, X_calib).sum()\n",
    "    ratio_test = ratio_est(model, x_test)\n",
    "    ratio_sum = ratio_calib + ratio_test\n",
    "    ratio_x = ratio_est(model, x)\n",
    "    return ratio_x/ratio_sum\n",
    "\n",
    "\n",
    "def weighted_quantile(values, weights, quantile):\n",
    "    \n",
    "    \"\"\" Compute the weighted quantile of a 1D numpy array.\n",
    "    \"\"\"\n",
    "    values = np.array(values)\n",
    "    weights = np.array(weights)\n",
    "    \n",
    "    sorter = np.argsort(values)\n",
    "    values = values[sorter]\n",
    "    weights = weights[sorter]\n",
    "    cumulative_weights = np.cumsum(weights)\n",
    "    if cumulative_weights[-1]>= quantile:\n",
    "        idx = np.argmax(cumulative_weights >= quantile)\n",
    "        return values[idx]\n",
    "    else:\n",
    "        return np.infty\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "4ce03eaa",
   "metadata": {},
   "outputs": [],
   "source": [
    "def weighted_conformal_cov_width(Dtrain, Dshift, alpha=0.05):\n",
    "    \n",
    "    n = Dshift.shape[0]\n",
    "    cov = []\n",
    "    width = []\n",
    "    \n",
    "    \n",
    "    prob_est_model = prob_est(Dtrain, Dshift, est_type = 'lr')\n",
    "    \n",
    "    # split weighted conformal\n",
    "    D = Dtrain\n",
    "    np.random.shuffle(D)\n",
    "    split_idx = int(D.shape[0] * 0.5)\n",
    "    D1, D2 = np.split(D, [split_idx])\n",
    "    X1 = D1[:,0:5]\n",
    "    y1 = D1[:,-1]\n",
    "    X2 = D2[:,0:5]\n",
    "    y2 = D2[:,-1]\n",
    "    \n",
    "    # Train the model on the first part of the data\n",
    "    model = LinearRegression()\n",
    "    model.fit(X1, y1)\n",
    "    \n",
    "    # Compute the model's predictions for the calibration set\n",
    "    y1_hat = model.predict(X1)\n",
    "    y_calib_pred = model.predict(X2)\n",
    "            \n",
    "    # Calculate the absolute errors on the calibration set\n",
    "    errors_calib = np.abs(y_calib_pred - y2)\n",
    "    \n",
    "    est_error_1 = (y1_hat - y1) ** 2\n",
    "    var_model = RandomForestRegressor(n_estimators = 1000, random_state = 42)\n",
    "    var_model.fit(X1, est_error_1)\n",
    "    var_hat = var_model.predict(X2)\n",
    "    sd_calib_pred = var_hat ** 0.5\n",
    "    errors_calib = np.abs(y_calib_pred - y2)/np.clip(sd_calib_pred, 0.001, 1e10)\n",
    "    \n",
    "    for i in range(n):\n",
    "        x_test = Dshift[i,0:5]\n",
    "        x_test = np.array(x_test).reshape(1,-1)\n",
    "        y_test = Dshift[i,5]\n",
    "    \n",
    "    \n",
    "        # Calculate the weights for the calibration set\n",
    "        weights_calib = weighting_function(prob_est_model, X2, x_test, X2)\n",
    "\n",
    "\n",
    "        # Calculate the weighted quantile of the errors\n",
    "        quantile = weighted_quantile(errors_calib, weights_calib, 1 - alpha)\n",
    "\n",
    "        # Now for the test\n",
    "        y_test_pred = model.predict(x_test)\n",
    "        sd_test_pred = var_model.predict(x_test) ** 0.5\n",
    "        \n",
    "        width.append(2*quantile*sd_test_pred)\n",
    "        cov.append((np.abs(y_test - y_test_pred)/sd_test_pred) <= quantile)\n",
    "\n",
    "    \n",
    "    return np.array(cov).mean(), np.array(width).mean()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "d5c39b15",
   "metadata": {},
   "outputs": [],
   "source": [
    "def conformal_test(Data,alpha=0.05, n = 200):\n",
    "    \n",
    "    coverage = []\n",
    "    bandwidth = []\n",
    "    for i in range(n):\n",
    "        Dtrain, Dtest, Dshift = mydata(Data)\n",
    "        cov, bw = weighted_conformal_cov_width(Dtrain, Dshift, alpha)\n",
    "        coverage.append(cov)\n",
    "        bandwidth.append(bw)\n",
    "        print(i, \"is done!\")\n",
    "    return coverage, bandwidth"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "c579342e",
   "metadata": {},
   "outputs": [],
   "source": [
    "alpha = 0.05\n",
    "def run_weighted_conformal(n):\n",
    "    Dtrain, Dtest, Dshift = mydata(Data)\n",
    "    cov, bw = weighted_conformal_cov_width(Dtrain, Dshift, alpha)\n",
    "    return cov, bw\n",
    "\n",
    "from joblib import Parallel, delayed\n",
    "results = Parallel(n_jobs=10)(delayed(run_weighted_conformal)(i) for i in range(200))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "81744f77",
   "metadata": {},
   "outputs": [],
   "source": [
    "coverage = np.array(results)[:,0]\n",
    "bandwidth = np.array(results)[:,1]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "0841be48",
   "metadata": {},
   "outputs": [
    {
     "ename": "NameError",
     "evalue": "name 'cov' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[10], line 1\u001b[0m\n\u001b[0;32m----> 1\u001b[0m np\u001b[38;5;241m.\u001b[39mmean(\u001b[43mcov\u001b[49m)\n",
      "\u001b[0;31mNameError\u001b[0m: name 'cov' is not defined"
     ]
    }
   ],
   "source": [
    "np.mean(cov)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "8cb155b8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Text(0.5, 0, 'Coverage--Weighted conformal')"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAjMAAAGwCAYAAABcnuQpAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy88F64QAAAACXBIWXMAAA9hAAAPYQGoP6dpAABQ5ElEQVR4nO3deXwTdf4/8FfSpmnapjc9Qu8CpZT7PkRABURlUdxdUEB01UVFPFhR+aKCuIurKB6ryA8PQEXQ9V4XOUQuuQoFBEoopRRCSw/SK22Tpm3y+f1RyRoo0IS2k2lfz8cjD5qZ+cy859Pp5MVkDoUQQoCIiIhIppRSF0BERER0LRhmiIiISNYYZoiIiEjWGGaIiIhI1hhmiIiISNYYZoiIiEjWGGaIiIhI1rylLqCl2e12nDt3DlqtFgqFQupyiIiIqAmEEKisrIROp4NSeeVjL20+zJw7dw6xsbFSl0FERERuOHv2LGJiYq44TZsPM1qtFkBDZwQGBkpcDRERETWFyWRCbGys43P8Stp8mLnw1VJgYCDDDBERkcw05RQRngBMREREssYwQ0RERLLGMENERESyxjBDREREssYwQ0RERLLGMENERESyxjBDREREssYwQ0RERLLGMENERESyxjBDREREssYwQ0RERLLGMENERESyxjBDREREssYwQ0RERLLmLXUBREREJC2DwQCj0ehW2/DwcMTFxTVzRa5hmCEiImrHDAYDuqamwmI2u9Ve4+eH43q9pIGGYYaIiKgdMxqNsJjNmPLMYkTGJbvUtsiQg9WvzIHRaGSYISIiImlFxiUjpnOa1GW4hScAExERkawxzBAREZGsMcwQERGRrDHMEBERkawxzBAREZGsMcwQERGRrDHMEBERkawxzBAREZGsMcwQERGRrDHMEBERkawxzBAREZGsMcwQERGRrDHMEBERkawxzBAREZGsMcwQERGRrDHMEBERkawxzBAREZGsMcwQERGRrDHMEBERkawxzBAREZGsMcwQERGRrDHMEBERkawxzBAREZGsMcwQERGRrDHMEBERkaxJGmZefvllDBgwAFqtFhEREbj99tuRlZXlNI0QAgsWLIBOp4NGo8HIkSORmZkpUcVERETkaSQNM9u2bcPMmTOxZ88ebNq0CfX19RgzZgyqq6sd07z66qtYsmQJ3nnnHezbtw9RUVEYPXo0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U6bgwIEDSE9Pxz333IMRI0Y4Hb6fOHEiTCYTHn74YYwaNcopDMycOROlpaW46667kJ6ejlOnTmHjxo34y1/+cskH7u8lJyejvr4e//rXv3Dq1Cl88sknWLZsmdM0s2bNwrp167BkyRJkZ2fj//2//4cff/zR6X/nL7zwAj7++GMsWLAAmZmZ0Ov1+Pzzz/Hcc8+53GejRo3CL7/8gkOHDjmOzAANYeb9999HTU2N40O9S5cumDJlCu655x58/fXXyM3Nxb59+/DKK69g3bp1jc5/1qxZ+PDDD7Fq1SpkZ2fj73//Ow4fPuzS0RKg4Wub7du3Iz8/33HV2DPPPIPdu3dj5syZOHToELKzs/H9999j1qxZABqC1M0334wHH3wQe/fuRUZGBh544IGrHn1ZsGABXn/9dbz99tvIzs7GgQMH8K9//QtA07cfV/n7++Phhx/GnDlzsH79ehw7dgwPPvggzGYz7r//frfnezmdOnXC/v37sWHDBpw4cQLPP/+805VTRK2BYYaoEYmJiThw4ABGjRqFv/3tb+jevTtGjx6NzZs347333gPQcLj922+/RUhICK6//nrcdNNNSEpKwueff+40r8DAQIwfPx6//vorpkyZ4jROp9Nh586dsNlsGDt2LLp3747HH38cQUFBjiMujenduzeWLFmCV155Bd27d8fq1avx8ssvO00zbNgwLFu2DEuWLEGvXr2wfv16PPnkk/D19XVMM3bsWPzwww/YtGkTBgwYgMGDB2PJkiWIj493uc9GjRoFi8WCTp06Oc6rABrCTGVlJZKTk53OD1mxYgXuuece/O1vf0NKSgr+8Ic/YO/evU7T/N6UKVMwd+5cPPXUU+jbty9yc3Nx7733Oq1PUyxcuBCnT59GcnIyOnToAKDhPJFt27YhOzsbw4cPR58+ffD88887fdW3YsUKxMbGYsSIEZg4cSL++te/XnLU6mLTp0/Hm2++iaVLlyItLQ233XYbsrOzATR9+3HHP//5T9x5552YNm0a+vbti5MnT2LDhg0ICQm55nlf7KGHHsLEiRMxadIkDBo0CCUlJU5HaYhag0I09qUzEbVJDz74II4fP44dO3ZIXUqzGD16NKKiovDJJ59IXQoRSYjnzBC1Ya+99hpGjx4Nf39//Pjjj1i1ahWWLl0qdVluMZvNWLZsGcaOHQsvLy+sWbMGP/30EzZt2iR1aUQkMR6ZIWrD/vznP2Pr1q2orKxEUlISZs2ahYceekjqstxisVgwfvx4HDhwAFarFSkpKXjuuecavRKLiNoXhhkiIiKSNZ4ATERERLLGMENERESyxjBDREREssYwQ0RERLLGMENERESyxjBDREREssYwQ0RERLLGMENERESy9v8BxChYPSdpn6kAAAAASUVORK5CYII=\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--Weighted conformal')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "b86210ff",
   "metadata": {},
   "outputs": [
    {
     "ename": "NameError",
     "evalue": "name 'sns' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[1], line 1\u001b[0m\n\u001b[0;32m----> 1\u001b[0m \u001b[43msns\u001b[49m\u001b[38;5;241m.\u001b[39mhistplot(np\u001b[38;5;241m.\u001b[39marray(bandwidth)\u001b[38;5;241m.\u001b[39mreshape(\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39m, \u001b[38;5;241m1\u001b[39m), kde \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mTrue\u001b[39;00m,bins \u001b[38;5;241m=\u001b[39m \u001b[38;5;241m30\u001b[39m, legend\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mFalse\u001b[39;00m)\n\u001b[1;32m      2\u001b[0m plt\u001b[38;5;241m.\u001b[39mxlabel(\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mBandwidth--Weighted conformal\u001b[39m\u001b[38;5;124m'\u001b[39m)\n",
      "\u001b[0;31mNameError\u001b[0m: name 'sns' is not defined"
     ]
    }
   ],
   "source": [
    "sns.histplot(np.array(bandwidth).reshape(-1, 1), kde = True,bins = 30, legend=False)\n",
    "plt.xlabel('Bandwidth--Weighted conformal')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "a98b9d17",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "4.391388604434692"
      ]
     },
     "execution_count": 28,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.sqrt(np.ma.masked_invalid(bandwidth[coverage >= 0.95]).var())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "aedae0d9",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.03600934187930062"
      ]
     },
     "execution_count": 27,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.sqrt(coverage.var())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "df0cf6d7",
   "metadata": {},
   "outputs": [],
   "source": [
    "np.savetxt('Coverage_WC_200.csv', coverage, delimiter=',')\n",
    "np.savetxt('Bandwidth_WC_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
}
