{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "d5a227e0",
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2\n",
    "\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "from scipy import optimize as opt\n",
    "import os\n",
    "from pathlib import Path\n",
    "import warnings\n",
    "import pickle\n",
    "import re\n",
    "from tqdm import tqdm\n",
    "from functools import partial\n",
    "\n",
    "from mbi import Dataset, Domain, LinearMeasurement, estimation\n",
    "\n",
    "import src.mst as mst\n",
    "import src.brownian_mechanism as bm\n",
    "from src.basic_mechanisms import gaussian_mechanism_rdp_variance\n",
    "from src.measurement_wrapper import MeasurementWrapper\n",
    "\n",
    "from sklearn.ensemble import GradientBoostingClassifier, RandomForestClassifier\n",
    "from sklearn.linear_model import LogisticRegression\n",
    "from sklearn.preprocessing import StandardScaler\n",
    "from sklearn.pipeline import make_pipeline\n",
    "from sklearn.model_selection import train_test_split\n",
    "from sklearn.base import clone\n",
    "from sklearn.metrics import brier_score_loss\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "from tueplots import figsizes, fontsizes, fonts"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "b91fc4f8",
   "metadata": {},
   "outputs": [],
   "source": [
    "result_base_path = \"results/accuracy-first/adult\"\n",
    "syn_df_base_path = \"results/accuracy-first/adult/synthetic-data\"\n",
    "cliques_file_path = \"results/accuracy-first/adult/cliques.p\""
   ]
  },
  {
   "cell_type": "markdown",
   "id": "34506c3f",
   "metadata": {},
   "source": [
    "# Data Loading"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "b8bbb600",
   "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>workclass</th>\n",
       "      <th>education</th>\n",
       "      <th>marital-status</th>\n",
       "      <th>occupation</th>\n",
       "      <th>relationship</th>\n",
       "      <th>race</th>\n",
       "      <th>gender</th>\n",
       "      <th>capital-gain</th>\n",
       "      <th>capital-loss</th>\n",
       "      <th>hours-per-week</th>\n",
       "      <th>native-country</th>\n",
       "      <th>income</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "      <td>4</td>\n",
       "      <td>6</td>\n",
       "      <td>3</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>3</td>\n",
       "      <td>38</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>2</td>\n",
       "      <td>2</td>\n",
       "      <td>11</td>\n",
       "      <td>2</td>\n",
       "      <td>4</td>\n",
       "      <td>0</td>\n",
       "      <td>4</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>4</td>\n",
       "      <td>38</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>1</td>\n",
       "      <td>1</td>\n",
       "      <td>7</td>\n",
       "      <td>2</td>\n",
       "      <td>10</td>\n",
       "      <td>0</td>\n",
       "      <td>4</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>3</td>\n",
       "      <td>38</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>3</td>\n",
       "      <td>2</td>\n",
       "      <td>15</td>\n",
       "      <td>2</td>\n",
       "      <td>6</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>1</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>3</td>\n",
       "      <td>38</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>2</td>\n",
       "      <td>2</td>\n",
       "      <td>0</td>\n",
       "      <td>4</td>\n",
       "      <td>7</td>\n",
       "      <td>1</td>\n",
       "      <td>4</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>2</td>\n",
       "      <td>38</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>...</th>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>45217</th>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>7</td>\n",
       "      <td>2</td>\n",
       "      <td>12</td>\n",
       "      <td>5</td>\n",
       "      <td>4</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>3</td>\n",
       "      <td>38</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>45218</th>\n",
       "      <td>2</td>\n",
       "      <td>2</td>\n",
       "      <td>11</td>\n",
       "      <td>2</td>\n",
       "      <td>6</td>\n",
       "      <td>0</td>\n",
       "      <td>4</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>3</td>\n",
       "      <td>38</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>45219</th>\n",
       "      <td>4</td>\n",
       "      <td>2</td>\n",
       "      <td>11</td>\n",
       "      <td>6</td>\n",
       "      <td>0</td>\n",
       "      <td>4</td>\n",
       "      <td>4</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>3</td>\n",
       "      <td>38</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>45220</th>\n",
       "      <td>1</td>\n",
       "      <td>2</td>\n",
       "      <td>11</td>\n",
       "      <td>4</td>\n",
       "      <td>0</td>\n",
       "      <td>3</td>\n",
       "      <td>4</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>38</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>45221</th>\n",
       "      <td>4</td>\n",
       "      <td>3</td>\n",
       "      <td>11</td>\n",
       "      <td>2</td>\n",
       "      <td>3</td>\n",
       "      <td>5</td>\n",
       "      <td>4</td>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>3</td>\n",
       "      <td>38</td>\n",
       "      <td>1</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "<p>45222 rows × 13 columns</p>\n",
       "</div>"
      ],
      "text/plain": [
       "       age  workclass  education  marital-status  occupation  relationship  \\\n",
       "0        1          2          1               4           6             3   \n",
       "1        2          2         11               2           4             0   \n",
       "2        1          1          7               2          10             0   \n",
       "3        3          2         15               2           6             0   \n",
       "4        2          2          0               4           7             1   \n",
       "...    ...        ...        ...             ...         ...           ...   \n",
       "45217    1          2          7               2          12             5   \n",
       "45218    2          2         11               2           6             0   \n",
       "45219    4          2         11               6           0             4   \n",
       "45220    1          2         11               4           0             3   \n",
       "45221    4          3         11               2           3             5   \n",
       "\n",
       "       race  gender  capital-gain  capital-loss  hours-per-week  \\\n",
       "0         2       1             0             0               3   \n",
       "1         4       1             0             0               4   \n",
       "2         4       1             0             0               3   \n",
       "3         2       1             1             0               3   \n",
       "4         4       1             0             0               2   \n",
       "...     ...     ...           ...           ...             ...   \n",
       "45217     4       0             0             0               3   \n",
       "45218     4       1             0             0               3   \n",
       "45219     4       0             0             0               3   \n",
       "45220     4       1             0             0               1   \n",
       "45221     4       0             1             0               3   \n",
       "\n",
       "       native-country  income  \n",
       "0                  38       0  \n",
       "1                  38       0  \n",
       "2                  38       1  \n",
       "3                  38       1  \n",
       "4                  38       0  \n",
       "...               ...     ...  \n",
       "45217              38       0  \n",
       "45218              38       1  \n",
       "45219              38       0  \n",
       "45220              38       0  \n",
       "45221              38       1  \n",
       "\n",
       "[45222 rows x 13 columns]"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df = pd.read_csv(\"datasets/adult-preprocessed.csv\", index_col=False)\n",
    "df"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "c9050bf7",
   "metadata": {},
   "outputs": [],
   "source": [
    "def X_y_split(df):\n",
    "    X = df.drop(columns=[\"income\"])\n",
    "    y = df[\"income\"]\n",
    "    return X, y"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "506e698e",
   "metadata": {},
   "outputs": [],
   "source": [
    "np.random.seed(54787529)\n",
    "\n",
    "y = df[\"income\"]\n",
    "train_df, test_validation_df = train_test_split(df, test_size=0.6)\n",
    "test_df, validation_df = train_test_split(test_validation_df, test_size=2/3, stratify=test_validation_df[\"income\"])\n",
    "\n",
    "domain_sizes = {col_name: df[col_name].nunique() for col_name in df.columns}\n",
    "domain = Domain.fromdict(domain_sizes)\n",
    "real_dataset = Dataset(train_df, domain)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "14bcf6b0",
   "metadata": {},
   "outputs": [],
   "source": [
    "train_X, train_y = X_y_split(train_df)\n",
    "test_X, test_y = X_y_split(test_df)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "802a44d4",
   "metadata": {},
   "outputs": [],
   "source": [
    "def save_cliques(filename, cliques):\n",
    "    with open(filename, \"wb\") as file:\n",
    "        pickle.dump(cliques, file)\n",
    "\n",
    "def load_cliques(filename):\n",
    "    with open(filename, \"rb\") as file:\n",
    "        return pickle.load(filename)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3f838187",
   "metadata": {},
   "source": [
    "# Privacy Calculations"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "bffc523b",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy Checking Variance Split: 0.13225482076512135\n",
      "Accuracy Checking Variance: 0.01806367606844982\n"
     ]
    }
   ],
   "source": [
    "alpha = 20.0\n",
    "epsilon_query_selection = 0.01\n",
    "epsilon_generation_list = list(np.logspace(-2, 0, 7)) # 0.01 to 1\n",
    "epsilon_accuracy_checking = 0.01\n",
    "accuracy_threshold = 0.825\n",
    "\n",
    "utility_sensitivity = 1 / validation_df.shape[0]\n",
    "\n",
    "def accuracy_check_noise_variance(split):\n",
    "    sigma_1, b = bm.calculate_mechanism_scales_for_svt_gauss_laplace(epsilon_accuracy_checking, split, alpha, utility_sensitivity)\n",
    "    return (sigma_1**2 + 2 * b**2)**0.5\n",
    "\n",
    "res = opt.minimize_scalar(accuracy_check_noise_variance, bounds=(0, 1))\n",
    "\n",
    "accuracy_checking_variance_split = res.x # type: ignore\n",
    "print(f\"Accuracy Checking Variance Split: {accuracy_checking_variance_split}\")\n",
    "print(f\"Accuracy Checking Variance: {res.fun}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "8a1b0025",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.00428190530260632"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "bm.calculate_mechanism_scale_no_svt(epsilon_accuracy_checking, utility_sensitivity, len(epsilon_generation_list), alpha)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "104d681d",
   "metadata": {},
   "outputs": [],
   "source": [
    "def rdp_to_adp_conversion(rdp_eps, alpha, delta):\n",
    "    return rdp_eps + np.log(1 / delta) / (alpha - 1)\n",
    "\n",
    "rdp_eps_col = \"RDP Epsilon @ $\\\\alpha = 20$\"\n",
    "eps_conversion_df = pd.DataFrame({rdp_eps_col: epsilon_generation_list})\n",
    "# eps_conversion_df[\"Total Epsilon (RDP)\"] = eps_conversion_df[rdp_eps_col] + epsilon_query_selection\n",
    "eps_conversion_df[\"ADP Epsilon @ $\\\\delta = 10^{-5}$\"] = rdp_to_adp_conversion(eps_conversion_df[rdp_eps_col], alpha, delta=1e-5)\n",
    "\n",
    "eps_conversion_df.to_latex(\"figures/adult/epsilon_conversion_table.tex\", index=False)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "930af96d",
   "metadata": {},
   "source": [
    "# Experiment"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "ba950089",
   "metadata": {},
   "outputs": [],
   "source": [
    "class IntermediateResultTracker:\n",
    "    def __init__(self):\n",
    "        self.syn_dfs = []\n",
    "        self.query_releases = []\n",
    "        self.trained_models = []\n",
    "        self.validation_accuracies = []\n",
    "        self.test_accuracies = []\n",
    "        self.epsilons = []\n",
    "        self.noise_variances = []\n",
    "\n",
    "    def add_result(self, syn_df, query_release, trained_model, validation_accuracy, test_accuracy, epsilon, noise_variance):\n",
    "        self.syn_dfs.append(syn_df)\n",
    "        self.query_releases.append(query_release)\n",
    "        self.trained_models.append(trained_model)\n",
    "        self.validation_accuracies.append(validation_accuracy)\n",
    "        self.test_accuracies.append(test_accuracy)\n",
    "        self.epsilons.append(epsilon)\n",
    "        self.noise_variances.append(noise_variance)\n",
    "\n",
    "    def save_results(self, filename):\n",
    "        with open(filename, \"wb\") as file:\n",
    "            pickle.dump({\n",
    "                \"syn_dfs\": self.syn_dfs,\n",
    "                \"validation_accuracies\": self.validation_accuracies,\n",
    "                \"test_accuracies\": self.test_accuracies,\n",
    "                \"epsilons\": self.epsilons,\n",
    "                \"noise_variances\": self.noise_variances,\n",
    "            }, file)\n",
    "\n",
    "    @staticmethod\n",
    "    def load_results(filename):\n",
    "        with open(filename, \"rb\") as file:\n",
    "            obj = pickle.load(file)\n",
    "        \n",
    "        res = IntermediateResultTracker()\n",
    "        res.syn_dfs = obj[\"syn_dfs\"]\n",
    "        res.validation_accuracies = obj[\"validation_accuracies\"]\n",
    "        res.test_accuracies = obj[\"test_accuracies\"]\n",
    "        res.epsilons = obj[\"epsilons\"]\n",
    "        res.noise_variances = obj[\"noise_variances\"]\n",
    "        return res\n",
    "        \n",
    "\n",
    "def generate_and_evaluate_synthetic_data(\n",
    "        release_vector: np.ndarray, validation_data: np.ndarray,\n",
    "        release_variance: float, epsilon: float, test_data: np.ndarray, measurement_wrapper, log1, compressed_domain, \n",
    "        undo_compress_fn, intermediate_result_tracker\n",
    "        ) -> float:\n",
    "    log2 = measurement_wrapper.convert_vector_to_measurements(release_vector)\n",
    "    log2 = [LinearMeasurement(m.noisy_measurement, m.clique, release_variance**0.5, m.query) for m in log2]\n",
    "\n",
    "    est = estimation.mirror_descent(compressed_domain, log1+log2, iters=10000)\n",
    "    synth = est.synthetic_data()\n",
    "    synth = undo_compress_fn(synth)\n",
    "\n",
    "    syn_df = synth.df\n",
    "    X_syn, y_syn = X_y_split(syn_df)\n",
    "\n",
    "    model = GradientBoostingClassifier()\n",
    "    model.fit(X_syn, y_syn)\n",
    "    validation_accuracy = model.score(*X_y_split(validation_data))\n",
    "    test_accuracy = model.score(*X_y_split(test_data))\n",
    "\n",
    "    intermediate_result_tracker.add_result(syn_df, log2, model, validation_accuracy, test_accuracy, epsilon, release_variance)\n",
    "\n",
    "    return float(validation_accuracy)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "4693670a",
   "metadata": {},
   "outputs": [],
   "source": [
    "n_repeats = 50\n",
    "threshold_checking_variants = [\"plain-gaussian\", \"svt\"]\n",
    "\n",
    "intermediate_results = {variant: [] for variant in threshold_checking_variants}"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d9739742",
   "metadata": {},
   "source": [
    "## Marginal Query Selection"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "8e6bd8aa",
   "metadata": {},
   "outputs": [],
   "source": [
    "np.random.seed(4234275)\n",
    "rho_selection = epsilon_query_selection / alpha\n",
    "cliques, log1, compressed_data, undo_compress_fn = mst.run_selection(real_dataset, rho_selection)\n",
    "cliques += [(col, \"income\") for col in df.columns if col != \"income\" and (col, \"income\") not in cliques and (\"income\", col) not in cliques]\n",
    "save_cliques(cliques_file_path, cliques)\n",
    "\n",
    "cliques_sensitivity = 1 * np.sqrt(len(cliques)) # add-remove neighbourhood\n",
    "\n",
    "log2 = mst.measure(compressed_data, cliques, sigma=0)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e152d195",
   "metadata": {},
   "source": [
    "## Brownian Mechanism + Synthetic Data Generation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "06b15078",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Repeat 1 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 2 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 3 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 4 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 5 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 6 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 7 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 8 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 9 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 10 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 11 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 12 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 13 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 14 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 15 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 16 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 17 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 18 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 19 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 20 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 21 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 22 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 23 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 24 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 25 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 26 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 27 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 28 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 29 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 30 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 31 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 32 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 33 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 34 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 35 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 36 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 37 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 38 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 39 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 40 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 41 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 42 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 43 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 44 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 45 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 46 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 47 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 48 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 49 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n",
      "Repeat 50 / 50\n",
      "    Variant plain-gaussian\n",
      "    Variant svt\n"
     ]
    }
   ],
   "source": [
    "np.random.seed(69674586)\n",
    "warnings.simplefilter(\"ignore\", FutureWarning, 102)\n",
    "for repeat_ind in range(n_repeats):\n",
    "    print(f\"Repeat {repeat_ind + 1} / {n_repeats}\")\n",
    "\n",
    "    measurement_wrapper = MeasurementWrapper(log2)\n",
    "    real_query_values = measurement_wrapper.get_concatenated_values()\n",
    "\n",
    "    for variant in threshold_checking_variants:\n",
    "        print(f\"    Variant {variant}\")\n",
    "\n",
    "        intermediate_result_tracker = IntermediateResultTracker()\n",
    "        utility_function = partial(\n",
    "            generate_and_evaluate_synthetic_data, \n",
    "            test_data=test_df,\n",
    "            measurement_wrapper=measurement_wrapper, log1=log1, \n",
    "            compressed_domain=compressed_data.domain, \n",
    "            undo_compress_fn=undo_compress_fn, \n",
    "            intermediate_result_tracker=intermediate_result_tracker\n",
    "        )\n",
    "\n",
    "        if variant == \"svt\":\n",
    "            final_values, final_utility = bm.run_until_private_utility_above_threshold_gauss_laplace(\n",
    "                real_query_values, validation_df, utility_function, accuracy_threshold,\n",
    "                epsilon_generation_list, epsilon_accuracy_checking, \n",
    "                accuracy_checking_variance_split, alpha, cliques_sensitivity,\n",
    "                utility_sensitivity\n",
    "            )\n",
    "        elif variant == \"plain-gaussian\":\n",
    "            final_values, final_utility = bm.run_until_private_utility_above_threshold_no_svt(\n",
    "                real_query_values, validation_df, utility_function, accuracy_threshold,\n",
    "                epsilon_accuracy_checking, utility_sensitivity, epsilon_generation_list,\n",
    "                alpha, cliques_sensitivity,\n",
    "            )\n",
    "        else:\n",
    "            raise RuntimeError(f\"Unknown variant {variant}\")\n",
    "\n",
    "        intermediate_results[variant].append(intermediate_result_tracker)\n",
    "        intermediate_result_tracker.save_results(f\"{result_base_path}/repeat_{repeat_ind}_{variant}.p\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "2d67b367",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Load saved results:\n",
    "for variant in threshold_checking_variants:\n",
    "    for repeat_ind in range(n_repeats):\n",
    "        filename = f\"{result_base_path}/repeat_{repeat_ind}_{variant}.p\"\n",
    "        intermediate_results[variant].append(IntermediateResultTracker.load_results(filename))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "a358abc9",
   "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>Epsilon</th>\n",
       "      <th>Validation Accuracy</th>\n",
       "      <th>Test Accuracy</th>\n",
       "      <th>repeat_ind</th>\n",
       "      <th>Variant</th>\n",
       "      <th>Is Final</th>\n",
       "      <th>Epsilon Rounded</th>\n",
       "      <th>Status</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>0.010000</td>\n",
       "      <td>0.822167</td>\n",
       "      <td>0.825630</td>\n",
       "      <td>0</td>\n",
       "      <td>plain-gaussian</td>\n",
       "      <td>True</td>\n",
       "      <td>0.010</td>\n",
       "      <td>Accepted Output</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>0.010000</td>\n",
       "      <td>0.808513</td>\n",
       "      <td>0.809708</td>\n",
       "      <td>1</td>\n",
       "      <td>plain-gaussian</td>\n",
       "      <td>False</td>\n",
       "      <td>0.010</td>\n",
       "      <td>Rejected Output</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>0.021544</td>\n",
       "      <td>0.809619</td>\n",
       "      <td>0.809487</td>\n",
       "      <td>1</td>\n",
       "      <td>plain-gaussian</td>\n",
       "      <td>False</td>\n",
       "      <td>0.022</td>\n",
       "      <td>Rejected Output</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>0.046416</td>\n",
       "      <td>0.821393</td>\n",
       "      <td>0.820212</td>\n",
       "      <td>1</td>\n",
       "      <td>plain-gaussian</td>\n",
       "      <td>True</td>\n",
       "      <td>0.046</td>\n",
       "      <td>Accepted Output</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>0.010000</td>\n",
       "      <td>0.805528</td>\n",
       "      <td>0.804843</td>\n",
       "      <td>2</td>\n",
       "      <td>plain-gaussian</td>\n",
       "      <td>False</td>\n",
       "      <td>0.010</td>\n",
       "      <td>Rejected Output</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>...</th>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>303</th>\n",
       "      <td>0.010000</td>\n",
       "      <td>0.814870</td>\n",
       "      <td>0.813578</td>\n",
       "      <td>48</td>\n",
       "      <td>svt</td>\n",
       "      <td>False</td>\n",
       "      <td>0.010</td>\n",
       "      <td>Rejected Output</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>304</th>\n",
       "      <td>0.021544</td>\n",
       "      <td>0.827253</td>\n",
       "      <td>0.826073</td>\n",
       "      <td>48</td>\n",
       "      <td>svt</td>\n",
       "      <td>False</td>\n",
       "      <td>0.022</td>\n",
       "      <td>Rejected Output</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>305</th>\n",
       "      <td>0.046416</td>\n",
       "      <td>0.825041</td>\n",
       "      <td>0.826736</td>\n",
       "      <td>48</td>\n",
       "      <td>svt</td>\n",
       "      <td>True</td>\n",
       "      <td>0.046</td>\n",
       "      <td>Accepted Output</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>306</th>\n",
       "      <td>0.010000</td>\n",
       "      <td>0.818297</td>\n",
       "      <td>0.818333</td>\n",
       "      <td>49</td>\n",
       "      <td>svt</td>\n",
       "      <td>False</td>\n",
       "      <td>0.010</td>\n",
       "      <td>Rejected Output</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>307</th>\n",
       "      <td>0.021544</td>\n",
       "      <td>0.824323</td>\n",
       "      <td>0.822977</td>\n",
       "      <td>49</td>\n",
       "      <td>svt</td>\n",
       "      <td>True</td>\n",
       "      <td>0.022</td>\n",
       "      <td>Accepted Output</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "<p>308 rows × 8 columns</p>\n",
       "</div>"
      ],
      "text/plain": [
       "      Epsilon  Validation Accuracy  Test Accuracy  repeat_ind         Variant  \\\n",
       "0    0.010000             0.822167       0.825630           0  plain-gaussian   \n",
       "1    0.010000             0.808513       0.809708           1  plain-gaussian   \n",
       "2    0.021544             0.809619       0.809487           1  plain-gaussian   \n",
       "3    0.046416             0.821393       0.820212           1  plain-gaussian   \n",
       "4    0.010000             0.805528       0.804843           2  plain-gaussian   \n",
       "..        ...                  ...            ...         ...             ...   \n",
       "303  0.010000             0.814870       0.813578          48             svt   \n",
       "304  0.021544             0.827253       0.826073          48             svt   \n",
       "305  0.046416             0.825041       0.826736          48             svt   \n",
       "306  0.010000             0.818297       0.818333          49             svt   \n",
       "307  0.021544             0.824323       0.822977          49             svt   \n",
       "\n",
       "     Is Final  Epsilon Rounded           Status  \n",
       "0        True            0.010  Accepted Output  \n",
       "1       False            0.010  Rejected Output  \n",
       "2       False            0.022  Rejected Output  \n",
       "3        True            0.046  Accepted Output  \n",
       "4       False            0.010  Rejected Output  \n",
       "..        ...              ...              ...  \n",
       "303     False            0.010  Rejected Output  \n",
       "304     False            0.022  Rejected Output  \n",
       "305      True            0.046  Accepted Output  \n",
       "306     False            0.010  Rejected Output  \n",
       "307      True            0.022  Accepted Output  \n",
       "\n",
       "[308 rows x 8 columns]"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "records = []\n",
    "\n",
    "for variant in threshold_checking_variants:\n",
    "    for repeat_ind, intermediate_result_tracker in enumerate(intermediate_results[variant]):\n",
    "        for i, (epsilon, validation_accuracy, test_accuracy) in enumerate(zip(intermediate_result_tracker.epsilons, intermediate_result_tracker.validation_accuracies, intermediate_result_tracker.test_accuracies)):\n",
    "            records.append({\n",
    "                \"Epsilon\": epsilon,\n",
    "                \"Validation Accuracy\": validation_accuracy,\n",
    "                \"Test Accuracy\": test_accuracy,\n",
    "                \"repeat_ind\": repeat_ind,\n",
    "                \"Variant\": variant,\n",
    "                \"Is Final\": i == len(intermediate_result_tracker.epsilons) - 1\n",
    "            })\n",
    "\n",
    "result_df = pd.DataFrame.from_records(records)\n",
    "result_df[\"Epsilon Rounded\"] = [round(eps, 3) for eps in result_df.Epsilon]\n",
    "result_df[\"Status\"] = [\"Accepted Output\" if is_final else \"Rejected Output\" for is_final in result_df[\"Is Final\"]]\n",
    "result_df"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3492150f",
   "metadata": {},
   "source": [
    "# No Synthetic Data Baseline"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "baef77fc",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.8423264042459089\n",
      "0.8414593698175787\n"
     ]
    }
   ],
   "source": [
    "np.random.seed(562219)\n",
    "model = GradientBoostingClassifier()\n",
    "model.fit(train_X, train_y)\n",
    "nonsyn_test_accuracy = model.score(test_X, test_y)\n",
    "nonsyn_validation_accuracy = model.score(*X_y_split(validation_df))\n",
    "\n",
    "print(nonsyn_test_accuracy)\n",
    "print(nonsyn_validation_accuracy)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e4a458df",
   "metadata": {},
   "source": [
    "# Non-DP Synthetic Data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "183ce711",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [01:18<00:00,  7.86s/it]\n"
     ]
    }
   ],
   "source": [
    "import mbi.marginal_loss\n",
    "\n",
    "\n",
    "warnings.simplefilter(\"ignore\", FutureWarning, 102)\n",
    "np.random.seed(54673568)\n",
    "nondp_validation_accuracies = []\n",
    "nondp_test_accuracies = []\n",
    "for i in tqdm(range(10)):\n",
    "    log2_nonzero_noise = [LinearMeasurement(m.noisy_measurement, m.clique, 1, m.query) for m in log2]\n",
    "    loss_fn = mbi.marginal_loss.from_linear_measurements(log1 + log2_nonzero_noise)\n",
    "\n",
    "    nondp_pgm_iters = 10000\n",
    "    losses = np.zeros(nondp_pgm_iters)\n",
    "    counter = 0\n",
    "    def callback(x):\n",
    "        global counter\n",
    "        losses[counter] = loss_fn.loss_fn(x)\n",
    "        counter += 1\n",
    "\n",
    "    est = estimation.mirror_descent(compressed_data.domain, log1+log2_nonzero_noise, iters=nondp_pgm_iters)#, callback_fn=callback)\n",
    "\n",
    "    synth = est.synthetic_data()\n",
    "    synth = undo_compress_fn(synth)\n",
    "\n",
    "    syn_df = synth.df\n",
    "    X_syn, y_syn = X_y_split(syn_df)\n",
    "\n",
    "    model = GradientBoostingClassifier()\n",
    "    model.fit(X_syn, y_syn)\n",
    "    nondp_validation_accuracy = model.score(*X_y_split(validation_df))\n",
    "    nondp_test_accuracy = model.score(test_X, test_y)\n",
    "\n",
    "    nondp_validation_accuracies.append(nondp_validation_accuracy)\n",
    "    nondp_test_accuracies.append(nondp_test_accuracy)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 54,
   "id": "7820cdfb",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.8292205638474297\n",
      "0.8286045997346306\n",
      "\n",
      "0.001353199506749182\n",
      "0.0010962108943568503\n"
     ]
    }
   ],
   "source": [
    "nondp_validation_accuracy = np.array(nondp_validation_accuracies).mean()\n",
    "nondp_test_accuracy = np.array(nondp_test_accuracies).mean()\n",
    "print(nondp_validation_accuracy)\n",
    "print(nondp_test_accuracy)\n",
    "print()\n",
    "print(np.array(nondp_validation_accuracies).std())\n",
    "print(np.array(nondp_test_accuracies).std())"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "07356363",
   "metadata": {},
   "source": [
    "# Plotting"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "2309e492",
   "metadata": {},
   "outputs": [],
   "source": [
    "figsize_factory = figsizes.icml2024_full\n",
    "plt.rcParams.update(fontsizes.icml2024())\n",
    "plt.rcParams.update(fonts.icml2024_tex())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "86d8e5d6",
   "metadata": {},
   "outputs": [],
   "source": [
    "variant_names = {\n",
    "    \"svt\": \"SVT\",\n",
    "    \"plain-gaussian\": \"Plain Gaussian\"\n",
    "}\n",
    "hue_order = [\"Plain Gaussian\", \"SVT\"]\n",
    "\n",
    "eps_label = \"Epsilon (RDP, $\\\\alpha = 20$)\"\n",
    "variant_colors={\n",
    "    \"Plain Gaussian\": \"tan\",\n",
    "    \"SVT\": \"turquoise\",\n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 158,
   "id": "0d940280",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 275x169.959 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "final_result_df = result_df[result_df[\"Is Final\"]].copy()\n",
    "final_result_df[\"Variant\"] = [variant_names[v] for v in final_result_df.Variant]\n",
    "final_epsilon_count_df = final_result_df.groupby(\"Variant\").value_counts(subset=[\"Epsilon Rounded\"], normalize=True).reset_index()\n",
    "\n",
    "fig, ax = plt.subplots(1, 1, figsize=figsize_factory(nrows=1, ncols=1, rel_width=0.5)[\"figure.figsize\"])\n",
    "sns.barplot(data=final_epsilon_count_df, x=\"Epsilon Rounded\", y=\"proportion\", hue=\"Variant\", hue_order=hue_order, palette=variant_colors, ax=ax)\n",
    "ax.set_ylabel(\"Proportion\")\n",
    "ax.set_xlabel(eps_label)\n",
    "ax.set_title(\"Distribution of Accepted Epsilon\")\n",
    "plt.savefig(\"figures/adult/final-epsilon-distribution.pdf\", bbox_inches=\"tight\")\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 159,
   "id": "80c2591f",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_legend_handles_labels_fix(ax):\n",
    "    handles = ax.legend_.legend_handles\n",
    "    labels = [t.get_text() for t in ax.legend_.get_texts()]\n",
    "    return handles, labels"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 160,
   "id": "653f3247",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 275x169.959 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(1, 1, figsize=figsize_factory(nrows=1, ncols=1, rel_width=0.5)[\"figure.figsize\"])\n",
    "sns.ecdfplot(data=final_result_df, x=\"Validation Accuracy\", hue=\"Variant\", hue_order=hue_order, palette=variant_colors, legend=True, ax=ax)\n",
    "sns_leg_info = get_legend_handles_labels_fix(ax)\n",
    "ax.axvline(accuracy_threshold, color=\"grey\", linestyle=\"dashed\", label=\"Accuracy Threshold\")\n",
    "ax.axvline(nondp_validation_accuracy, color=\"C2\", linestyle=\"dashed\", label=\"Non-DP Validation Accuracy\")\n",
    "leg_info = ax.get_legend_handles_labels()\n",
    "ax.get_legend().remove()\n",
    "ax.legend(sns_leg_info[0] + leg_info[0], sns_leg_info[1] + leg_info[1])\n",
    "ax.set_title(\"eCDF of Accepted Validation Accuracy\")\n",
    "plt.savefig(\"figures/adult/final-validation-accuracy-distribution.pdf\", bbox_inches=\"tight\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 161,
   "id": "149fb915",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 275x169.959 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(1, 1, figsize=figsize_factory(nrows=1, ncols=1, rel_width=0.5)[\"figure.figsize\"])\n",
    "sns.ecdfplot(data=final_result_df, x=\"Test Accuracy\", hue=\"Variant\", hue_order=hue_order, palette=variant_colors, legend=True, ax=ax)\n",
    "sns_leg_info = get_legend_handles_labels_fix(ax)\n",
    "ax.axvline(accuracy_threshold, color=\"grey\", linestyle=\"dashed\", label=\"Accuracy Threshold\")\n",
    "ax.axvline(nondp_test_accuracy, color=\"C2\", linestyle=\"dashed\", label=\"Non-DP Test Accuracy\")\n",
    "leg_info = ax.get_legend_handles_labels()\n",
    "ax.get_legend().remove()\n",
    "ax.legend(sns_leg_info[0] + leg_info[0], sns_leg_info[1] + leg_info[1])\n",
    "ax.set_title(\"eCDF of Accepted Test Accuracy\")\n",
    "plt.savefig(\"figures/adult/final-test-accuracy-distribution.pdf\", bbox_inches=\"tight\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 162,
   "id": "b093b41e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 275x169.959 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(1, 1, figsize=figsize_factory(nrows=1, ncols=1, rel_width=0.5)[\"figure.figsize\"])\n",
    "sns.swarmplot(data=final_result_df, x=\"Epsilon Rounded\", y=\"Validation Accuracy\", hue=\"Variant\", hue_order=hue_order, palette=variant_colors, dodge=True, size=2.6, ax=ax)\n",
    "ax.axhline(accuracy_threshold, color=\"grey\", linestyle=\"dashed\", label=\"Accuracy Threshold\")\n",
    "ax.axhline(nondp_validation_accuracy, color=\"C2\", linestyle=\"dashed\", label=\"Non-DP Validation Accuracy\")\n",
    "leg_info = ax.get_legend_handles_labels()\n",
    "ax.get_legend().remove()\n",
    "ax.legend(*leg_info, loc=\"upper center\", ncol=2, bbox_to_anchor=(0.4, -0.25))\n",
    "ax.set_title(\"Accepted Validation Accuracy by Epsilon\")\n",
    "ax.set_xlabel(eps_label)\n",
    "plt.savefig(\"figures/adult/final-validation-accuracy-by-epsilon.pdf\", bbox_inches=\"tight\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 163,
   "id": "6dfe33a8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 275x169.959 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(1, 1, figsize=figsize_factory(nrows=1, ncols=1, rel_width=0.5)[\"figure.figsize\"])\n",
    "sns.swarmplot(data=final_result_df, x=\"Epsilon Rounded\", y=\"Test Accuracy\", hue=\"Variant\", hue_order=hue_order, palette=variant_colors, dodge=True, size=2.6, ax=ax)\n",
    "ax.axhline(accuracy_threshold, color=\"grey\", linestyle=\"dashed\", label=\"Accuracy Threshold\")\n",
    "ax.axhline(nondp_test_accuracy, color=\"C2\", linestyle=\"dashed\", label=\"Non-DP Test Accuracy\")\n",
    "leg_info = ax.get_legend_handles_labels()\n",
    "ax.get_legend().remove()\n",
    "ax.legend(*leg_info, loc=\"upper center\", ncol=2, bbox_to_anchor=(0.4, -0.25))\n",
    "ax.set_title(\"Accepted Test Accuracy by Epsilon\")\n",
    "ax.set_xlabel(eps_label)\n",
    "plt.savefig(\"figures/adult/final-test-accuracy-by-epsilon.pdf\", bbox_inches=\"tight\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 164,
   "id": "0d4215c2",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 275x169.959 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(1, 1, figsize=figsize_factory(nrows=1, ncols=1, rel_width=0.5)[\"figure.figsize\"])\n",
    "sns.scatterplot(data=final_result_df, x=\"Validation Accuracy\", y=\"Test Accuracy\", hue=\"Variant\", palette=variant_colors, s=7.5, ax=ax)\n",
    "ax.plot((0.805, 0.835), (0.805, 0.835), color=\"grey\", linestyle=\"dashed\")\n",
    "ax.set_title(\"Accepted Validation vs. Test Accuracy\")\n",
    "plt.savefig(\"figures/adult/final-test-validation-accuracy-scatter.pdf\", bbox_inches=\"tight\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "a3c3a491",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 675x168.75 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, axes = plt.subplots(1, 2, figsize=figsize_factory(nrows=1, ncols=2, height_to_width_ratio=0.5)[\"figure.figsize\"], sharey=True)\n",
    "\n",
    "for i, variant in enumerate(threshold_checking_variants):\n",
    "    ax = axes[i]\n",
    "    variant_df = result_df[result_df.Variant == variant]\n",
    "\n",
    "    sns.swarmplot(data=variant_df, x=\"Epsilon Rounded\", y=\"Validation Accuracy\", hue=\"Status\", hue_order=[\"Rejected Output\", \"Accepted Output\"], ax=ax, legend=True, size=2.6, dodge=True)\n",
    "    ax.axhline(accuracy_threshold, color=\"grey\", linestyle=\"dashed\", label=\"Accuracy Threshold\")\n",
    "    ax.axhline(nondp_validation_accuracy, color=\"C2\", linestyle=\"dashed\", label=\"Non-DP Validation Accuracy\")\n",
    "    ax.set_xlabel(eps_label)\n",
    "    leg_info = ax.get_legend_handles_labels()\n",
    "    ax.get_legend().remove()\n",
    "    ax.set_title(f\"All Releases - {variant_names[variant]}\")\n",
    "\n",
    "fig.legend(*leg_info, ncol=2, loc=\"upper center\", bbox_to_anchor=(0.5, -0.10)) # type: ignore\n",
    "plt.savefig(f\"figures/adult/output-plots.pdf\", bbox_inches=\"tight\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "37012767",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "weakening-dp-bounds",
   "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.12.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
