{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 41,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>age</th>\n",
       "      <th>sex</th>\n",
       "      <th>cp</th>\n",
       "      <th>trestbps</th>\n",
       "      <th>chol</th>\n",
       "      <th>fbs</th>\n",
       "      <th>restecg</th>\n",
       "      <th>thalach</th>\n",
       "      <th>exang</th>\n",
       "      <th>oldpeak</th>\n",
       "      <th>slope</th>\n",
       "      <th>ca</th>\n",
       "      <th>thal</th>\n",
       "      <th>class</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>63.0</td>\n",
       "      <td>1.0</td>\n",
       "      <td>1.0</td>\n",
       "      <td>145.0</td>\n",
       "      <td>233.0</td>\n",
       "      <td>1.0</td>\n",
       "      <td>2.0</td>\n",
       "      <td>150.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>2.3</td>\n",
       "      <td>3.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>6.0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>67.0</td>\n",
       "      <td>1.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>160.0</td>\n",
       "      <td>286.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>2.0</td>\n",
       "      <td>108.0</td>\n",
       "      <td>1.0</td>\n",
       "      <td>1.5</td>\n",
       "      <td>2.0</td>\n",
       "      <td>3.0</td>\n",
       "      <td>3.0</td>\n",
       "      <td>2</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>67.0</td>\n",
       "      <td>1.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>120.0</td>\n",
       "      <td>229.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>2.0</td>\n",
       "      <td>129.0</td>\n",
       "      <td>1.0</td>\n",
       "      <td>2.6</td>\n",
       "      <td>2.0</td>\n",
       "      <td>2.0</td>\n",
       "      <td>7.0</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>37.0</td>\n",
       "      <td>1.0</td>\n",
       "      <td>3.0</td>\n",
       "      <td>130.0</td>\n",
       "      <td>250.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>187.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>3.5</td>\n",
       "      <td>3.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>3.0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>41.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>2.0</td>\n",
       "      <td>130.0</td>\n",
       "      <td>204.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>2.0</td>\n",
       "      <td>172.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>1.4</td>\n",
       "      <td>1.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>3.0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>56.0</td>\n",
       "      <td>1.0</td>\n",
       "      <td>2.0</td>\n",
       "      <td>120.0</td>\n",
       "      <td>236.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>178.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>0.8</td>\n",
       "      <td>1.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>3.0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6</th>\n",
       "      <td>62.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>140.0</td>\n",
       "      <td>268.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>2.0</td>\n",
       "      <td>160.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>3.6</td>\n",
       "      <td>3.0</td>\n",
       "      <td>2.0</td>\n",
       "      <td>3.0</td>\n",
       "      <td>3</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>7</th>\n",
       "      <td>57.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>120.0</td>\n",
       "      <td>354.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>163.0</td>\n",
       "      <td>1.0</td>\n",
       "      <td>0.6</td>\n",
       "      <td>1.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>3.0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>8</th>\n",
       "      <td>63.0</td>\n",
       "      <td>1.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>130.0</td>\n",
       "      <td>254.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>2.0</td>\n",
       "      <td>147.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>1.4</td>\n",
       "      <td>2.0</td>\n",
       "      <td>1.0</td>\n",
       "      <td>7.0</td>\n",
       "      <td>2</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>9</th>\n",
       "      <td>53.0</td>\n",
       "      <td>1.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>140.0</td>\n",
       "      <td>203.0</td>\n",
       "      <td>1.0</td>\n",
       "      <td>2.0</td>\n",
       "      <td>155.0</td>\n",
       "      <td>1.0</td>\n",
       "      <td>3.1</td>\n",
       "      <td>3.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>7.0</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "    age  sex   cp  trestbps   chol  fbs  restecg  thalach  exang  oldpeak  \\\n",
       "0  63.0  1.0  1.0     145.0  233.0  1.0      2.0    150.0    0.0      2.3   \n",
       "1  67.0  1.0  4.0     160.0  286.0  0.0      2.0    108.0    1.0      1.5   \n",
       "2  67.0  1.0  4.0     120.0  229.0  0.0      2.0    129.0    1.0      2.6   \n",
       "3  37.0  1.0  3.0     130.0  250.0  0.0      0.0    187.0    0.0      3.5   \n",
       "4  41.0  0.0  2.0     130.0  204.0  0.0      2.0    172.0    0.0      1.4   \n",
       "5  56.0  1.0  2.0     120.0  236.0  0.0      0.0    178.0    0.0      0.8   \n",
       "6  62.0  0.0  4.0     140.0  268.0  0.0      2.0    160.0    0.0      3.6   \n",
       "7  57.0  0.0  4.0     120.0  354.0  0.0      0.0    163.0    1.0      0.6   \n",
       "8  63.0  1.0  4.0     130.0  254.0  0.0      2.0    147.0    0.0      1.4   \n",
       "9  53.0  1.0  4.0     140.0  203.0  1.0      2.0    155.0    1.0      3.1   \n",
       "\n",
       "   slope   ca thal  class  \n",
       "0    3.0  0.0  6.0      0  \n",
       "1    2.0  3.0  3.0      2  \n",
       "2    2.0  2.0  7.0      1  \n",
       "3    3.0  0.0  3.0      0  \n",
       "4    1.0  0.0  3.0      0  \n",
       "5    1.0  0.0  3.0      0  \n",
       "6    3.0  2.0  3.0      3  \n",
       "7    1.0  0.0  3.0      0  \n",
       "8    2.0  1.0  7.0      2  \n",
       "9    3.0  0.0  7.0      1  "
      ]
     },
     "execution_count": 41,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.preprocessing import StandardScaler\n",
    "from sklearn.metrics import roc_auc_score\n",
    "\n",
    "from numpy.random import default_rng\n",
    "from scipy import linalg as la\n",
    "from scipy.special import softmax\n",
    "from scipy.stats import chi2\n",
    "from sklearn.linear_model import LogisticRegression\n",
    "from lib_fast_solve import * \n",
    "\n",
    "# import the heart disease dataset\n",
    "dataset = \"http://archive.ics.uci.edu/ml/machine-learning-databases/heart-disease/processed.cleveland.data\"\n",
    "column_names =  ['age','sex','cp','trestbps','chol','fbs','restecg','thalach','exang','oldpeak','slope','ca','thal','class']\n",
    "dataset = pd.read_csv(dataset, names=column_names)\n",
    "\n",
    "# remove missing data with \"?\"\n",
    "df = dataset[~dataset.isin(['?'])]\n",
    "# drop rows with NaN values from DataFrame\n",
    "df = df.dropna(axis=0)\n",
    "\n",
    "# look at the data\n",
    "df.head(10)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "age         float64\n",
      "sex         float64\n",
      "cp          float64\n",
      "trestbps    float64\n",
      "chol        float64\n",
      "fbs         float64\n",
      "restecg     float64\n",
      "thalach     float64\n",
      "exang       float64\n",
      "oldpeak     float64\n",
      "slope       float64\n",
      "ca           object\n",
      "thal         object\n",
      "class         int64\n",
      "dtype: object\n"
     ]
    }
   ],
   "source": [
    "# checking data type of the dataframe\n",
    "print (df.dtypes)\n",
    "# transform data to numeric because the columns ca and thal are object datatypes\n",
    "data = df.apply(pd.to_numeric)\n",
    "\n",
    "# create X and Y datasets for training\n",
    "X = np.array(data.iloc[:,0:13])\n",
    "y = np.array(data.iloc[:,-1])\n",
    "\n",
    "# convert to three class classification\n",
    "y[y==3] = 2\n",
    "y[y==4] = 2\n",
    "# plt.hist(y)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 43,
   "metadata": {},
   "outputs": [],
   "source": [
    "clf = LogisticRegression(penalty='none', fit_intercept=False, multi_class=\"multinomial\")\n",
    "\n",
    "n, p = np.shape(X)\n",
    "K = len(np.unique(y)) - 1\n",
    "\n",
    "# useful deterministic matrices: P Q R \n",
    "M = np.eye(K+1) - np.ones((K+1, K+1))/(K+1)\n",
    "Q = la.cholesky(M, lower=True)[:,:K]\n",
    "assert np.allclose(Q @ Q.T, M)\n",
    "assert np.allclose(Q.T @ Q, np.eye(K))\n",
    "\n",
    "P = np.vstack((np.eye(K), -np.ones((1, K))))\n",
    "R = np.vstack((np.eye(K), np.zeros((1, K))))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_P_values(X, y, j):\n",
    "\n",
    "    scaler = StandardScaler()\n",
    "    X = scaler.fit_transform(X) # standardize covariates\n",
    "\n",
    "    n, p = np.shape(X)\n",
    "    K = len(np.unique(y)) - 1\n",
    "\n",
    "    # estimate omega[j,j]\n",
    "    X_j = np.delete(X, j, axis=1)\n",
    "    a = (np.eye(n) - X_j @ la.inv(X_j.T @ X_j) @ X_j.T) @ X[:,j]\n",
    "    wj = (n - p + 1) / la.norm(a)**2\n",
    "\n",
    "    Y = np.eye(K+1)[y]\n",
    "    clf.fit(X, y)\n",
    "    B_bar = clf.coef_.T # shape (p, K)\n",
    "    A_hat = B_bar @ P # shape (p, K)\n",
    "\n",
    "    # old test statistics \n",
    "    prob = clf.predict_proba(X) # shape (n, K+1)\n",
    "    E_Hbar = np.mean(\n",
    "            prob[:, :, np.newaxis] * np.eye(K+1)[np.newaxis, :, :]\n",
    "            - prob[:,:, np.newaxis] * prob[:, np.newaxis, :], axis=0)\n",
    "    cov_old = wj * la.inv(R.T @ E_Hbar @ R) \n",
    "    chi2_old = n * A_hat[j,:] @ la.inv(cov_old) @ A_hat.T[:,j]\n",
    "    \n",
    "    # new test statistics\n",
    "    prob_hat = clf.predict_proba(X) # shape (n, K+1)\n",
    "    G_bar = prob_hat - Y # shape (n, K+1)\n",
    "    \n",
    "    if (p*(K+1) <= n+p):\n",
    "        V_bar = compute_V_solve(X=X, pi=prob_hat, checks=False)\n",
    "    else:\n",
    "        V_bar = compute_V_Woodberry(X=X, pi=prob_hat, checks=False)\n",
    "    \n",
    "    chi2_new = (wj)**(-1) * A_hat[j,:] @ R.T @ V_bar @ la.pinvh(G_bar.T @ G_bar) @ V_bar @ R @ A_hat.T[:,j]\n",
    "\n",
    "    p_values = 1 - chi2.cdf([chi2_old, chi2_new], df=K, loc=0, scale=1)\n",
    "    return p_values"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([0.0734, 0.0508])"
      ]
     },
     "execution_count": 48,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from numpy.random import default_rng\n",
    "\n",
    "rep = 10000\n",
    "p_values = np.zeros((rep, 2))\n",
    "for i in range(rep):\n",
    "    rng = default_rng(i) \n",
    "    X_fake = np.concatenate((X, rng.normal(size = (n, 1))), axis=1)\n",
    "    p_values[i] = compute_P_values(X_fake, y, p)\n",
    "\n",
    "flag = (p_values <= 0.05) + 0\n",
    "np.mean(flag, axis=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "metadata": {},
   "outputs": [],
   "source": [
    "# array([0.0734, 0.0508]) # rep = 10000\n",
    "# array([0.07419, 0.04968]) # rep = 100000\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x17f510f50>"
      ]
     },
     "execution_count": 47,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAjkAAAGdCAYAAADwjmIIAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjYuMCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy89olMNAAAACXBIWXMAAA9hAAAPYQGoP6dpAAA0O0lEQVR4nO3deXQV9f3/8VcWsgBZCDRbDRCRJSAIBMGIbDWHIJGC0gKSCmogWpNWSMtWIGwqGNmRQnEB/H5BkH6FKlAkDSIFAoRAFCECSlhaTdIKSQSEJGR+f3gyP68sGrxJyIfn45w5xzvzns985gN6X35m5o6LZVmWAAAADONa0x0AAACoCoQcAABgJEIOAAAwEiEHAAAYiZADAACMRMgBAABGIuQAAAAjEXIAAICR3Gu6AzWpvLxcX3zxhXx8fOTi4lLT3QEAAD+CZVn6+uuvFRoaKlfX68/X3NYh54svvlBYWFhNdwMAANyEM2fO6I477rju9ts65Pj4+Ej6dpB8fX1ruDcAAODHKC4uVlhYmP09fj23dcipuETl6+tLyAEAoJb5oVtNuPEYAAAYiZADAACMRMgBAABGuq3vyQEAmMmyLJWVlenKlSs13RXcBDc3N7m7u//kn3ch5AAAjFJSUqIvv/xSFy9erOmu4CeoW7euQkJC5OHhcdNtEHIAAMYoLy9Xbm6u3NzcFBoaKg8PD37stZaxLEslJSX6z3/+o9zcXDVv3vyGP/h3I4QcAIAxSkpKVF5errCwMNWtW7emu4Ob5O3trTp16ujUqVMqKSmRl5fXTbXDjccAAOPc7P/549bhjD9D/hYAAAAjEXIAAICRuCcHAGC8puM3VevxTs6KdV5bJ08qPDxcBw8eVPv27Z3W7rWsWLFCo0aNUmFhoVPa2759u3r16qVz587J39/fKW1WBjM5AABAkjR48GAdO3asprvhNMzkAAAASd8+1eTt7V3T3XAaZnIAALgFlJeXKzU1VXfddZc8PT3VuHFjvfDCC1fVXblyRfHx8QoPD5e3t7datmypBQsWONRs375dnTt3Vr169eTv76+uXbvq1KlTkqSPPvpIvXr1ko+Pj3x9fRUZGan9+/dL+vZy1fcvK7333nu699575eXlpUaNGumRRx6xt/3P//yPOnXqJB8fHwUHB2vo0KEqKChw8sjcPGZyqogzr/8689ouAODWNGHCBL366quaN2+eHnjgAX355Zf69NNPr6orLy/XHXfcoXXr1qlhw4bavXu3EhISFBISokGDBqmsrEwDBgzQyJEj9dZbb6mkpET79u2zfxQxLi5OHTp00JIlS+Tm5qbs7GzVqVPnmn3atGmTHnnkEU2cOFFvvvmmSkpKtHnzZnt7aWmpZsyYoZYtW6qgoEDJycl64oknHGpqEiEHAIAa9vXXX2vBggV65ZVXNHz4cElSs2bN9MADD+jkyZMOtXXq1NG0adPsz+Hh4crIyNDbb7+tQYMGqbi4WEVFRXr44YfVrFkzSVJERIRdf/r0aY0ZM0atWrWSJDVv3vy6/XrhhRc0ZMgQh+Pdc8899j8/9dRT9j/feeedWrhwoe69916dP39e9evXv4mRcC4uVwEAUMNycnJ0+fJlPfjggz+qfvHixYqMjNTPfvYz1a9fX8uWLdPp06clSQEBAXriiScUExOjfv36acGCBfryyy/tfZOTkzVixAhFR0dr1qxZ+vzzz697nOzs7Bv2KSsrS/369VPjxo3l4+OjHj16SJLdl5pGyAEAoIZV5mbfNWvW6I9//KPi4+O1detWZWdn68knn1RJSYlds3z5cmVkZOj+++/X2rVr1aJFC+3Zs0eSNHXqVB0+fFixsbHatm2bWrdurfXr11e6XxcuXFBMTIx8fX21atUqZWZm2u18ty81iZADAEANa968uby9vZWenv6Dtbt27dL999+vZ599Vh06dNBdd911zdmYDh06aMKECdq9e7fuvvturV692t7WokULjR49Wlu3btWjjz6q5cuXX/NY7dq1u26fPv30U3311VeaNWuWunXrplatWt1SNx1LhBwAAGqcl5eXxo0bp7Fjx+rNN9/U559/rj179uj111+/qrZ58+bav3+/3n//fR07dkyTJ09WZmamvT03N1cTJkxQRkaGTp06pa1bt+r48eOKiIjQN998o6SkJG3fvl2nTp3Srl27lJmZ6XDPzndNmTJFb731lqZMmaKcnBwdOnRIL730kiSpcePG8vDw0KJFi3TixAm9++67mjFjRtUM0E3ixmMAgPFqw1OqkydPlru7u1JSUvTFF18oJCREzzzzzFV1Tz/9tA4ePKjBgwfLxcVFjz32mJ599ln9/e9/lyTVrVtXn376qVauXKmvvvpKISEhSkxM1NNPP62ysjJ99dVXGjZsmPLz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jETIAQAARiLkAAAAIzk95MycOVP33nuvfHx8FBgYqAEDBujo0aMONZcuXVJiYqIaNmyo+vXra+DAgcrPz3eoOX36tGJjY1W3bl0FBgZqzJgxKisrc6jZvn27OnbsKE9PT911111asWKFs08HAADUUk4POR9++KESExO1Z88epaWlqbS0VL1799aFCxfsmtGjR+u9997TunXr9OGHH+qLL77Qo48+am+/cuWKYmNjVVJSot27d2vlypVasWKFUlJS7Jrc3FzFxsaqV69eys7O1qhRozRixAi9//77zj4lAABQC7k7u8EtW7Y4fF6xYoUCAwOVlZWl7t27q6ioSK+//rpWr16tX/ziF5Kk5cuXKyIiQnv27NF9992nrVu36siRI/rHP/6hoKAgtW/fXjNmzNC4ceM0depUeXh4aOnSpQoPD9ecOXMkSREREdq5c6fmzZunmJgYZ58WAACoZar8npyioiJJUkBAgCQpKytLpaWlio6OtmtatWqlxo0bKyMjQ5KUkZGhtm3bKigoyK6JiYlRcXGxDh8+bNd8t42Kmoo2ruXy5csqLi52WAAAgJmqNOSUl5dr1KhR6tq1q+6++25JUl5enjw8POTv7+9QGxQUpLy8PLvmuwGnYnvFthvVFBcX65tvvrlmf2bOnCk/Pz97CQsL+8nnCAAAbk1VGnISExP1ySefaM2aNVV5mB9twoQJKioqspczZ87UdJcAAEAVcfo9ORWSkpK0ceNG7dixQ3fccYe9Pjg4WCUlJSosLHSYzcnPz1dwcLBds2/fPof2Kp6++m7N95/Iys/Pl6+vr7y9va/ZJ09PT3l6ev7kcwMAALc+p8/kWJalpKQkrV+/Xtu2bVN4eLjD9sjISNWpU0fp6en2uqNHj+r06dOKioqSJEVFRenQoUMqKCiwa9LS0uTr66vWrVvbNd9to6Kmog0AAHB7c/pMTmJiolavXq2//e1v8vHxse+h8fPzk7e3t/z8/BQfH6/k5GQFBATI19dXv/vd7xQVFaX77rtPktS7d2+1bt1ajz/+uFJTU5WXl6dJkyYpMTHRnol55pln9Morr2js2LF66qmntG3bNr399tvatGmTs08JAADUQk6fyVmyZImKiorUs2dPhYSE2MvatWvtmnnz5unhhx/WwIED1b17dwUHB+udd96xt7u5uWnjxo1yc3NTVFSUfvOb32jYsGGaPn26XRMeHq5NmzYpLS1N99xzj+bMmaPXXnuNx8cBAICkKpjJsSzrB2u8vLy0ePFiLV68+Lo1TZo00ebNm2/YTs+ePXXw4MFK9xEAAJiPd1cBAAAjEXIAAICRCDkAAMBIhBwAAGAkQg4AADASIQcAABiJkAMAAIxEyAEAAEYi5AAAACMRcgAAgJEIOQAAwEiEHAAAYCRCDgAAMBIhBwAAGImQAwAAjETIAQAARiLkAAAAIxFyAACAkQg5AADASIQcAABgJEIOAAAwEiEHAAAYiZADAACMRMgBAABGIuQAAAAjEXIAAICRCDkAAMBIhBwAAGAkQg4AADASIQcAABiJkAMAAIxEyAEAAEYi5AAAACMRcgAAgJEIOQAAwEiEHAAAYCRCDgAAMBIhBwAAGImQAwAAjETIAQAARiLkAAAAIxFyAACAkWp9yFm8eLGaNm0qLy8vdenSRfv27avpLgEAgFtArQ45a9euVXJysqZMmaIDBw7onnvuUUxMjAoKCmq6awAAoIbV6pAzd+5cjRw5Uk8++aRat26tpUuXqm7dunrjjTdqumsAAKCGudd0B25WSUmJsrKyNGHCBHudq6uroqOjlZGRcc19Ll++rMuXL9ufi4qKJEnFxcVO71/55YtOa6vYxXJSQz98nvSbftPvH3k4+v2DJfSbfv+Yft9cs9+2a1k/0E+rlvr3v/9tSbJ2797tsH7MmDFW586dr7nPlClTLEksLCwsLCwsBixnzpy5YVaotTM5N2PChAlKTk62P5eXl+vs2bNq2LChXFxcbrrd4uJihYWF6cyZM/L19XVGV3EDjHf1YryrF+NdvRjv6uWs8bYsS19//bVCQ0NvWFdrQ06jRo3k5uam/Px8h/X5+fkKDg6+5j6enp7y9PR0WOfv7++0Pvn6+vIvSTVivKsX4129GO/qxXhXL2eMt5+f3w/W1Nobjz08PBQZGan09HR7XXl5udLT0xUVFVWDPQMAALeCWjuTI0nJyckaPny4OnXqpM6dO2v+/Pm6cOGCnnzyyZruGgAAqGG1OuQMHjxY//nPf5SSkqK8vDy1b99eW7ZsUVBQULX2w9PTU1OmTLnqUhiqBuNdvRjv6sV4Vy/Gu3pV93i7WNYPPX8FAABQ+9Tae3IAAABuhJADAACMRMgBAABGIuQAAAAjEXJ+hMWLF6tp06by8vJSly5dtG/fvhvWr1u3Tq1atZKXl5fatm2rzZs3V1NPzVGZMX/11VfVrVs3NWjQQA0aNFB0dPQP/hnBUWX/jldYs2aNXFxcNGDAgKrtoEEqO9aFhYVKTExUSEiIPD091aJFC/6bUkmVHfP58+erZcuW8vb2VlhYmEaPHq1Lly5VU29rrx07dqhfv34KDQ2Vi4uLNmzY8IP7bN++XR07dpSnp6fuuusurVixwrmdcs6bpMy1Zs0ay8PDw3rjjTesw4cPWyNHjrT8/f2t/Pz8a9bv2rXLcnNzs1JTU60jR45YkyZNsurUqWMdOnSomntee1V2zIcOHWotXrzYOnjwoJWTk2M98cQTlp+fn/Wvf/2rmnteO1V2vCvk5uZaP//5z61u3bpZ/fv3r57O1nKVHevLly9bnTp1svr27Wvt3LnTys3NtbZv325lZ2dXc89rr8qO+apVqyxPT09r1apVVm5urvX+++9bISEh1ujRo6u557XP5s2brYkTJ1rvvPOOJclav379DetPnDhh1a1b10pOTraOHDliLVq0yHJzc7O2bNnitD4Rcn5A586drcTERPvzlStXrNDQUGvmzJnXrB80aJAVGxvrsK5Lly7W008/XaX9NEllx/z7ysrKLB8fH2vlypVV1UWj3Mx4l5WVWffff7/12muvWcOHDyfk/EiVHeslS5ZYd955p1VSUlJdXTROZcc8MTHR+sUvfuGwLjk52eratWuV9tM0PybkjB071mrTpo3DusGDB1sxMTFO6weXq26gpKREWVlZio6Otte5uroqOjpaGRkZ19wnIyPDoV6SYmJirlsPRzcz5t938eJFlZaWKiAgoKq6aYybHe/p06crMDBQ8fHx1dFNI9zMWL/77ruKiopSYmKigoKCdPfdd+vFF1/UlStXqqvbtdrNjPn999+vrKws+5LWiRMntHnzZvXt27da+nw7qY7vy1r9i8dV7b///a+uXLly1S8oBwUF6dNPP73mPnl5edesz8vLq7J+muRmxvz7xo0bp9DQ0Kv+5cHVbma8d+7cqddff13Z2dnV0ENz3MxYnzhxQtu2bVNcXJw2b96szz77TM8++6xKS0s1ZcqU6uh2rXYzYz506FD997//1QMPPCDLslRWVqZnnnlGf/rTn6qjy7eV631fFhcX65tvvpG3t/dPPgYzOTDKrFmztGbNGq1fv15eXl413R3jfP3113r88cf16quvqlGjRjXdHeOVl5crMDBQy5YtU2RkpAYPHqyJEydq6dKlNd01Y23fvl0vvvii/vznP+vAgQN65513tGnTJs2YMaOmu4abwEzODTRq1Ehubm7Kz893WJ+fn6/g4OBr7hMcHFypeji6mTGvMHv2bM2aNUv/+Mc/1K5du6rspjEqO96ff/65Tp48qX79+tnrysvLJUnu7u46evSomjVrVrWdrqVu5u92SEiI6tSpIzc3N3tdRESE8vLyVFJSIg8Pjyrtc213M2M+efJkPf744xoxYoQkqW3btrpw4YISEhI0ceJEuboyN+As1/u+9PX1dcosjsRMzg15eHgoMjJS6enp9rry8nKlp6crKirqmvtERUU51EtSWlradevh6GbGXJJSU1M1Y8YMbdmyRZ06daqOrhqhsuPdqlUrHTp0SNnZ2fbyy1/+Ur169VJ2drbCwsKqs/u1ys383e7atas+++wzO0hK0rFjxxQSEkLA+RFuZswvXrx4VZCpCJkWr3p0qmr5vnTaLcyGWrNmjeXp6WmtWLHCOnLkiJWQkGD5+/tbeXl5lmVZ1uOPP26NHz/ert+1a5fl7u5uzZ4928rJybGmTJnCI+SVVNkxnzVrluXh4WH99a9/tb788kt7+frrr2vqFGqVyo739/F01Y9X2bE+ffq05ePjYyUlJVlHjx61Nm7caAUGBlrPP/98TZ1CrVPZMZ8yZYrl4+NjvfXWW9aJEyesrVu3Ws2aNbMGDRpUU6dQa3z99dfWwYMHrYMHD1qSrLlz51oHDx60Tp06ZVmWZY0fP956/PHH7fqKR8jHjBlj5eTkWIsXL+YR8pqwaNEiq3HjxpaHh4fVuXNna8+ePfa2Hj16WMOHD3eof/vtt60WLVpYHh4eVps2baxNmzZVc49rv8qMeZMmTSxJVy1Tpkyp/o7XUpX9O/5dhJzKqexY79692+rSpYvl6elp3XnnndYLL7xglZWVVXOva7fKjHlpaak1depUq1mzZpaXl5cVFhZmPfvss9a5c+eqv+O1zAcffHDN/xZXjO/w4cOtHj16XLVP+/btLQ8PD+vOO++0li9f7tQ+uVgW828AAMA83JMDAACMRMgBAABGIuQAAAAjEXIAAICRCDkAAMBIhBwAAGAkQg4AADASIQcAABiJkAMAAIxEyAEAAEYi5AAAACMRcgAAgJH+H4NkdfBwZO/2AAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# plt.hist(p_values, label= [\"low\", 'ss'])\n",
    "colors = ['classical', 'proposed']\n",
    "plt.hist(p_values, bins=10, label=colors)\n",
    "plt.legend(prop={'size': 10})"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.11.0"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
