{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c7392688-0311-471b-90b2-97e2b7833650",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "from torch.utils.data import DataLoader, TensorDataset, Subset\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "from sklearn.preprocessing import MinMaxScaler, OneHotEncoder\n",
    "from sklearn.compose import ColumnTransformer\n",
    "from sklearn.ensemble import RandomForestClassifier\n",
    "from sklearn.metrics import roc_curve\n",
    "import matplotlib.pyplot as plt\n",
    "from opacus import PrivacyEngine\n",
    "import warnings\n",
    "import os\n",
    "import requests\n",
    "from scipy.stats import norm\n",
    "from concurrent.futures import ProcessPoolExecutor, as_completed\n",
    "from tqdm import tqdm\n",
    "import multiprocessing as mp\n",
    "from torchvision import datasets, transforms\n",
    "\n",
    "device = 'cuda'\n",
    "C_GAMMA = 1.0\n",
    "USE_PRV_ACCOUNTANT = True\n",
    "\n",
    "# --- 0. Global Parameters ---\n",
    "NUM_SHADOW_MODELS = 20\n",
    "NUM_TEST_RUNS = 1000\n",
    "TARGET_EPSILON = 4.0\n",
    "TARGET_DELTA = 1e-5\n",
    "LATENT_DIM = 4\n",
    "MAX_GRAD_NORM = 1.0\n",
    "EPOCHS = 30\n",
    "SYNTHETIC_SAMPLES = 1000\n",
    "\n",
    "warnings.filterwarnings(\"ignore\")\n",
    "\n",
    "# --- 1. (ε, δ) -> μ Conversion and GDP Tradeoff Function ---\n",
    "def convert_eps_delta_to_mu(epsilon, delta, *, tol=1e-12, max_iter=200):\n",
    "    if epsilon < 0:\n",
    "        raise ValueError(\"epsilon must be non-negative\")\n",
    "    if delta <= 0:\n",
    "        return 0.0\n",
    "    if delta >= 1:\n",
    "        return float('inf')\n",
    "\n",
    "    def f(mu):\n",
    "        mu = max(mu, 1e-16)\n",
    "        a = -epsilon / mu\n",
    "        return norm.cdf(a + mu/2) - np.exp(epsilon) * norm.cdf(a - mu/2)\n",
    "\n",
    "    low, high = 0.0, 1.0\n",
    "    it = 0\n",
    "    while f(high) < delta and it < 60:\n",
    "        high *= 2.0\n",
    "        it += 1\n",
    "    if f(high) < delta:\n",
    "        return float('inf')\n",
    "\n",
    "    for _ in range(max_iter):\n",
    "        mid = (low + high) / 2\n",
    "        if f(mid) >= delta:\n",
    "            high = mid\n",
    "        else:\n",
    "            low = mid\n",
    "        if high - low <= tol * max(1.0, high):\n",
    "            break\n",
    "    return high\n",
    "\n",
    "def gdp_tradeoff_full(alpha, mu):\n",
    "    alpha = np.asarray(alpha, dtype=float)\n",
    "    LARGE = 12.0\n",
    "    z = np.empty_like(alpha)\n",
    "    mid = (alpha > 0.0) & (alpha < 1.0)\n",
    "    z[mid] = norm.ppf(1.0 - alpha[mid])\n",
    "    z[alpha <= 0.0] = LARGE\n",
    "    z[alpha >= 1.0] = -LARGE\n",
    "    return norm.cdf(z - mu)\n",
    "\n",
    "def gdp_shifted(alpha, mu, gamma):\n",
    "    return np.clip(gdp_tradeoff_full(alpha + gamma, mu) - gamma, 0.0, 1.0)\n",
    "\n",
    "def compute_mu_dp_baseline(target_eps, target_delta):\n",
    "    if USE_PRV_ACCOUNTANT:\n",
    "        try:\n",
    "            from prv_accountant.dpsgd import DPSGDAccountant\n",
    "            acc = DPSGDAccountant(noise_multiplier=1.0,\n",
    "                                  sampling_probability=1.0,\n",
    "                                  eps_error=0.05, delta_error=1e-12, max_steps=1)\n",
    "            _, _, eps_upper = acc.compute_epsilon(delta=target_delta, num_steps=1)\n",
    "            return convert_eps_delta_to_mu(eps_upper, target_delta)\n",
    "        except Exception:\n",
    "            pass\n",
    "    return convert_eps_delta_to_mu(target_eps, target_delta)\n",
    "\n",
    "def estimate_mu_eff_placeholder(n, d, scale=0.7):\n",
    "    mu_base = compute_mu_dp_baseline(TARGET_EPSILON, TARGET_DELTA)\n",
    "    mu_eff = max(1e-8, scale * mu_base * np.sqrt(max(d, 1)) / np.sqrt(max(n, 1)))\n",
    "    return mu_eff\n",
    "\n",
    "# --- 2. Experiment Framework Functions ---\n",
    "\n",
    "def preprocess_adult_dataset():\n",
    "    print(\"Preparing UCI Adult dataset...\")\n",
    "    filename = \"adult.data\"\n",
    "    url = \"https://archive.ics.uci.edu/ml/machine-learning-databases/adult/adult.data\"\n",
    "\n",
    "    if not os.path.exists(filename):\n",
    "        print(f\"'{filename}' not found locally, downloading from the web...\")\n",
    "        try:\n",
    "            res = requests.get(url)\n",
    "            res.raise_for_status()\n",
    "            with open(filename, 'wb') as f:\n",
    "                f.write(res.content)\n",
    "            print(\"Download complete and saved locally.\")\n",
    "        except requests.exceptions.RequestException as e:\n",
    "            print(f\"Download failed: {e}\")\n",
    "            return None\n",
    "    else:\n",
    "        print(f\"Found '{filename}' locally, loading directly.\")\n",
    "\n",
    "    columns = [\n",
    "        \"age\", \"workclass\", \"fnlwgt\", \"education\", \"education-num\",\n",
    "        \"marital-status\", \"occupation\", \"relationship\", \"race\", \"sex\",\n",
    "        \"capital-gain\", \"capital-loss\", \"hours-per-week\", \"native-country\", \"income\"\n",
    "    ]\n",
    "    df = pd.read_csv(filename, header=None, names=columns, na_values=\"?\", sep=r'\\s*,\\s*', engine='python')\n",
    "    df.dropna(inplace=True)\n",
    "    df.drop(\"fnlwgt\", axis=1, inplace=True)\n",
    "    df['income'] = df['income'].apply(lambda x: 1 if x in ['>50K', '>50K.'] else 0)\n",
    "\n",
    "    preprocessor = ColumnTransformer(\n",
    "        transformers=[\n",
    "            ('num', MinMaxScaler(), df.select_dtypes(include=np.number).columns),\n",
    "            ('cat', OneHotEncoder(handle_unknown='ignore', sparse_output=False), df.select_dtypes(include=['object']).columns.drop('income'))\n",
    "        ])\n",
    "    X = df.drop('income', axis=1)\n",
    "    X_processed = preprocessor.fit_transform(X)\n",
    "    print(\"Data preprocessing complete.\")\n",
    "    return torch.tensor(X_processed, dtype=torch.float32)\n",
    "\n",
    "def preprocess_mnist_dataset(subset_size=60000):\n",
    "    print(\"Preparing MNIST dataset...\")\n",
    "    transform = transforms.Compose([\n",
    "        transforms.ToTensor(),\n",
    "        transforms.Lambda(lambda x: torch.flatten(x))\n",
    "    ])\n",
    "\n",
    "    full_dataset = datasets.MNIST(root='../data', train=True, download=True, transform=transform)\n",
    "\n",
    "    if subset_size < len(full_dataset):\n",
    "        indices = np.random.choice(len(full_dataset), subset_size, replace=False)\n",
    "        subset = Subset(full_dataset, indices)\n",
    "    else:\n",
    "        subset = full_dataset\n",
    "\n",
    "    loader = DataLoader(subset, batch_size=len(subset))\n",
    "    all_data_tensor = next(iter(loader))[0]\n",
    "    print(f\"MNIST data preprocessing complete. Dataset shape: {all_data_tensor.shape}\")\n",
    "    return all_data_tensor\n",
    "\n",
    "def create_worst_case_datasets(full_dataset_tensor):\n",
    "    xt_idx = np.random.randint(len(full_dataset_tensor))\n",
    "    xt = full_dataset_tensor[xt_idx].unsqueeze(0)\n",
    "    other_indices = np.random.choice(\n",
    "        [i for i in range(len(full_dataset_tensor)) if i != xt_idx], size=2, replace=False\n",
    "    )\n",
    "    D_prime_tensor = full_dataset_tensor[other_indices]\n",
    "    D_tensor = torch.cat([D_prime_tensor, xt], dim=0)\n",
    "    print(f\"Created worst-case datasets: |D'|={len(D_prime_tensor)}, |D|={len(D_tensor)}\")\n",
    "    return D_tensor, D_prime_tensor, xt\n",
    "\n",
    "class VAE(nn.Module):\n",
    "    def __init__(self, input_dim, latent_dim):\n",
    "        super(VAE, self).__init__()\n",
    "        self.encoder = nn.Sequential(\n",
    "            nn.Linear(input_dim, 128), nn.ReLU(),\n",
    "            nn.Linear(128, latent_dim * 2)\n",
    "        )\n",
    "        self.decoder = nn.Sequential(\n",
    "            nn.Linear(latent_dim, 128), nn.ReLU(),\n",
    "            nn.Linear(128, input_dim), nn.Sigmoid()\n",
    "        )\n",
    "\n",
    "    def reparameterize(self, mu, logvar):\n",
    "        std = torch.exp(0.5 * logvar)\n",
    "        eps = torch.randn_like(std)\n",
    "        return mu + eps * std\n",
    "\n",
    "    def forward(self, x):\n",
    "        h = self.encoder(x.view(x.size(0), -1))\n",
    "        mu, logvar = torch.chunk(h, 2, dim=-1)\n",
    "        z = self.reparameterize(mu, logvar)\n",
    "        return self.decoder(z), mu, logvar\n",
    "\n",
    "def vae_loss_function(recon_x, x, mu, logvar):\n",
    "    BCE = nn.functional.binary_cross_entropy(recon_x, x.view(x.size(0), -1), reduction='sum')\n",
    "    KLD = -0.5 * torch.sum(1 + logvar - mu.pow(2) - logvar.exp())\n",
    "    return BCE + KLD\n",
    "\n",
    "def train_dp_model(dataset_tensor, input_dim):\n",
    "    model = VAE(input_dim, LATENT_DIM).cuda()\n",
    "    optimizer = torch.optim.Adam(model.parameters(), lr=1e-3)\n",
    "    dataloader = DataLoader(TensorDataset(dataset_tensor), batch_size=len(dataset_tensor))\n",
    "\n",
    "    privacy_engine = PrivacyEngine()\n",
    "    model, optimizer, dataloader = privacy_engine.make_private_with_epsilon(\n",
    "        module=model, optimizer=optimizer, data_loader=dataloader,\n",
    "        target_epsilon=TARGET_EPSILON, target_delta=TARGET_DELTA,\n",
    "        max_grad_norm=MAX_GRAD_NORM, epochs=EPOCHS\n",
    "    )\n",
    "\n",
    "    model.train()\n",
    "    for _ in range(EPOCHS):\n",
    "        for (data,) in dataloader:\n",
    "            data = data.to(device)\n",
    "            recon_batch, mu, logvar = model(data)\n",
    "            loss = vae_loss_function(recon_batch, data, mu, logvar)\n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "            optimizer.zero_grad()\n",
    "\n",
    "    model._modules.pop('autograd_grad_sample_hooks', None)\n",
    "    return model\n",
    "\n",
    "def generate_synthetic_data(model):\n",
    "    model.eval()\n",
    "    with torch.no_grad():\n",
    "        z = torch.randn(SYNTHETIC_SAMPLES, LATENT_DIM).to(device)\n",
    "        samples = model.decoder(z)\n",
    "    return samples\n",
    "\n",
    "# ---------- QBS Helper Functions ----------\n",
    "def _to_numpy_float(x):\n",
    "    if isinstance(x, torch.Tensor):\n",
    "        return x.detach().cpu().numpy()\n",
    "    return np.asarray(x)\n",
    "\n",
    "def _binarize01(arr, thr=0.5):\n",
    "    return (arr >= thr).astype(np.int32)\n",
    "\n",
    "def preprocess_dataset_qbs(synth_array01, xt_array01):\n",
    "    return (synth_array01 == xt_array01).astype(np.int32)\n",
    "\n",
    "def get_queries_qbs(d, n_queries=20000, rng=None, k_per_query=5):\n",
    "    if rng is None:\n",
    "        rng = np.random\n",
    "    queries = np.zeros((n_queries, d), dtype=np.int32)\n",
    "    k = max(1, min(k_per_query, d))\n",
    "    for i in range(n_queries):\n",
    "        idx = rng.choice(d, size=k, replace=False)\n",
    "        queries[i, idx] = 1\n",
    "    return queries\n",
    "\n",
    "def extract_query_counts_numpy(X_bin, queries):\n",
    "    hits = X_bin @ queries.T\n",
    "    need = queries.sum(axis=1, keepdims=True)\n",
    "    matches = (hits == need.T)\n",
    "    counts = matches.sum(axis=0).astype(np.int64)\n",
    "    return counts\n",
    "\n",
    "def compute_query_features(synthetic_data, target_record_xt, queries, xt_bin=None):\n",
    "    synth_np = _to_numpy_float(synthetic_data)\n",
    "    xt_np = _to_numpy_float(target_record_xt)\n",
    "\n",
    "    synth01 = _binarize01(synth_np)\n",
    "    xt01 = _binarize01(xt_np) if xt_bin is None else xt_bin\n",
    "    if xt01.ndim == 1:\n",
    "        xt01 = xt01.reshape(1, -1)\n",
    "\n",
    "    X_bin = preprocess_dataset_qbs(synth01, xt01)\n",
    "    feats = extract_query_counts_numpy(X_bin, queries)\n",
    "    return feats.astype(np.float32)\n",
    "\n",
    "def train_attacker(input_dim, D, D_prime, xt, queries, rng_seed=42):\n",
    "    np.random.seed(rng_seed)\n",
    "    torch.manual_seed(rng_seed)\n",
    "\n",
    "    print(f\"Training attacker with {NUM_SHADOW_MODELS} pairs of shadow models...\")\n",
    "    features_list, labels_list = [], []\n",
    "    xt_bin = _binarize01(_to_numpy_float(xt))\n",
    "\n",
    "    for i in tqdm(range(NUM_SHADOW_MODELS), desc=\"Training shadow models\"):\n",
    "        model_d = train_dp_model(D, input_dim)\n",
    "        synth_d = generate_synthetic_data(model_d)\n",
    "        feats_d = compute_query_features(synth_d, xt, queries, xt_bin=xt_bin)\n",
    "        features_list.append(feats_d); labels_list.append(1)\n",
    "\n",
    "        model_d_prime = train_dp_model(D_prime, input_dim)\n",
    "        synth_d_prime = generate_synthetic_data(model_d_prime)\n",
    "        feats_d_prime = compute_query_features(synth_d_prime, xt, queries, xt_bin=xt_bin)\n",
    "        features_list.append(feats_d_prime); labels_list.append(0)\n",
    "\n",
    "    features_arr = np.vstack(features_list)\n",
    "    labels_arr = np.array(labels_list)\n",
    "\n",
    "    attacker_model = RandomForestClassifier(n_estimators=200, random_state=rng_seed, n_jobs=-1)\n",
    "    attacker_model.fit(features_arr, labels_arr)\n",
    "    print(\"Attacker training complete!\")\n",
    "    return attacker_model\n",
    "\n",
    "def run_audit(attacker_model, input_dim, D, D_prime, xt, queries, rng_seed=123):\n",
    "    np.random.seed(rng_seed)\n",
    "    torch.manual_seed(rng_seed)\n",
    "\n",
    "    print(f\"Running audit with {NUM_TEST_RUNS} test runs...\")\n",
    "    member_scores, non_member_scores = [], []\n",
    "    xt_bin = _binarize01(_to_numpy_float(xt))\n",
    "\n",
    "    for i in tqdm(range(NUM_TEST_RUNS), desc=\"Auditing test runs\"):\n",
    "        model_d = train_dp_model(D, input_dim)\n",
    "        feats_in = compute_query_features(generate_synthetic_data(model_d), xt, queries, xt_bin=xt_bin)\n",
    "        member_scores.append(attacker_model.predict_proba(feats_in.reshape(1, -1))[0, 1])\n",
    "\n",
    "        model_d_prime = train_dp_model(D_prime, input_dim)\n",
    "        feats_out = compute_query_features(generate_synthetic_data(model_d_prime), xt, queries, xt_bin=xt_bin)\n",
    "        non_member_scores.append(attacker_model.predict_proba(feats_out.reshape(1, -1))[0, 1])\n",
    "\n",
    "    return member_scores, non_member_scores\n",
    "\n",
    "# --- 3. Final Plotting Function ---\n",
    "def plot_final_curves(member_scores, non_member_scores):\n",
    "    y_true = np.concatenate([np.ones(len(member_scores)), np.zeros(len(non_member_scores))])\n",
    "    y_scores = np.concatenate([member_scores, non_member_scores])\n",
    "    type_I_error, tpr, _ = roc_curve(y_true, y_scores)\n",
    "    type_II_error = 1 - tpr\n",
    "\n",
    "    mu_base = compute_mu_dp_baseline(TARGET_EPSILON, TARGET_DELTA)\n",
    "    print(f\"Baseline μ = {mu_base}\")\n",
    "\n",
    "    n_proxy, d_proxy = 60000, LATENT_DIM\n",
    "    mu_eff = estimate_mu_eff_placeholder(n_proxy, d_proxy, scale=0.8)\n",
    "    gamma = C_GAMMA / max(d_proxy, 1)\n",
    "\n",
    "    alpha = np.linspace(0.0, 1.0, 1201)\n",
    "    beta_base = gdp_tradeoff_full(alpha, mu_base)\n",
    "    beta_ours = gdp_shifted(alpha, mu_eff, gamma)\n",
    "    beta_env = np.maximum(beta_base, beta_ours)\n",
    "\n",
    "    plt.figure(figsize=(10, 8))\n",
    "    plt.plot(alpha, beta_env, color=\"tab:green\", linewidth=2.6, label=\"Envelope max{baseline, ours}\")\n",
    "    plt.plot(alpha, beta_base, \"--\", color=\"tab:orange\", linewidth=2.0, label=f\"Baseline GDP (μ={mu_base:.3f})\")\n",
    "    plt.plot(alpha, beta_ours, \"--\", color=\"tab:blue\", linewidth=2.0, label=f\"Our bound (μ_eff={mu_eff:.3f}, γ={gamma:.3f})\")\n",
    "    plt.plot(type_I_error, type_II_error, 'o-', linewidth=2, markersize=4, color=\"tab:purple\", label='Empirical Tradeoff (VAE + query-based)')\n",
    "\n",
    "    plt.axvline(gamma, color=\"gray\", linestyle=\":\", linewidth=1.0, alpha=0.6)\n",
    "    plt.axvline(1.0 - gamma, color=\"gray\", linestyle=\":\", linewidth=1.0, alpha=0.6)\n",
    "    plt.text(gamma, 0.04, \"γ\", ha=\"center\", va=\"bottom\", fontsize=9, color=\"gray\")\n",
    "    plt.text(1.0 - gamma, 0.04, \"1-γ\", ha=\"center\", va=\"bottom\", fontsize=9, color=\"gray\")\n",
    "\n",
    "    plt.plot([0, 1], [1, 0], 'k:', linewidth=1.2, label='Random Guess')\n",
    "\n",
    "    plt.xlabel('Type I Error (False Positive Rate, $\\\\alpha$)', fontsize=13)\n",
    "    plt.ylabel('Type II Error (False Negative Rate, $\\\\beta$)', fontsize=13)\n",
    "    plt.title('Empirical vs Theoretical (GDP) with Envelope', fontsize=16)\n",
    "    plt.legend(fontsize=11)\n",
    "    plt.grid(True, alpha=0.3)\n",
    "    plt.xlim(0.0, 1.0); plt.ylim(0.0, 1.0)\n",
    "    plt.gca().set_aspect('equal', adjustable='box')\n",
    "    plt.tight_layout()\n",
    "    plt.show()\n",
    "\n",
    "# all_data_tensor = preprocess_mnist_dataset()\n",
    "all_data_tensor = preprocess_adult_dataset()\n",
    "\n",
    "if all_data_tensor is not None:\n",
    "    INPUT_DIM = all_data_tensor.shape[1]\n",
    "    D, D_prime, xt = create_worst_case_datasets(all_data_tensor)\n",
    "\n",
    "    np.random.seed(42)\n",
    "    N_QUERIES = 2000\n",
    "    SPARSE_K = 10\n",
    "    queries = get_queries_qbs(d=INPUT_DIM, n_queries=N_QUERIES, rng=np.random, k_per_query=SPARSE_K)\n",
    "\n",
    "    attacker = train_attacker(INPUT_DIM, D, D_prime, xt, queries)\n",
    "    member_scores, non_member_scores = run_audit(attacker, INPUT_DIM, D, D_prime, xt, queries)\n",
    "\n",
    "    print(\"\\n--- Experiment finished, generating final plot ---\")\n",
    "    plot_final_curves(member_scores, non_member_scores)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "04e4e285-4d31-40c5-a029-36871d71dc1c",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from scipy.stats import norm\n",
    "from sklearn.metrics import roc_curve\n",
    "\n",
    "plt.rcParams.update({\n",
    "    \"font.family\": \"serif\",\n",
    "    \"font.size\": 14,\n",
    "    \"axes.labelsize\": 16,\n",
    "    \"axes.titlesize\": 18,\n",
    "    \"legend.fontsize\": 13,\n",
    "    \"xtick.labelsize\": 13,\n",
    "    \"ytick.labelsize\": 13,\n",
    "    \"lines.linewidth\": 2,\n",
    "    \"lines.markersize\": 5,\n",
    "})\n",
    "\n",
    "def convert_eps_delta_to_mu(epsilon, delta, *, tol=1e-12, max_iter=200):\n",
    "    if epsilon < 0: raise ValueError(\"epsilon >= 0 required\")\n",
    "    if delta <= 0:  return 0.0\n",
    "    if delta >= 1:  return float(\"inf\")\n",
    "    def f(mu):\n",
    "        mu = max(mu, 1e-16); a = -epsilon / mu\n",
    "        return norm.cdf(a + mu/2) - np.exp(epsilon) * norm.cdf(a - mu/2)\n",
    "    lo, hi = 0.0, 1.0\n",
    "    while f(hi) < delta and hi < 1e6: hi *= 2.0\n",
    "    if f(hi) < delta: return float('inf')\n",
    "    for _ in range(max_iter):\n",
    "        mid = 0.5*(lo+hi)\n",
    "        if f(mid) >= delta: hi = mid\n",
    "        else: lo = mid\n",
    "        if hi-lo <= tol * max(1.0, hi): break\n",
    "    return hi\n",
    "\n",
    "def gdp_tradeoff_full(alpha, mu):\n",
    "    alpha = np.asarray(alpha, dtype=float)\n",
    "    LARGE = 12.0\n",
    "    z = np.empty_like(alpha)\n",
    "    mid = (alpha > 0.0) & (alpha < 1.0)\n",
    "    z[mid] = norm.ppf(1.0 - alpha[mid])\n",
    "    z[alpha <= 0.0] = LARGE\n",
    "    z[alpha >= 1.0] = -LARGE\n",
    "    return norm.cdf(z - mu)\n",
    "\n",
    "def gdp_shifted(alpha, mu, gamma):\n",
    "    return np.clip(gdp_tradeoff_full(alpha + gamma, mu) - gamma, 0.0, 1.0)\n",
    "\n",
    "def get_empirical_curve(member_scores, non_member_scores):\n",
    "    y_true  = np.concatenate([np.ones(len(member_scores)), np.zeros(len(non_member_scores))])\n",
    "    y_score = np.concatenate([member_scores, non_member_scores])\n",
    "    alpha_emp, tpr, _ = roc_curve(y_true, y_score)\n",
    "    beta_emp = 1 - tpr\n",
    "    return alpha_emp, beta_emp\n",
    "\n",
    "alpha_emp, beta_emp = get_empirical_curve(member_scores, non_member_scores)\n",
    "\n",
    "def plot_with_theory(\n",
    "    mu_base=None,\n",
    "    mu_eff=0.05,\n",
    "    gamma=0.02,\n",
    "    label_suffix=\"\",\n",
    "    show_empirical=True,\n",
    "    show_envelope=True,\n",
    "    show_components=True,\n",
    "    fig=None, ax=None\n",
    "):\n",
    "    if ax is None:\n",
    "        fig, ax = plt.subplots(figsize=(9, 7))\n",
    "    if show_empirical:\n",
    "        ax.plot(alpha_emp, beta_emp, 'o-', linewidth=1.8, markersize=3.5,\n",
    "                color=\"tab:purple\", alpha=0.9, label='Empirical (Query-based Worst Case)')\n",
    "\n",
    "    if mu_base is None:\n",
    "        mu_base = convert_eps_delta_to_mu(TARGET_EPSILON, TARGET_DELTA)\n",
    "    \n",
    "    alpha = np.linspace(0.0, 1.0, 1401)\n",
    "    beta_base = gdp_tradeoff_full(alpha, mu_base)\n",
    "    beta_ours = gdp_shifted(alpha, mu_eff, gamma)\n",
    "    beta_env  = np.maximum(beta_base, beta_ours)\n",
    "\n",
    "    if show_envelope:\n",
    "        ax.plot(alpha, beta_env, color=\"tab:green\", linewidth=2.8,\n",
    "                label=f\"Envelope{label_suffix}\")\n",
    "\n",
    "    if show_components:\n",
    "        ax.plot(alpha, beta_base, \"--\", color=\"tab:orange\", linewidth=2.0,\n",
    "                label=f\"Baseline GDP{label_suffix} ($\\mu_{{\\mathrm{{base}}}}={mu_base:.3f}$)\")\n",
    "        ax.plot(alpha, beta_ours, \"--\", color=\"tab:blue\", linewidth=2.0,\n",
    "                label=f\"Our bound{label_suffix}($\\mu_{{\\mathrm{{eff}}}}={mu_eff:.3f}, \\gamma_d={gamma:.3f}$)\")\n",
    "\n",
    "    ax.axvline(gamma, color=\"gray\", linestyle=\":\", linewidth=1.0, alpha=0.6)\n",
    "    ax.axvline(1.0 - gamma, color=\"gray\", linestyle=\":\", linewidth=1.0, alpha=0.6)\n",
    "    ax.text(gamma, 0.04, \"γ\", ha=\"center\", va=\"bottom\", fontsize=9, color=\"gray\")\n",
    "    ax.text(1.0 - gamma, 0.04, \"1-γ\", ha=\"center\", va=\"bottom\", fontsize=9, color=\"gray\")\n",
    "\n",
    "    ax.plot([0, 1], [1, 0], 'k:', linewidth=1.2, label='Random Guess')\n",
    "\n",
    "    ax.set_xlabel('Type I Error (False Positive Rate, $\\\\alpha$)')\n",
    "    ax.set_ylabel('Type II Error (False Negative Rate, $\\\\beta$)')\n",
    "    ax.set_title('Empirical (fixed) vs Adjustable Theory')\n",
    "    ax.grid(True, alpha=0.3)\n",
    "    ax.set_xlim(0, 1); ax.set_ylim(0, 1)\n",
    "    ax.set_aspect('equal', adjustable='box')\n",
    "    ax.legend()\n",
    "    plt.tight_layout()\n",
    "    return fig, ax\n",
    "\n",
    "def compare_multiple_schemes(schemes, mu_base=None, show_empirical=True):\n",
    "    fig, ax = plt.subplots(figsize=(9,7))\n",
    "    for sch in schemes:\n",
    "        mu_eff = sch.get(\"mu_eff\", 0.02)\n",
    "        gamma  = sch.get(\"gamma\", 0.01)\n",
    "        label  = sch.get(\"label\", f\"mu_eff={mu_eff}, gamma={gamma}\")\n",
    "        plot_with_theory(mu_base=mu_base, mu_eff=mu_eff, gamma=gamma,\n",
    "                         label_suffix=f\" [{label}]\",\n",
    "                         show_empirical=show_empirical, show_envelope=True, show_components=True,\n",
    "                         fig=fig, ax=ax)\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a89a3e2e-865e-44bd-9c83-afcc697753e5",
   "metadata": {},
   "outputs": [],
   "source": [
    "from ipywidgets import interact, FloatLogSlider, FloatSlider, fixed\n",
    "\n",
    "def _ui(mu_eff=0.03, gamma=0.01, mu_base_override=None):\n",
    "    mu_base = None if (mu_base_override is None or mu_base_override <= 0) else mu_base_override\n",
    "    plt.close('all')\n",
    "    fig, ax = plot_with_theory(mu_base=mu_base, mu_eff=mu_eff, gamma=gamma)\n",
    "    fig.savefig(\"best_fit_curve.pdf\", bbox_inches=\"tight\")\n",
    "\n",
    "mu_base = convert_eps_delta_to_mu(TARGET_EPSILON, TARGET_DELTA)\n",
    "interact(\n",
    "    _ui,\n",
    "    mu_eff=FloatLogSlider(value=0.03, base=10, min=-3, max=0, step=0.01, description='mu_eff'),\n",
    "    gamma=FloatLogSlider(value=0.01, base=10, min=-3, max=0, step=0.01, description='gamma'),\n",
    "    mu_base_override=FloatLogSlider(value=mu_base, base=10, min=-3, max=1, step=0.01, description='mu_base (optional)')\n",
    ");"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "ecd9ee62-98de-4e9c-ab28-6bc6c2375dcc",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/root/anaconda3/lib/python3.10/site-packages/opacus/privacy_engine.py:95: UserWarning: Secure RNG turned off. This is perfectly fine for experimentation as it allows for much faster training performance, but remember to turn it on and retrain one last time before production with ``secure_mode`` turned on.\n",
      "  warnings.warn(\n",
      "09/25/2025 12:49:37:WARNING:Ignoring drop_last as it is not compatible with DPDataLoader.\n",
      "/root/anaconda3/lib/python3.10/site-packages/torch/nn/modules/module.py:1359: UserWarning: Using a non-full backward hook when the forward contains multiple autograd Nodes is deprecated and will be removed in future versions. This hook will be missing some grad_input. Please use register_full_backward_hook to get the documented behavior.\n",
      "  warnings.warn(\"Using a non-full backward hook when the forward contains multiple autograd Nodes \"\n",
      "09/25/2025 12:49:38:WARNING:Ignoring drop_last as it is not compatible with DPDataLoader.\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[MNIST][D ] d_TV ≈ 0.0254   (KS upper bound: 0.0201)\n",
      "[MNIST][D'] d_TV ≈ 0.0217  (KS upper bound: 0.0151)\n",
      "[MNIST][γ_empirical] = max(d_TV(D), d_TV(D')) ≈ 0.0254\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA90AAAGGCAYAAABmGOKbAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy88F64QAAAACXBIWXMAAA9hAAAPYQGoP6dpAABGTUlEQVR4nO3dfVzV9f3/8edJrgov5lUcKUFsW+JFFzuYQhGVhsPVXOE3dU0txe8Y2xyQ2y+0plnfMCXH1yWyTFQ2U/tmrS1pSk3JJlooLmdscxNFkZNfqIlTA8TP749unG+nc7jm44FzHvfb7XO7ed6f1+d9wUd8+zrvz4XFMAxDAAAAAACgy13l6Q4AAAAAAOCtSLoBAAAAADAJSTcAAAAAACYh6QYAAAAAwCQk3QAAAAAAmISkGwAAAAAAk5B0AwAAAABgEpJuAAAAAABMQtINAAAAAIBJSLqBZmzYsEEWi8Wx+fn56frrr9ejjz6qyspKj/Vr2LBheuSRRxyfd+/eLYvFot27d1/xvrz88svKzs52u89isWjJkiVXtD8tWbJkiSwWS5tijx8/7nTuW9oeeOABWSwW/fWvf222vkWLFslisejgwYOttj1hwgQlJyc7Pjed36YtICBAgwcP1u23365FixbpxIkTLnWsW7dO1113nc6fP9+m8QJAd8L82zrm3+45/zb93f2iO++8U6mpqW0aP7yXn6c7AHR369ev14gRI3Tx4kW9++67yszMVFFRkQ4fPqzg4GBPd0/f+MY3VFxcrJEjR17xtl9++WX95S9/cTuZFBcX6/rrr7/ifeoKQ4YMUXFxsVNZSkqKzp49q02bNjmVX3311frtb3+rvLw8LV++3KWuy5cvKz8/X7fccou+8Y1vtNjuG2+8oT/96U/Kz8932ffss8/q7rvvVmNjo2pqarR//37l5eXpF7/4hdauXauHH37YETt79mw999xzWr58uZ566qn2DB0Aug3m3+Yx/3bP+dedp59+Wvfee69+8IMf6MYbb2wxFl7MAODW+vXrDUnGBx984FT+5JNPGpKM3/zmN51u4/z58+0+Jjw83Jg9e/YVaas13/rWt4zw8PAur9cMixcvNjrzT15cXJwxatQot/tuu+02w2q1Gg0NDS773nrrLUOS8ctf/rLVNm677TZj+vTpTmW7du0yJBn/8z//4xJfU1Nj3HrrrYafn5/x4YcfOu3Lysoy+vXrZ8p5BwAzMf+2jvn3c91t/m36u/tlo0ePNubNm9dqP+C9uLwcaKfx48dLkuOyIsMwlJOTo1tuuUVXX321+vfvr6lTp+rYsWNOx911110aPXq03n33XcXExOiaa67RnDlzmm2noaFBP/vZz2S1WnXNNdfojjvu0Pvvv+8S5+7ytkceeUS9e/fW4cOHFR8frz59+mjChAmSpPr6ej3zzDMaMWKEAgMDNXjwYD366KP63//9X5e6X375ZUVHR6t3797q3bu3brnlFq1bt84xnu3bt+vEiRNOl181cXd521/+8hdNmTJF/fv3V1BQkG655RZt3LjR7Xg2b96sRYsWKTQ0VH379tXEiRP1t7/9rdmf1xdt375dt9xyiwIDAxUREaGsrKw2HddRc+fOld1u11tvveWyb/369QoMDGz1m/DS0lK9//77mjlzZpvbHTBggH71q1/p0qVL+sUvfuG07+GHH1Ztba22bNnS5voAoDtj/mX+/bLuOP+6M3PmTL388ss6d+5cm9uAdyHpBtrpH//4hyRp8ODBkqTvf//7Sk1N1cSJE/Xb3/5WOTk5OnLkiGJiYvTxxx87HVtVVaXvfe97+u53v6uCggKlpKQ02868efOUlZWlWbNm6Y033lBiYqIefPBBffrpp23qZ319vb797W/rnnvu0RtvvKGnnnpKly9f1pQpU7Rs2TJ997vf1fbt27Vs2TIVFhbqrrvu0sWLFx3H//znP9fDDz+s0NBQbdiwQa+//rpmz57t+M9OTk6Obr/9dlmtVhUXFzu25vztb39TTEyMjhw5olWrVum1117TyJEj9cgjj7i9LGzhwoU6ceKEXnrpJb344os6evSo7r//fjU2NrY47nfeeUdTpkxRnz59tGXLFq1YsUKvvPKK1q9f36afW0fMmDFD11xzjfLy8pzKP/30U73xxht64IEH1L9//xbrePPNN9WrVy/deeed7Wp77NixGjJkiN59912ncqvVqhEjRmj79u3tqg8AuivmX+bfL+tu8+8jjzwiwzBcYu+66y6dP3/eI/f/o5vw8Eo70G01XSK0b98+o6GhwTh37pzx5ptvGoMHDzb69Olj2O12o7i42JBkPP/8807Hnjx50rj66quNn/3sZ46yuLg4Q5LxzjvvtNp2WVmZIclIS0tzKt+0aZMhyenytqbLn3bt2uUomz17tiHJyMvLczp+8+bNhiRj27ZtTuUffPCBIcnIyckxDMMwjh07ZvTq1ct4+OGHW+xnS5e3STIWL17s+Dx9+nQjMDDQqKiocIpLSEgwrrnmGuNf//qX03gmT57sFPfKK68Ykozi4uIW+zRu3DgjNDTUuHjxoqOstrbWGDBggGmXtxnG5z9zf39/4+OPP3aU/fKXvzQkGYWFha3Wn5CQYIwYMcKlvKXL25qMGzfOuPrqq13KH374YSMkJKTVtgGgO2H+Zf79op44/35ZfX29YbFYjP/3//5fq7HwTqx0A60YP368/P391adPH913332yWq166623FBISojfffFMWi0Xf+973dOnSJcdmtVp18803u3yj2b9/f91zzz2Oz5cvX3Y6rulb5F27dkmSyyVRDz30kPz82v78w8TERKfPb775pr7yla/o/vvvd2r3lltukdVqdfS3sLBQjY2N+uEPf9jmtlrzxz/+URMmTNDQoUOdyh955BFduHDB5Vv6b3/7206fb7rpJkly+7TQJufPn9cHH3ygBx98UEFBQY7yPn366P777+/sEFo0d+5cNTQ06Ne//rWjbP369QoPD3dcWtiS06dP69prr+1Q24abb9Ul6dprr9WZM2d06dKlDtULAJ7E/Ns1mH9bZsb8+2X+/v76yle+4tGn78OzSLqBVuTn5+uDDz5QaWmpTp8+rQ8//FC33367JOnjjz+WYRgKCQmRv7+/07Zv3z5VV1c71TVkyBCnz3PmzHE6pmlyqKmpkfT5JcJf5Ofnp4EDB7ap39dcc4369u3rVPbxxx/rX//6lwICAlz6a7fbHf1tur+sK59+WlNT4zJ+SQoNDXXs/6IvjzMwMFCSnC7B+7JPP/1Uly9fdvm5Sa4/y64WGxurr3/9647L6D788EMdPHhQjz76aJtelXLx4kWn/6i0R0VFhePn+EVBQUEyDEOfffZZh+oFAE9i/u0azL8tM2P+dScoKKjFnyG8G68MA1oRGRmpqKgot/sGDRoki8WiPXv2OCalL/py2Zf/8V+yZIl+9KMfOT736dNH0v9NeHa7Xdddd51j/6VLl1wmx+a4m2gGDRqkgQMH6g9/+IPbY5rab7pf7tSpUy7fjHfUwIEDVVVV5VJ++vRpR986q3///rJYLLLb7S773JV1tTlz5ujxxx/X+++/r5dffllXXXWV0ztd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",
      "text/plain": [
       "<Figure size 1000x400 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import numpy as np, torch, torch.nn as nn, math, os\n",
    "from torch.utils.data import DataLoader, Subset\n",
    "from typing import Optional, Tuple\n",
    "import torchvision\n",
    "from torchvision import datasets, transforms\n",
    "\n",
    "DEVICE = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
    "torch.set_num_threads(4)\n",
    "\n",
    "# Hyperparameters\n",
    "SEED         = 1234\n",
    "LATENT_DIM   = 32\n",
    "HIDDEN       = 256\n",
    "BATCH        = 512\n",
    "SUBSET       = 10000\n",
    "STEPS        = 20\n",
    "LR           = 3e-3\n",
    "USE_DP       = True\n",
    "SIGMA        = 1.0\n",
    "CLIP_C       = 1.0\n",
    "DELTA        = 1e-5\n",
    "\n",
    "# Empirical d_TV estimation settings\n",
    "N_SAMPLES    = 200_000\n",
    "MODE         = \"linear\"\n",
    "NUM_DIRS     = 32\n",
    "BINS         = 401\n",
    "\n",
    "mnist_root = \"../data\"\n",
    "if not os.path.exists(mnist_root):\n",
    "    raise FileNotFoundError(\"../data directory not found. Please ensure the dataset is downloaded to ../data.\")\n",
    "\n",
    "tfm = transforms.Compose([transforms.ToTensor(), transforms.Lambda(lambda t: t.view(-1))])\n",
    "ds  = datasets.MNIST(mnist_root, train=True, download=False, transform=tfm)\n",
    "rng = np.random.RandomState(SEED)\n",
    "idx = rng.choice(len(ds), size=min(SUBSET, len(ds)), replace=False)\n",
    "loader_D  = DataLoader(Subset(ds, idx), batch_size=BATCH, shuffle=True, drop_last=True, num_workers=0)\n",
    "loader_Dp = DataLoader(Subset(ds, rng.choice(len(ds), size=min(SUBSET, len(ds)), replace=False)),\n",
    "                       batch_size=BATCH, shuffle=True, drop_last=True, num_workers=0)\n",
    "\n",
    "class TinyVAE(nn.Module):\n",
    "    def __init__(self, input_dim=784, latent_dim=LATENT_DIM, hidden=HIDDEN):\n",
    "        super().__init__()\n",
    "        act = nn.Softplus(beta=1.0)\n",
    "        self.enc = nn.Sequential(nn.Linear(input_dim, hidden), act, nn.Linear(hidden, 2*latent_dim))\n",
    "        self.dec = nn.Sequential(nn.Linear(latent_dim, hidden), act, nn.Linear(hidden, input_dim))\n",
    "        self.latent_dim = latent_dim\n",
    "    @property\n",
    "    def decoder(self): return self.dec\n",
    "    def reparam(self, mu, logv):\n",
    "        std = (0.5*logv).exp(); eps = torch.randn_like(std); return mu + eps*std\n",
    "    def forward(self, x):\n",
    "        h = self.enc(x); mu, logv = torch.chunk(h, 2, dim=-1)\n",
    "        z = self.reparam(mu, logv); xhat = self.dec(z); return xhat, mu, logv\n",
    "\n",
    "def vae_loss(xhat, x, mu, logv):\n",
    "    recon = ((xhat - x)**2).sum(dim=1).mean()\n",
    "    kld   = -0.5 * (1 + logv - mu.pow(2) - logv.exp()).sum(dim=1).mean()\n",
    "    return recon + kld\n",
    "\n",
    "def train_vae(loader, *, seed, use_dp=USE_DP, sigma=SIGMA, clip_c=CLIP_C, steps=STEPS, lr=LR):\n",
    "    torch.manual_seed(seed); np.random.seed(seed)\n",
    "    model = TinyVAE().to(DEVICE)\n",
    "    opt   = torch.optim.Adam(model.parameters(), lr=lr)\n",
    "    if use_dp:\n",
    "        try:\n",
    "            from opacus import PrivacyEngine\n",
    "            pe = PrivacyEngine()\n",
    "            try:\n",
    "                model, opt, loader = pe.make_private(\n",
    "                    module=model, optimizer=opt, data_loader=loader,\n",
    "                    noise_multiplier=sigma, max_grad_norm=clip_c, grad_sample_mode=\"hooks\"\n",
    "                )\n",
    "            except TypeError:\n",
    "                model, opt, criterion, loader = pe.make_private(\n",
    "                    module=model, optimizer=opt, criterion=nn.MSELoss(reduction=\"mean\"),\n",
    "                    data_loader=loader, noise_multiplier=sigma, max_grad_norm=clip_c, grad_sample_mode=\"hooks\"\n",
    "                )\n",
    "        except Exception as e:\n",
    "            print(\"[WARN] Opacus initialization failed, falling back to non-DP:\", e)\n",
    "            use_dp = False\n",
    "            pe = None\n",
    "    model.train(); it = 0\n",
    "    while it < steps:\n",
    "        for batch in loader:\n",
    "            xb = batch[0].to(DEVICE)\n",
    "            xhat, mu, logv = model(xb)\n",
    "            loss = vae_loss(xhat, xb, mu, logv)\n",
    "            opt.zero_grad(); loss.backward(); opt.step()\n",
    "            it += 1\n",
    "            if it >= steps: break\n",
    "    model.eval()\n",
    "    return plain_clone_for_eval(model, LATENT_DIM)\n",
    "\n",
    "def plain_clone_for_eval(dp_model: nn.Module, latent_dim: int) -> nn.Module:\n",
    "    state = dp_model.state_dict()\n",
    "    clean_state = {}\n",
    "    for k, v in state.items():\n",
    "        nk = k\n",
    "        if nk.startswith(\"_module.\"): nk = nk[len(\"_module.\"):]\n",
    "        if nk.startswith(\"module.\"):  nk = nk[len(\"module.\"):]\n",
    "        clean_state[nk] = v.detach().cpu()\n",
    "    plain = TinyVAE(latent_dim=latent_dim).to(DEVICE).eval()\n",
    "    plain.load_state_dict(clean_state, strict=False)\n",
    "    return plain\n",
    "\n",
    "vae_D  = train_vae(loader_D,  seed=SEED)\n",
    "vae_Dp = train_vae(loader_Dp, seed=SEED+7)\n",
    "\n",
    "@torch.no_grad()\n",
    "def scalarize_linear(decoder: nn.Module, z: torch.Tensor, out_dim: Optional[int]=None,\n",
    "                     dirs: int=8, seed: int=0):\n",
    "    with torch.no_grad():\n",
    "        y0 = decoder(z[:1]).reshape(1, -1)\n",
    "    D = y0.shape[1] if out_dim is None else out_dim\n",
    "    torch.manual_seed(seed); np.random.seed(seed)\n",
    "    U = torch.randn(dirs, D, device=DEVICE)\n",
    "    U = U / (U.norm(dim=1, keepdim=True) + 1e-12)\n",
    "    Ys = []\n",
    "    B = 4096\n",
    "    for r in range(dirs):\n",
    "        ys = []\n",
    "        u  = U[r].view(-1,1)\n",
    "        for i in range(0, z.size(0), B):\n",
    "            zi = z[i:i+B]\n",
    "            xi = decoder(zi)\n",
    "            yi = (xi @ u).squeeze(-1)\n",
    "            ys.append(yi.cpu())\n",
    "        Ys.append(torch.cat(ys,0).numpy())\n",
    "    return Ys\n",
    "\n",
    "@torch.no_grad()\n",
    "def scalarize_energy(decoder: nn.Module, z: torch.Tensor):\n",
    "    Ys = []\n",
    "    B = 4096\n",
    "    for i in range(0, z.size(0), B):\n",
    "        zi = z[i:i+B]\n",
    "        xi = decoder(zi)\n",
    "        yi = 0.5 * (xi.pow(2).sum(dim=1))\n",
    "        Ys.append(yi.cpu())\n",
    "    return torch.cat(Ys,0).numpy()\n",
    "\n",
    "def tv_histogram_vs_normal(y: np.ndarray, bins: int=401) -> float:\n",
    "    y = (y - y.mean()) / (y.std(ddof=1) + 1e-12)\n",
    "    q = np.linspace(0.0, 1.0, bins+1)\n",
    "    q = np.clip(q, 1e-12, 1-1e-12)\n",
    "    from scipy.stats import norm\n",
    "    edges = norm.ppf(q)\n",
    "    hist, _ = np.histogram(y, bins=edges)\n",
    "    p = hist / hist.sum()\n",
    "    tv = 0.5 * np.abs(p - 1.0/bins).sum()\n",
    "    return float(tv)\n",
    "\n",
    "def ks_distance_vs_normal(y: np.ndarray) -> float:\n",
    "    y = (y - y.mean()) / (y.std(ddof=1) + 1e-12)\n",
    "    y_sorted = np.sort(y)\n",
    "    n = y_sorted.size\n",
    "    from scipy.stats import norm\n",
    "    F0 = norm.cdf(y_sorted)\n",
    "    i = np.arange(1, n+1)\n",
    "    D_plus  = np.max(i/n - F0)\n",
    "    D_minus = np.max(F0 - (i-1)/n)\n",
    "    return float(max(D_plus, D_minus))\n",
    "\n",
    "def empirical_dtv_for_model(model: nn.Module, latent_dim: int,\n",
    "                            N: int=200_000, mode: str=\"linear\", dirs: int=8,\n",
    "                            seed: int=0, bins: int=401):\n",
    "    model.eval().to(DEVICE)\n",
    "    torch.manual_seed(seed); np.random.seed(seed)\n",
    "    Z = torch.randn(N, latent_dim, device=DEVICE)\n",
    "    if mode == \"linear\":\n",
    "        Ys = scalarize_linear(model.decoder, Z, dirs=dirs, seed=seed)\n",
    "        tvs = []; kss = []\n",
    "        for y in Ys:\n",
    "            tvs.append(tv_histogram_vs_normal(y, bins=bins))\n",
    "            kss.append(2.0 * ks_distance_vs_normal(y))\n",
    "        return float(np.max(tvs)), float(np.max(kss)), dict(per_dir_tv=tvs, per_dir_ks=kss)\n",
    "    elif mode == \"energy\":\n",
    "        y = scalarize_energy(model.decoder, Z)\n",
    "        tv = tv_histogram_vs_normal(y, bins=bins)\n",
    "        ks = ks_distance_vs_normal(y)\n",
    "        return float(tv), float(2.0*ks), {}\n",
    "    else:\n",
    "        raise ValueError(\"mode must be 'linear' or 'energy'\")\n",
    "\n",
    "tv_D,  ks_up_D,  st_D  = empirical_dtv_for_model(vae_D,  LATENT_DIM, N=N_SAMPLES, mode=MODE, dirs=NUM_DIRS, bins=BINS, seed=2025)\n",
    "tv_Dp, ks_up_Dp, st_Dp = empirical_dtv_for_model(vae_Dp, LATENT_DIM, N=N_SAMPLES, mode=MODE, dirs=NUM_DIRS, bins=BINS, seed=2026)\n",
    "\n",
    "gamma_emp = max(tv_D, tv_Dp)\n",
    "\n",
    "print(f\"[MNIST][D ] d_TV ≈ {tv_D:.4f}   (KS upper bound: {ks_up_D:.4f})\")\n",
    "print(f\"[MNIST][D'] d_TV ≈ {tv_Dp:.4f}  (KS upper bound: {ks_up_Dp:.4f})\")\n",
    "print(f\"[MNIST][γ_empirical] = max(d_TV(D), d_TV(D')) ≈ {gamma_emp:.4f}\")\n",
    "\n",
    "if MODE == \"linear\":\n",
    "    import matplotlib.pyplot as plt\n",
    "    fig, ax = plt.subplots(1,2, figsize=(10,4))\n",
    "    ax[0].bar(np.arange(len(st_D[\"per_dir_tv\"])), st_D[\"per_dir_tv\"], alpha=0.8)\n",
    "    ax[0].set_title(\"Per-direction d_TV (D)\"); ax[0].set_xlabel(\"dir\"); ax[0].set_ylabel(\"d_TV\")\n",
    "    ax[1].bar(np.arange(len(st_Dp[\"per_dir_tv\"])), st_Dp[\"per_dir_tv\"], alpha=0.8, color=\"#ff7f0e\")\n",
    "    ax[1].set_title(\"Per-direction d_TV (D')\"); ax[1].set_xlabel(\"dir\")\n",
    "    plt.tight_layout(); plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "11663638-23d2-4579-a6df-49ae29085f04",
   "metadata": {
    "jupyter": {
     "source_hidden": true
    },
    "tags": []
   },
   "outputs": [],
   "source": [
    "# LPK stability under DP-SGD (smoothed via replicates + antithetic probes)\n",
    "import math, numpy as np, torch, torch.nn as nn, torch.nn.functional as F\n",
    "import matplotlib.pyplot as plt\n",
    "from tqdm.notebook import tqdm\n",
    "\n",
    "# =========================\n",
    "# Config\n",
    "# =========================\n",
    "class Cfg:\n",
    "    d = 128\n",
    "    width = 256\n",
    "    n_steps = 1000\n",
    "    lr = 3e-3\n",
    "    weight_decay = 1e-4\n",
    "    C = 1.0\n",
    "    sigma_dp = 1.0\n",
    "    q_fixed = 0.05\n",
    "    n_list = [100, 200, 400, 1000, 2000, 4000, 8000]\n",
    "    log_every = 40\n",
    "    M = 64                  # larger probe pool\n",
    "    lam_reg = 1e-6\n",
    "    device = \"cpu\"\n",
    "    base_seed = 2025\n",
    "    R_avg = 5               # number of independent repeats for averaging\n",
    "    fix_batch = True        # use fixed-size minibatch to reduce variance\n",
    "\n",
    "cfg = Cfg()\n",
    "torch.set_num_threads(max(1, torch.get_num_threads()))\n",
    "\n",
    "def set_global_seed(s: int):\n",
    "    torch.manual_seed(s)\n",
    "    np.random.seed(s % (2**32 - 1))\n",
    "\n",
    "# =========================\n",
    "# Data\n",
    "# =========================\n",
    "def make_neighboring_datasets(n: int, d: int, noise_std: float = 0.1, seed: int = 0):\n",
    "    g = torch.Generator(device=cfg.device).manual_seed(seed)\n",
    "    x = torch.randn(n, d, generator=g)\n",
    "    u = torch.randn(d, generator=g) / math.sqrt(d)\n",
    "    y = x @ u + noise_std * torch.randn(n, generator=g)\n",
    "    x2 = x.clone(); y2 = y.clone()\n",
    "    x2[-1] = torch.randn(d, generator=g)\n",
    "    y2[-1] = x2[-1] @ u + noise_std * torch.randn((), generator=g)\n",
    "    return (x, y), (x2, y2)\n",
    "\n",
    "# =========================\n",
    "# Model\n",
    "# =========================\n",
    "class MLP(nn.Module):\n",
    "    def __init__(self, d: int, width: int):\n",
    "        super().__init__()\n",
    "        self.fc1 = nn.Linear(d, width)\n",
    "        self.fc2 = nn.Linear(width, 1)\n",
    "        nn.init.xavier_normal_(self.fc1.weight); nn.init.zeros_(self.fc1.bias)\n",
    "        nn.init.xavier_normal_(self.fc2.weight); nn.init.zeros_(self.fc2.bias)\n",
    "    def forward(self, x: torch.Tensor) -> torch.Tensor:\n",
    "        h = F.gelu(self.fc1(x))\n",
    "        return self.fc2(h).squeeze(-1)\n",
    "\n",
    "# =========================\n",
    "# DP-SGD step\n",
    "# =========================\n",
    "def dp_sgd_step(model, x_batch, y_batch, lr, C, sigma, noise_vec, weight_decay=0.0):\n",
    "    params = [p for p in model.parameters() if p.requires_grad]\n",
    "    P = sum(p.numel() for p in params)\n",
    "    bsz = x_batch.size(0)\n",
    "    device = x_batch.device\n",
    "\n",
    "    G = torch.zeros((bsz, P), device=device)\n",
    "    for i in range(bsz):\n",
    "        model.zero_grad(set_to_none=True)\n",
    "        pred = model(x_batch[i:i+1])\n",
    "        loss = 0.5 * (pred - y_batch[i:i+1])**2\n",
    "        loss.backward()\n",
    "        grads = [ (p.grad if p.grad is not None else torch.zeros_like(p)).reshape(-1) for p in params ]\n",
    "        G[i] = torch.cat(grads)\n",
    "\n",
    "    if weight_decay > 0.0:\n",
    "        with torch.no_grad():\n",
    "            wflat = torch.cat([p.detach().reshape(-1) for p in params])\n",
    "        G = G + weight_decay * wflat.unsqueeze(0).expand_as(G)\n",
    "\n",
    "    norms = torch.linalg.vector_norm(G, dim=1) + 1e-12\n",
    "    scale = (C / norms).clamp(max=1.0).unsqueeze(1)\n",
    "    Gc = G * scale\n",
    "\n",
    "    g_bar = Gc.mean(dim=0)\n",
    "    if sigma > 0.0:\n",
    "        g_bar = g_bar + (sigma * C / bsz) * noise_vec\n",
    "\n",
    "    offset = 0\n",
    "    with torch.no_grad():\n",
    "        for p in params:\n",
    "            n = p.numel()\n",
    "            p.add_(-lr * g_bar[offset:offset+n].view_as(p))\n",
    "            offset += n\n",
    "\n",
    "# =========================\n",
    "# LPK increment via probes (antithetic)\n",
    "# =========================\n",
    "def lpk_increment_via_probes(model, Z):\n",
    "    \"\"\"\n",
    "    Z is assumed to be already antithetic-augmented: stack([Z0, -Z0])\n",
    "    \"\"\"\n",
    "    params = [p for p in model.parameters() if p.requires_grad]\n",
    "    P = sum(p.numel() for p in params)\n",
    "    M = Z.size(0); device = Z.device\n",
    "\n",
    "    f_vals = torch.zeros(M, device=device)\n",
    "    J = torch.zeros(M, P, device=device)\n",
    "    with torch.enable_grad():\n",
    "        for m in range(M):\n",
    "            for p in params:\n",
    "                if p.grad is not None: p.grad = None\n",
    "            z = Z[m:m+1].requires_grad_(True)\n",
    "            f = model(z)\n",
    "            f_vals[m] = f.detach().squeeze(0)\n",
    "            f.backward()\n",
    "            grads = [ (p.grad if p.grad is not None else torch.zeros_like(p)).reshape(-1) for p in params ]\n",
    "            J[m] = torch.cat(grads)\n",
    "\n",
    "    JJt = J @ J.t()\n",
    "    K_inc = (f_vals.unsqueeze(1) @ f_vals.unsqueeze(0)) * JJt\n",
    "    return f_vals, K_inc\n",
    "\n",
    "# =========================\n",
    "# Training pair (fixed-q only, DP only)\n",
    "# =========================\n",
    "def train_pair_and_measure_fixed_q(n: int, q: float, sigma: float, C_clip: float, seed: int):\n",
    "    set_global_seed(seed)\n",
    "    (x, y), (x2, y2) = make_neighboring_datasets(n, cfg.d, seed=seed)\n",
    "    device = cfg.device\n",
    "    x, y, x2, y2 = x.to(device), y.to(device), x2.to(device), y2.to(device)\n",
    "\n",
    "    # identical init + state dict copy\n",
    "    mD = MLP(cfg.d, cfg.width).to(device)\n",
    "    mDp = MLP(cfg.d, cfg.width).to(device)\n",
    "    mDp.load_state_dict(mD.state_dict())\n",
    "\n",
    "    # antithetic probes\n",
    "    gZ = torch.Generator().manual_seed(seed + 30_000)\n",
    "    M_half = cfg.M // 2\n",
    "    Z0 = torch.randn(M_half, cfg.d, generator=gZ, device=device)\n",
    "    Z = torch.cat([Z0, -Z0], dim=0)  # antithetic\n",
    "    if Z.size(0) < cfg.M:  # odd M fallback\n",
    "        Z = torch.cat([Z, torch.randn(1, cfg.d, generator=gZ, device=device)], dim=0)\n",
    "\n",
    "    K_D = torch.zeros(cfg.M, cfg.M, device=device)\n",
    "    K_Dp = torch.zeros(cfg.M, cfg.M, device=device)\n",
    "\n",
    "    base_lr = cfg.lr\n",
    "    g_idx = torch.Generator().manual_seed(seed + 123)\n",
    "    P = sum(p.numel() for p in mD.parameters())\n",
    "\n",
    "    b = max(1, int(round(q * n)))  # fixed-size minibatch to reduce variance\n",
    "\n",
    "    for step in tqdm(range(1, cfg.n_steps + 1), desc=f\"n={n}, sigma={sigma}\", leave=False):\n",
    "        # cosine schedule\n",
    "        eta_t = base_lr * 0.5 * (1 + math.cos(math.pi * (step - 1) / cfg.n_steps))\n",
    "\n",
    "        # fixed-size minibatch (without replacement) for stability\n",
    "        perm = torch.randperm(n, generator=g_idx)\n",
    "        idx = perm[:b]\n",
    "        xb, yb = x[idx], y[idx]\n",
    "        xb2, yb2 = x2[idx], y2[idx]\n",
    "\n",
    "        # shared DP noise\n",
    "        g_noise = torch.Generator().manual_seed(seed * 10_000 + step)\n",
    "        noise_vec = torch.randn(P, generator=g_noise, device=device)\n",
    "\n",
    "        dp_sgd_step(mD, xb, yb, eta_t, C=C_clip, sigma=sigma, noise_vec=noise_vec, weight_decay=cfg.weight_decay)\n",
    "        dp_sgd_step(mDp, xb2, yb2, eta_t, C=C_clip, sigma=sigma, noise_vec=noise_vec, weight_decay=cfg.weight_decay)\n",
    "\n",
    "        if (step % cfg.log_every == 0) or (step == cfg.n_steps):\n",
    "            _, Kinc_D = lpk_increment_via_probes(mD, Z)\n",
    "            _, Kinc_Dp = lpk_increment_via_probes(mDp, Z)\n",
    "            K_D += eta_t * Kinc_D\n",
    "            K_Dp += eta_t * Kinc_Dp\n",
    "\n",
    "    K_oplus = (K_D + K_Dp).cpu().numpy()\n",
    "    with torch.no_grad():\n",
    "        # same antithetic Z for readout\n",
    "        y_vec = (mD(Z) - mDp(Z)).cpu().numpy()\n",
    "\n",
    "    K_reg = K_oplus + (cfg.lam_reg * np.eye(cfg.M, dtype=K_oplus.dtype))\n",
    "    try:\n",
    "        alpha = np.linalg.solve(K_reg, y_vec)\n",
    "    except np.linalg.LinAlgError:\n",
    "        alpha = np.linalg.lstsq(K_reg, y_vec, rcond=None)[0]\n",
    "    rkhs_lb = float(y_vec @ alpha)\n",
    "    rkhs_lb = max(rkhs_lb, 0.0)\n",
    "    rkhs_norm = math.sqrt(rkhs_lb + 1e-20)\n",
    "\n",
    "    diag_mean = float(np.mean(np.diag(K_oplus)))\n",
    "    mu_eff_hat = rkhs_norm / math.sqrt(max(diag_mean, 1e-12))\n",
    "    return rkhs_norm, mu_eff_hat\n",
    "\n",
    "def run_fixed_q_suite_dp_only(n_list, q, sigma_dp, base_seed):\n",
    "    out = []\n",
    "    for i, n in enumerate(tqdm(n_list, desc=\"Sweep n\")):\n",
    "        rk_arr, mu_arr = [], []\n",
    "        for r in range(cfg.R_avg):\n",
    "            seed = base_seed + 10_000*i + 97*r\n",
    "            rk, mu = train_pair_and_measure_fixed_q(n, q, sigma_dp, cfg.C, seed)\n",
    "            rk_arr.append(rk); mu_arr.append(mu)\n",
    "        rk_arr, mu_arr = np.array(rk_arr), np.array(mu_arr)\n",
    "        out.append({\n",
    "            \"n\": n,\n",
    "            \"rkhs_mean\": rk_arr.mean(),\n",
    "            \"rkhs_se\": rk_arr.std(ddof=1) / math.sqrt(len(rk_arr)),\n",
    "            \"mu_mean\": mu_arr.mean(),\n",
    "            \"mu_se\": mu_arr.std(ddof=1) / math.sqrt(len(mu_arr)),\n",
    "        })\n",
    "    return out\n",
    "\n",
    "# =========================\n",
    "# Run\n",
    "# =========================\n",
    "res = run_fixed_q_suite_dp_only(cfg.n_list, cfg.q_fixed, cfg.sigma_dp, cfg.base_seed)\n",
    "\n",
    "# =========================\n",
    "# Plot (mean ± SE) — DP only\n",
    "# =========================\n",
    "ns = np.array([r[\"n\"] for r in res])\n",
    "rk_mean = np.array([r[\"rkhs_mean\"] for r in res])\n",
    "rk_se   = np.array([r[\"rkhs_se\"] for r in res])\n",
    "mu_mean = np.array([r[\"mu_mean\"] for r in res])\n",
    "mu_se   = np.array([r[\"mu_se\"] for r in res])\n",
    "\n",
    "plt.figure()\n",
    "plt.loglog(ns, rk_mean, marker='o', label='DP-SGD (mean)')\n",
    "# reference slope -1 anchored at first point\n",
    "refy = rk_mean[0] * (ns / ns[0])**(-1.0)\n",
    "plt.loglog(ns, refy, '--', label='slope -1 ref')\n",
    "# error band\n",
    "plt.fill_between(ns, rk_mean*np.exp(-rk_se/rk_mean), rk_mean*np.exp(rk_se/rk_mean), alpha=0.15)\n",
    "plt.xlabel('n'); plt.ylabel(r'||$\\Delta f$||$_{\\mathcal H}$ (RKHS lower bound)')\n",
    "plt.legend(); plt.tight_layout(); plt.show()\n",
    "\n",
    "plt.figure()\n",
    "plt.loglog(ns, mu_mean, marker='o', label=r'DP-SGD $\\hat\\mu_{\\mathrm{eff}}$ (mean)')\n",
    "plt.fill_between(ns, mu_mean*np.exp(-mu_se/mu_mean), mu_mean*np.exp(mu_se/mu_mean), alpha=0.15)\n",
    "plt.xlabel('n'); plt.ylabel(r'$\\hat\\mu_{\\mathrm{eff}}$ surrogate')\n",
    "plt.legend(); plt.tight_layout(); plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "4223488e-9b2a-4947-a1a6-d8dd43cb4097",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                                    \r"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 620x460 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Results:\n",
      "{'d': 8, 'gamma': 0.11883101006423935, 'omega': 0.00016598858476028552}\n",
      "{'d': 16, 'gamma': 0.06698051949009853, 'omega': 0.00016418057407261284}\n",
      "{'d': 32, 'gamma': 0.04075644061811168, 'omega': 1.046880896883802e-06}\n",
      "{'d': 64, 'gamma': 0.027238220534807622, 'omega': 7.693023334964994e-05}\n",
      "{'d': 96, 'gamma': 0.029448661653962427, 'omega': 3.0429738790751125e-06}\n",
      "{'d': 128, 'gamma': 0.02277405014796907, 'omega': 6.729665861781941e-05}\n",
      "{'d': 192, 'gamma': 0.019304499136741482, 'omega': 0.00010316322762190273}\n",
      "{'d': 256, 'gamma': 0.020594122427553208, 'omega': 9.053572487192725e-06}\n"
     ]
    }
   ],
   "source": [
    "# ==============================================================================\n",
    "# DP-SGD VAE: Estimating Gaussianization error γ_d and variance slack ω (scanning over latent dimension d)\n",
    "# - Trains VAEs on two neighboring datasets (with coupled DP noise).\n",
    "# - Scalarization: Uses linear projection (max over directions) or an energy-based method to compute a KS→TV proxy and the variance slack term.\n",
    "# ==============================================================================\n",
    "import math, numpy as np, torch, torch.nn as nn, torch.nn.functional as F\n",
    "import matplotlib.pyplot as plt\n",
    "from tqdm import tqdm\n",
    "from typing import Tuple, List\n",
    "\n",
    "class Cfg:\n",
    "    # Data/Training\n",
    "    n            = 2000\n",
    "    img_side     = 28\n",
    "    steps        = 250\n",
    "    lr           = 3e-3\n",
    "    weight_decay = 0.0\n",
    "\n",
    "    # DP-SGD\n",
    "    q            = 0.05\n",
    "    C_clip       = 1.0\n",
    "    sigma_dp     = 1.0\n",
    "    max_batch    = 64\n",
    "\n",
    "    # VAE Structure\n",
    "    hidden       = 256\n",
    "    beta_kl      = 1.0\n",
    "\n",
    "    # Probing / Evaluation\n",
    "    M_probe      = 50_000\n",
    "    dirs         = 16\n",
    "    scalar_mode  = \"linear\"\n",
    "    bins         = 401\n",
    "\n",
    "    # Latent dimensions to scan\n",
    "    d_list       = [8, 16, 32, 64, 96, 128, 192, 256]\n",
    "    base_seed    = 2025\n",
    "    device       = \"cuda\" if torch.cuda.is_available() else \"cpu\"\n",
    "\n",
    "cfg = Cfg()\n",
    "torch.set_num_threads(max(1, torch.get_num_threads()))\n",
    "\n",
    "def make_neighboring_images(n:int, side:int, n_modes:int=6, seed:int=0):\n",
    "    g = torch.Generator().manual_seed(seed)\n",
    "    D = side * side\n",
    "    centers = torch.randn(n_modes, D, generator=g) * 0.8\n",
    "    mix_idx = torch.randint(0, n_modes, (n,), generator=g)\n",
    "    X = centers[mix_idx] + 0.3 * torch.randn(n, D, generator=g)\n",
    "    X = X.clamp_(-3, 3)\n",
    "    Xp = X.clone()\n",
    "    j = torch.randint(0, n_modes, (1,), generator=g).item()\n",
    "    Xp[-1] = centers[j] + 0.3 * torch.randn(D, generator=g)\n",
    "    return X.view(n, 1, side, side), Xp.view(n, 1, side, side)\n",
    "\n",
    "class VAE(nn.Module):\n",
    "    def __init__(self, in_ch: int, side: int, latent_dim: int, hidden: int):\n",
    "        super().__init__()\n",
    "        D = side * side\n",
    "        act = nn.GELU()\n",
    "        self.enc = nn.Sequential(\n",
    "            nn.Flatten(),\n",
    "            nn.Linear(D, hidden), act,\n",
    "            nn.Linear(hidden, hidden), act,\n",
    "        )\n",
    "        self.enc_mu    = nn.Linear(hidden, latent_dim)\n",
    "        self.enc_logv  = nn.Linear(hidden, latent_dim)\n",
    "        self.dec = nn.Sequential(\n",
    "            nn.Linear(latent_dim, hidden), act,\n",
    "            nn.Linear(hidden, hidden), act,\n",
    "            nn.Linear(hidden, D)\n",
    "        )\n",
    "        self.side = side\n",
    "        self.latent_dim = latent_dim\n",
    "\n",
    "    @property\n",
    "    def decoder(self):\n",
    "        return self.dec\n",
    "\n",
    "    def encode(self, x):\n",
    "        h = self.enc(x)\n",
    "        return self.enc_mu(h), self.enc_logv(h)\n",
    "\n",
    "    def reparam(self, mu, logv):\n",
    "        std = (0.5*logv).exp()\n",
    "        eps = torch.randn_like(std)\n",
    "        return mu + std * eps\n",
    "\n",
    "    def decode(self, z):\n",
    "        x = self.dec(z)\n",
    "        return x.view(-1, 1, self.side, self.side)\n",
    "\n",
    "    def forward(self, x):\n",
    "        mu, logv = self.encode(x)\n",
    "        z = self.reparam(mu, logv)\n",
    "        xh = self.decode(z)\n",
    "        return xh, mu, logv\n",
    "\n",
    "def dp_sgd_step(model: nn.Module, xb: torch.Tensor, lr: float, C: float,\n",
    "                sigma: float, noise_vec: torch.Tensor, beta_kl: float, weight_decay: float=0.0):\n",
    "    params = [p for p in model.parameters() if p.requires_grad]\n",
    "    P = sum(p.numel() for p in params)\n",
    "    bsz = xb.size(0)\n",
    "    device = xb.device\n",
    "\n",
    "    G = torch.zeros((bsz, P), device=device)\n",
    "\n",
    "    for i in range(bsz):\n",
    "        model.zero_grad(set_to_none=True)\n",
    "        x1 = xb[i:i+1]\n",
    "        xh, mu, logv = model(x1)\n",
    "        rec = 0.5 * F.mse_loss(xh, x1, reduction='sum')\n",
    "        kl  = -0.5 * torch.sum(1 + logv - mu.pow(2) - logv.exp())\n",
    "        loss = rec + beta_kl * kl\n",
    "        loss.backward()\n",
    "\n",
    "        grads = []\n",
    "        for p in params:\n",
    "            if p.grad is None:\n",
    "                grads.append(torch.zeros_like(p).reshape(-1))\n",
    "            else:\n",
    "                grads.append(p.grad.reshape(-1))\n",
    "        G[i] = torch.cat(grads)\n",
    "\n",
    "    if weight_decay > 0.0:\n",
    "        with torch.no_grad():\n",
    "            wflat = torch.cat([p.detach().reshape(-1) for p in params])\n",
    "        G = G + weight_decay * wflat.unsqueeze(0).expand_as(G)\n",
    "\n",
    "    norms = torch.linalg.vector_norm(G, dim=1) + 1e-12\n",
    "    scale = (C / norms).clamp(max=1.0).unsqueeze(1)\n",
    "    Gc = G * scale\n",
    "\n",
    "    g_bar = Gc.mean(dim=0)\n",
    "    if sigma > 0.0:\n",
    "        g_bar = g_bar + (sigma * C / bsz) * noise_vec\n",
    "\n",
    "    with torch.no_grad():\n",
    "        off = 0\n",
    "        for p in params:\n",
    "            n = p.numel()\n",
    "            p.add_(-lr * g_bar[off:off+n].view_as(p))\n",
    "            off += n\n",
    "\n",
    "def train_pair_vae(d_latent: int, seed: int) -> Tuple[VAE, VAE]:\n",
    "    torch.manual_seed(seed); np.random.seed(seed)\n",
    "    X, Xp = make_neighboring_images(cfg.n, cfg.img_side, seed=seed)\n",
    "    X, Xp = X.to(cfg.device), Xp.to(cfg.device)\n",
    "\n",
    "    mD  = VAE(1, cfg.img_side, d_latent, cfg.hidden).to(cfg.device)\n",
    "    mDp = VAE(1, cfg.img_side, d_latent, cfg.hidden).to(cfg.device)\n",
    "    mDp.load_state_dict(mD.state_dict())\n",
    "\n",
    "    n = cfg.n\n",
    "    b = max(1, min(cfg.max_batch, int(round(cfg.q * n))))\n",
    "    P = sum(p.numel() for p in mD.parameters())\n",
    "\n",
    "    g_pick = torch.Generator(device='cuda').manual_seed(seed+123)\n",
    "    base_lr = cfg.lr\n",
    "\n",
    "    for step in tqdm(range(1, cfg.steps+1), desc=f\"train VAE (d={d_latent})\", leave=False):\n",
    "        lr = base_lr * 0.5 * (1 + math.cos(math.pi * (step-1) / cfg.steps))\n",
    "\n",
    "        while True:\n",
    "            mask = torch.rand(n, generator=g_pick, device='cuda') < cfg.q\n",
    "            idx = mask.nonzero(as_tuple=False).squeeze(1)\n",
    "            if idx.numel() > 0:\n",
    "                break\n",
    "        if idx.numel() > b:\n",
    "            idx = idx[torch.randperm(idx.numel(), generator=g_pick, device='cuda')[:b]]\n",
    "        xb, xb2 = X[idx], Xp[idx]\n",
    "\n",
    "        g_noise = torch.Generator(device='cuda').manual_seed(10_000*seed + step)\n",
    "        noise_vec = torch.randn(P, generator=g_noise, device=cfg.device)\n",
    "\n",
    "        dp_sgd_step(mD,  xb,  lr, cfg.C_clip, cfg.sigma_dp, noise_vec, cfg.beta_kl, cfg.weight_decay)\n",
    "        dp_sgd_step(mDp, xb2, lr, cfg.C_clip, cfg.sigma_dp, noise_vec, cfg.beta_kl, cfg.weight_decay)\n",
    "\n",
    "    return mD.eval(), mDp.eval()\n",
    "\n",
    "@torch.no_grad()\n",
    "def scalarize_linear(decoder: nn.Module, Z: torch.Tensor, out_dim: int, dirs:int, seed:int=0) -> List[np.ndarray]:\n",
    "    torch.manual_seed(seed); np.random.seed(seed)\n",
    "    U = torch.randn(dirs, out_dim, device=Z.device)\n",
    "    U = U / (U.norm(dim=1, keepdim=True) + 1e-12)\n",
    "    Ys = []\n",
    "    B = 4096\n",
    "    for r in range(dirs):\n",
    "        u = U[r].view(-1,1)\n",
    "        ys = []\n",
    "        for i in range(0, Z.size(0), B):\n",
    "            zi = Z[i:i+B]\n",
    "            xi = decoder(zi).view(zi.size(0), -1)\n",
    "            yi = (xi @ u).squeeze(-1)\n",
    "            ys.append(yi)\n",
    "        Ys.append(torch.cat(ys,0).cpu().numpy())\n",
    "    return Ys\n",
    "\n",
    "@torch.no_grad()\n",
    "def scalarize_energy(decoder: nn.Module, Z: torch.Tensor) -> np.ndarray:\n",
    "    Ys = []\n",
    "    B = 4096\n",
    "    for i in range(0, Z.size(0), B):\n",
    "        zi = Z[i:i+B]\n",
    "        xi = decoder(zi).view(zi.size(0), -1)\n",
    "        yi = 0.5 * (xi.pow(2).sum(dim=1))\n",
    "        Ys.append(yi)\n",
    "    return torch.cat(Ys,0).cpu().numpy()\n",
    "\n",
    "def ks_vs_normal(y: np.ndarray) -> float:\n",
    "    y = (y - y.mean()) / (y.std(ddof=1)+1e-12)\n",
    "    y_sorted = np.sort(y); n = y_sorted.size\n",
    "    from math import sqrt\n",
    "    from scipy.stats import norm\n",
    "    F0 = norm.cdf(y_sorted)\n",
    "    i = np.arange(1, n+1)\n",
    "    D_plus  = np.max(i/n - F0)\n",
    "    D_minus = np.max(F0 - (i-1)/n)\n",
    "    return float(max(D_plus, D_minus))\n",
    "\n",
    "def tv_hist_vs_normal(y: np.ndarray, bins:int=401) -> float:\n",
    "    from scipy.stats import norm\n",
    "    y = (y - y.mean()) / (y.std(ddof=1)+1e-12)\n",
    "    qs = np.linspace(0.0, 1.0, bins+1)\n",
    "    qs = np.clip(qs, 1e-12, 1-1e-12)\n",
    "    edges = norm.ppf(qs)\n",
    "    hist, _ = np.histogram(y, bins=edges)\n",
    "    p = hist / max(1, hist.sum())\n",
    "    q = np.ones_like(p) / p.size\n",
    "    return 0.5 * np.abs(p - q).sum()\n",
    "\n",
    "def estimate_gamma(model:VAE, d_latent:int, M:int, mode:str, dirs:int, bins:int, seed:int) -> float:\n",
    "    g = torch.Generator(device='cuda').manual_seed(seed)\n",
    "    Z = torch.randn(M, d_latent, generator=g, device=cfg.device)\n",
    "    D = cfg.img_side * cfg.img_side\n",
    "    if mode == \"linear\":\n",
    "        Ys = scalarize_linear(model.decoder, Z, out_dim=D, dirs=dirs, seed=seed)\n",
    "        ks_vals = [ks_vs_normal(y) for y in Ys]\n",
    "        return float(2.0 * np.max(ks_vals))\n",
    "    else:\n",
    "        y = scalarize_energy(model.decoder, Z)\n",
    "        return float(2.0 * ks_vs_normal(y))\n",
    "\n",
    "def estimate_omega(mD:VAE, mDp:VAE, d_latent:int, M:int, mode:str, dirs:int, seed:int) -> float:\n",
    "    g1 = torch.Generator(device='cuda').manual_seed(seed)\n",
    "    Z  = torch.randn(M, d_latent, generator=g1, device=cfg.device)\n",
    "    D = cfg.img_side * cfg.img_side\n",
    "    def one(model):\n",
    "        if mode == \"linear\":\n",
    "            Ys = scalarize_linear(model.decoder, Z, out_dim=D, dirs=dirs, seed=seed)\n",
    "            vars_ = [np.var(y, ddof=1) for y in Ys]\n",
    "            return float(np.mean(vars_))\n",
    "        else:\n",
    "            y = scalarize_energy(model.decoder, Z)\n",
    "            return float(np.var(y, ddof=1))\n",
    "    v1 = one(mD); v2 = one(mDp)\n",
    "    bar = 0.5 * (v1 + v2)\n",
    "    return abs(v1 - v2) / max(bar, 1e-12)\n",
    "\n",
    "results = []\n",
    "for i, d_latent in enumerate(cfg.d_list):\n",
    "    seed = cfg.base_seed + 100*i\n",
    "    mD, mDp = train_pair_vae(d_latent, seed)\n",
    "    gamma_D  = estimate_gamma(mD,  d_latent, cfg.M_probe, cfg.scalar_mode, cfg.dirs, cfg.bins, seed)\n",
    "    gamma_Dp = estimate_gamma(mDp, d_latent, cfg.M_probe, cfg.scalar_mode, cfg.dirs, cfg.bins, seed+1)\n",
    "    gamma = max(gamma_D, gamma_Dp)\n",
    "    omega = estimate_omega(mD, mDp, d_latent, cfg.M_probe, cfg.scalar_mode, cfg.dirs, seed+2)\n",
    "    results.append({\"d\": d_latent, \"gamma\": gamma, \"omega\": omega})\n",
    "\n",
    "ds   = np.array([r[\"d\"] for r in results])\n",
    "gams = np.array([r[\"gamma\"] for r in results])\n",
    "omes = np.array([r[\"omega\"] for r in results])\n",
    "\n",
    "plt.figure(figsize=(6.2,4.6))\n",
    "plt.loglog(ds, gams, 'o-', label=r'$\\gamma_d$ (Gaussianization error)')\n",
    "plt.loglog(ds, omes, 's-', label=r'$\\omega$ (variance slack)')\n",
    "plt.xlabel('Latent dimension $d$')\n",
    "plt.ylabel('Scale')\n",
    "plt.title('Scaling of Gaussianization error and variance slack (VAE, DP–SGD)')\n",
    "# plt.grid(True, which='both', ls='--', alpha=0.25)\n",
    "plt.legend()\n",
    "plt.tight_layout()\n",
    "plt.show()\n",
    "\n",
    "print(\"Results:\")\n",
    "for r in results: print(r)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5deec62d-4401-418e-a9c4-bf64f10afd4e",
   "metadata": {},
   "outputs": [],
   "source": [
    "# ================== Fixed model (d0), expand effective dimension on probe/kernel side ==================\n",
    "# Frozen random features, train head only (DP-SGD, fixed-q). Then during LPK estimation, add isotropic\n",
    "# Gaussian block to the kernel to emulate extra latent dimensions r, so d_eff = d0 + r. Expect mu_eff ~ 1/sqrt(d_eff).\n",
    "import math, numpy as np, torch, torch.nn.functional as F\n",
    "import matplotlib.pyplot as plt\n",
    "from tqdm.notebook import tqdm\n",
    "\n",
    "# ---------------- Config ----------------\n",
    "class Cfg:\n",
    "    # training model (fixed)\n",
    "    d0 = 64                 # base input dim (model trained only once at d0)\n",
    "    width_factor = 2.0      # m = kappa * d0\n",
    "    n = 4000\n",
    "    noise_std = 0.1\n",
    "\n",
    "    # DP-SGD (fixed-q)\n",
    "    q = 0.05\n",
    "    C_clip = 1.0\n",
    "    sigma_dp = 5.0\n",
    "    steps = 100\n",
    "    lr = 5e-4\n",
    "    weight_decay = 1e-4\n",
    "\n",
    "    # probes & LPK\n",
    "    M = 48                  # number of Gaussian probes\n",
    "    log_every = 40\n",
    "    tau_reg = 1e-3          # relative regularization: lam = tau * mean_diag + lam0\n",
    "    lam0_frac = 1e-3        # absolute floor: lam0 = lam0_frac * (trace(K)/M)\n",
    "\n",
    "    # effective-d sweep (add r synthetic latent dims on kernel side)\n",
    "    r_list = [0, 32, 64, 128, 256, 512]   # d_eff = d0 + r\n",
    "\n",
    "    device = \"cpu\"\n",
    "    base_seed = 2026\n",
    "\n",
    "cfg = Cfg()\n",
    "\n",
    "def set_seed(s): torch.manual_seed(s); np.random.seed(s % (2**32-1))\n",
    "\n",
    "# ---------------- Data: neighboring datasets ----------------\n",
    "def make_neighboring_datasets(n, d, noise_std, seed):\n",
    "    g = torch.Generator().manual_seed(seed)\n",
    "    x = torch.randn(n, d, generator=g)\n",
    "    u = torch.randn(d, generator=g); u = u / (u.norm() + 1e-12)\n",
    "    y = x @ u + noise_std * torch.randn(n, generator=g)\n",
    "    x2, y2 = x.clone(), y.clone()\n",
    "    x2[-1] = torch.randn(d, generator=g)\n",
    "    y2[-1] = x2[-1] @ u + noise_std * torch.randn((), generator=g)\n",
    "    return (x, y), (x2, y2)\n",
    "\n",
    "# ---------------- Frozen random features ----------------\n",
    "def init_W(d, m, seed, device):\n",
    "    g = torch.Generator().manual_seed(seed)\n",
    "    W = torch.randn(m, d, generator=g, device=device) / math.sqrt(d)  # fan-in scaling\n",
    "    return W\n",
    "\n",
    "def phi_forward(W, x):  # (B,d)->(B,m)\n",
    "    return F.gelu(x @ W.t())\n",
    "\n",
    "def f_forward(a, phi):  # (m,),(B,m)->(B,)\n",
    "    return (phi @ a) / math.sqrt(a.numel())\n",
    "\n",
    "# ---------------- DP-SGD step on head (per-sample closed form) ----------------\n",
    "def dp_head_step(a, W, xb, yb, lr, C_clip, sigma, noise_vec, weight_decay):\n",
    "    m = a.numel()\n",
    "    phi = phi_forward(W, xb)                    # (b,m)\n",
    "    f   = f_forward(a, phi)                     # (b,)\n",
    "    err = f - yb\n",
    "    G = (err.unsqueeze(1) * phi) / math.sqrt(m) # per-sample grad wrt a: (b,m)\n",
    "    if weight_decay > 0:\n",
    "        G = G + weight_decay * a.detach().unsqueeze(0).expand_as(G)\n",
    "    norms = torch.linalg.vector_norm(G, dim=1) + 1e-12\n",
    "    scale = (C_clip / norms).clamp(max=1.0).unsqueeze(1)\n",
    "    Gc = G * scale\n",
    "    g_bar = Gc.mean(dim=0)\n",
    "    if sigma > 0:\n",
    "        g_bar = g_bar + (sigma * C_clip / xb.size(0)) * noise_vec\n",
    "    with torch.no_grad():\n",
    "        a.add_(-lr * g_bar)\n",
    "\n",
    "# ---------------- LPK increment on probes (ℓ_cal = f, grad wrt a = phi/√m) ----------------\n",
    "def lpk_inc_base(a, W, Z):\n",
    "    m = a.numel()\n",
    "    phiZ = phi_forward(W, Z)            # (M,m)\n",
    "    Gcal = phiZ / math.sqrt(m)          # (M,m)   (NO factor f)\n",
    "    Kinc = Gcal @ Gcal.t()              # (M,M)\n",
    "    fZ   = f_forward(a, phiZ)           # (M,)\n",
    "    return fZ.detach(), Kinc.detach()\n",
    "\n",
    "# ---------------- Train once at d0; during accumulation, build augmented K for r in r_list ----------------\n",
    "def train_and_build_augmented_K(d0, n, m, q, sigma, r_list, seed):\n",
    "    set_seed(seed); dev = cfg.device\n",
    "\n",
    "    # data & frozen features\n",
    "    (x, y), (x2, y2) = make_neighboring_datasets(n, d0, cfg.noise_std, seed=seed)\n",
    "    x, y, x2, y2 = x.to(dev), y.to(dev), x2.to(dev), y2.to(dev)\n",
    "    W = init_W(d0, m, seed+1000, dev)\n",
    "\n",
    "    # head init\n",
    "    a  = torch.zeros(m, device=dev); a2 = a.clone()\n",
    "\n",
    "    # probes (fixed across time)\n",
    "    gZ = torch.Generator().manual_seed(seed+3000)\n",
    "    Z = torch.randn(cfg.M, d0, generator=gZ, device=dev)\n",
    "\n",
    "    # kernel accumulators\n",
    "    K_base_1 = torch.zeros(cfg.M, cfg.M, device=dev)  # for D\n",
    "    K_base_2 = torch.zeros(cfg.M, cfg.M, device=dev)  # for D'\n",
    "    # for each r, augmented K^{oplus} = (K1+K2) + sum_t eta_acc * s_t * (U U^T)\n",
    "    K_aug = {r: torch.zeros(cfg.M, cfg.M, device=dev) for r in r_list}  # will hold K^{oplus}_aug (not just extra)\n",
    "\n",
    "    gnoise = torch.Generator().manual_seed(seed+4000)\n",
    "    eta_acc = 0.0\n",
    "\n",
    "    for step in tqdm(range(1, cfg.steps+1), desc=\"train (fixed d0)\", leave=False):\n",
    "        eta_t = cfg.lr * 0.5 * (1 + math.cos(math.pi * (step-1) / cfg.steps))\n",
    "        eta_acc += eta_t\n",
    "\n",
    "        # Poisson(q) minibatch (coupled)\n",
    "        while True:\n",
    "            mask = torch.rand(n, generator=gnoise) < q\n",
    "            if mask.any(): break\n",
    "        idx = mask.nonzero(as_tuple=False).squeeze(1)\n",
    "        xb, yb = x[idx], y[idx]; xb2, yb2 = x2[idx], y2[idx]\n",
    "\n",
    "        noise_vec = torch.randn(m, generator=gnoise, device=dev)\n",
    "        dp_head_step(a,  W, xb,  yb,  eta_t, cfg.C_clip, sigma, noise_vec, cfg.weight_decay)\n",
    "        dp_head_step(a2, W, xb2, yb2, eta_t, cfg.C_clip, sigma, noise_vec, cfg.weight_decay)\n",
    "\n",
    "        if (step % cfg.log_every == 0) or (step == cfg.steps):\n",
    "            _, Kinc1 = lpk_inc_base(a,  W, Z)\n",
    "            _, Kinc2 = lpk_inc_base(a2, W, Z)\n",
    "            K_base_1 += eta_acc * Kinc1\n",
    "            K_base_2 += eta_acc * Kinc2\n",
    "\n",
    "            # 本次 base 的“每个基础维度对角贡献”尺度（近似）：s_t = mean_diag / d0\n",
    "            mean_diag_step = float(torch.diag(Kinc1 + Kinc2).mean().item())\n",
    "            s_t = mean_diag_step / max(d0, 1)\n",
    "\n",
    "            # 为每个 r 叠加等方差各向同性块（U U^T），使对角期望∝ r\n",
    "            for r in r_list:\n",
    "                if r <= 0:  # r=0 仅累加base\n",
    "                    K_aug[r] += eta_acc * (Kinc1 + Kinc2)\n",
    "                else:\n",
    "                    gU = torch.Generator().manual_seed(seed + 9000 + r*100 + step)  # 可重复的随机性\n",
    "                    U = torch.randn(cfg.M, r, generator=gU, device=dev)\n",
    "                    K_aug[r] += eta_acc * (Kinc1 + Kinc2) + eta_acc * s_t * (U @ U.t())\n",
    "\n",
    "            eta_acc = 0.0\n",
    "\n",
    "    # y on probes (fixed; Δf 不随 r 变)\n",
    "    with torch.no_grad():\n",
    "        phiZ = phi_forward(W, Z)\n",
    "        y_vec = (f_forward(a, phiZ) - f_forward(a2, phiZ)).cpu().numpy()\n",
    "\n",
    "    # K^{oplus}_base\n",
    "    K_base_oplus = (K_base_1 + K_base_2).cpu().numpy()\n",
    "    K_aug_np = {r: K_aug[r].cpu().numpy() for r in r_list}\n",
    "    return y_vec, K_base_oplus, K_aug_np\n",
    "\n",
    "# ---------------- Compute mu_hat for each K (with stable regularization) ----------------\n",
    "def mu_from_K(y_vec, K_oplus, tau=1e-3, lam0_frac=1e-3):\n",
    "    mean_diag = float(np.mean(np.diag(K_oplus)))\n",
    "    mean_diag_fallback = float(np.trace(K_oplus) / max(len(K_oplus),1))\n",
    "    lam = tau * max(mean_diag, 1e-12) + lam0_frac * max(mean_diag_fallback, 1e-12)\n",
    "    K_reg = K_oplus + lam * np.eye(K_oplus.shape[0], dtype=K_oplus.dtype)\n",
    "    try:\n",
    "        alpha = np.linalg.solve(K_reg, y_vec)\n",
    "    except np.linalg.LinAlgError:\n",
    "        alpha = np.linalg.lstsq(K_reg, y_vec, rcond=None)[0]\n",
    "    rkhs_lb = float(y_vec @ alpha); rkhs_lb = max(rkhs_lb, 0.0)\n",
    "    rkhs_norm = math.sqrt(rkhs_lb + 1e-20)\n",
    "    mu_hat = rkhs_norm / math.sqrt(max(mean_diag, 1e-12))\n",
    "    # also return kernel scales for sanity plots\n",
    "    off_mean = float((K_oplus.sum() - np.trace(K_oplus)) / (K_oplus.shape[0]*(K_oplus.shape[0]-1) + 1e-12))\n",
    "    return mu_hat, rkhs_norm, mean_diag, off_mean, lam\n",
    "\n",
    "# ================= RUN =================\n",
    "set_seed(cfg.base_seed)\n",
    "m = int(max(8, round(cfg.width_factor * cfg.d0)))\n",
    "y_vec, K_base_oplus, K_aug_np = train_and_build_augmented_K(\n",
    "    d0=cfg.d0, n=cfg.n, m=m, q=cfg.q, sigma=cfg.sigma_dp,\n",
    "    r_list=cfg.r_list, seed=cfg.base_seed+7\n",
    ")\n",
    "\n",
    "# gather results over r (thus d_eff)\n",
    "rows = []\n",
    "for r in cfg.r_list:\n",
    "    mu_hat, rkhs_norm, Kdiag, Koff, lam = mu_from_K(y_vec, K_aug_np[r], tau=cfg.tau_reg, lam0_frac=cfg.lam0_frac)\n",
    "    rows.append({\"d_eff\": cfg.d0 + r, \"r\": r, \"mu_hat\": mu_hat, \"rkhs_norm\": rkhs_norm,\n",
    "                 \"K_diag\": Kdiag, \"K_off\": Koff, \"lam_used\": lam})\n",
    "\n",
    "# ================= PLOTS =================\n",
    "rows = sorted(rows, key=lambda x: x[\"d_eff\"])\n",
    "d_eff = np.array([r[\"d_eff\"] for r in rows], float)\n",
    "mu     = np.array([r[\"mu_hat\"] for r in rows], float)\n",
    "mu_s   = mu * np.sqrt(d_eff)\n",
    "Kdiag  = np.array([r[\"K_diag\"] for r in rows], float)\n",
    "Koff   = np.array([r[\"K_off\"]  for r in rows], float)\n",
    "\n",
    "# --- Object-Oriented Plotting ---\n",
    "# 1. Create a figure and a set of subplots (axes).\n",
    "# This is the main change. We'll use 'ax' for all plotting commands.\n",
    "fig, ax = plt.subplots(figsize=(5, 5)) # You can set figure size here\n",
    "# 2. Plot the data using the 'ax' object.\n",
    "ax.loglog(d_eff, mu, marker='o', label=r'$\\hat{\\mu}_{\\mathrm{eff}}$')\n",
    "# Calculate the reference line\n",
    "ref_m = mu[0] * (d_eff / d_eff[0])**(-0.5)\n",
    "ax.loglog(d_eff, ref_m, '--', label='slope -1/2 ref')\n",
    "# 3. Set labels and title using 'ax.set_*' methods.\n",
    "ax.set_xlabel(r'$d_{\\mathrm{eff}}$')\n",
    "ax.set_ylabel(r'$\\hat{\\mu}_{\\mathrm{eff}}$')\n",
    "ax.set_title('Scaling via Probe-side Expansion')\n",
    "# 4. Add the legend and grid to the 'ax' object.\n",
    "ax.legend()\n",
    "# ax.grid(True, which=\"both\", ls=\"--\", linewidth=0.5) # Optional: add a grid\n",
    "# 5. Save the figure using the 'fig' object.\n",
    "fig.savefig('probe.pdf', bbox_inches=\"tight\")\n",
    "# 6. Show the plot.\n",
    "plt.show()\n",
    "\n",
    "# 2) flatness check for mu*sqrt(d_eff)\n",
    "plt.figure()\n",
    "plt.plot(d_eff, mu_s, marker='o')\n",
    "plt.xlabel(r'$d_{\\mathrm{eff}}$'); plt.ylabel(r'$\\hat{\\mu}_{\\mathrm{eff}}\\sqrt{d_{\\mathrm{eff}}}$')\n",
    "plt.title('Flatness check (~ constant if theory holds)'); plt.show()\n",
    "\n",
    "# 3) kernel scales: diag ~ O(d_eff), off ~ const\n",
    "plt.figure()\n",
    "plt.plot(d_eff, Kdiag, marker='o', label='E[K(W,W)]')\n",
    "plt.plot(d_eff, Koff,  marker='s', label=\"E_{W,W'}[K(W,W')]\")\n",
    "plt.xlabel(r'$d_{\\mathrm{eff}}$'); plt.ylabel('kernel scale (LPK)')\n",
    "plt.title('Near-isotropy via augmented block'); plt.legend(); plt.show()\n",
    "\n",
    "# Optional table\n",
    "try:\n",
    "    import pandas as pd\n",
    "    display(pd.DataFrame(rows))\n",
    "except:\n",
    "    print(rows)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d2d32513-cda4-4866-9583-7d3695ea9804",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from scipy.stats import norm\n",
    "\n",
    "def gdp_tradeoff_full(alpha, mu):\n",
    "    alpha = np.asarray(alpha, dtype=float)\n",
    "    LARGE = 12.0\n",
    "    z = np.empty_like(alpha)\n",
    "    mid = (alpha > 0.0) & (alpha < 1.0)\n",
    "    z[mid] = norm.ppf(1.0 - alpha[mid])\n",
    "    z[alpha <= 0.0] = LARGE\n",
    "    z[alpha >= 1.0] = -LARGE\n",
    "    return norm.cdf(z - mu)\n",
    "\n",
    "def gdp_shifted(alpha, mu, gamma):\n",
    "    return np.clip(gdp_tradeoff_full(np.clip(alpha + gamma, 0, 1), mu) - gamma, 0.0, 1.0)\n",
    "\n",
    "def compute_auc(mu, gamma, mu_base, n_grid=5001):\n",
    "    grid = np.linspace(0, 1, n_grid)\n",
    "    env = gdp_shifted(grid, mu, gamma)\n",
    "    base = gdp_tradeoff_full(grid, mu_base)\n",
    "    curve = np.maximum(env, base)\n",
    "    auc = np.trapz(curve, grid)\n",
    "    return auc, curve, grid\n",
    "\n",
    "def compute_auc_base(mu, gamma, n_grid=5001):\n",
    "    grid = np.linspace(0, 1, n_grid)\n",
    "    vals = gdp_shifted(grid, mu, gamma)\n",
    "    auc = np.trapz(vals, grid)\n",
    "    return auc\n",
    "\n",
    "# Parameters\n",
    "mu0 = 0.007\n",
    "eps0 = 0.004\n",
    "mu_base = 0.933\n",
    "m_list = [1, 2, 4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096, 8192]\n",
    "\n",
    "results = []\n",
    "for m in m_list:\n",
    "    mu_m = np.sqrt(m) * mu0\n",
    "    eps_m = np.sqrt(m) * eps0\n",
    "    A_m, curve, grid = compute_auc(mu_m, eps_m, mu_base)\n",
    "    results.append((m, mu_m, eps_m, A_m))\n",
    "    print(f\"m={m:3d}, mu={mu_m:.4f}, eps={eps_m:.4f}, AUC={A_m:.4f}\")\n",
    "\n",
    "# Plotting\n",
    "ms = [r[0] for r in results]\n",
    "AUCs = [r[3] for r in results]\n",
    "\n",
    "plt.figure(figsize=(6,4))\n",
    "plt.plot(ms, AUCs, marker='o', label=\"AUC\")\n",
    "plt.axhline(compute_auc_base(mu_base, 0.0), color='r', linestyle='--', label=\"Baseline\")\n",
    "plt.xscale(\"log\")\n",
    "plt.ylim(0.2, 0.55)\n",
    "plt.xlabel(\"Number of releases $m$\")\n",
    "plt.ylabel(\"AUC\")\n",
    "plt.title(\"AUC decay with multiple releases\")\n",
    "# plt.grid(True, which=\"both\", ls=\"--\", alpha=0.5)\n",
    "plt.legend(loc=\"upper right\")\n",
    "plt.savefig('auc.pdf', bbox_inches=\"tight\")\n",
    "plt.show()"
   ]
  }
 ],
 "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.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
