{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "0c2635df",
   "metadata": {},
   "source": [
    "This notebook reproduces **Table 2** in the submission: numerical **lower bounds** on the threshold \\(\\delta_\\star\\) as a function of \\(\\varepsilon\\) (with fixed sensitivity \\(s=1\\))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "877bb5fb",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import sys\n",
    "import time\n",
    "\n",
    "# Navigate to the parent directory of the project structure\n",
    "project_dir = os.path.abspath(os.path.join(os.getcwd(), '../../'))\n",
    "src_dir = os.path.join(project_dir, 'src')\n",
    "data_dir = os.path.join(project_dir, 'data')\n",
    "fig_dir = os.path.join(project_dir, 'fig')\n",
    "os.makedirs(fig_dir, exist_ok=True)\n",
    "os.makedirs(data_dir, exist_ok=True)\n",
    "\n",
    "# Add the src directory to sys.path\n",
    "sys.path.append(src_dir)\n",
    "\n",
    "import numpy as np\n",
    "\n",
    "from highD_regime_analysis.verify_gauss_tangent import verify_tangent_test, find_u0_for_delta"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "34e1bbaf",
   "metadata": {},
   "outputs": [],
   "source": [
    "def passes_tangent(eps, delta, s=1.0, **kwargs):\n",
    "    return verify_tangent_test(eps=eps, delta=delta, s=s, **kwargs)[\"ok\"]\n",
    "\n",
    "def estimate_delta_star(\n",
    "    eps,\n",
    "    s=1.0,\n",
    "    delta_lo=1e-30,      # should PASS\n",
    "    delta_hi=0.5,        # often FAIL; if it passes, we will expand upward\n",
    "    max_expand=20,\n",
    "    max_iter=50,\n",
    "    **verify_kwargs\n",
    "):\n",
    "    # Ensure lower endpoint passes\n",
    "    if not passes_tangent(eps, delta_lo, s=s, **verify_kwargs):\n",
    "        raise RuntimeError(f\"delta_lo={delta_lo} did not pass; reduce delta_lo.\")\n",
    "\n",
    "    # Ensure upper endpoint fails (or find a failing one)\n",
    "    hi = delta_hi\n",
    "    for _ in range(max_expand):\n",
    "        if not passes_tangent(eps, hi, s=s, **verify_kwargs):\n",
    "            break\n",
    "        hi = min(0.99, hi * 1.5)\n",
    "        if hi >= 0.99:\n",
    "            # Everything passed up to 0.99 (unlikely, but handle it)\n",
    "            return 0.99\n",
    "\n",
    "    if passes_tangent(eps, hi, s=s, **verify_kwargs):\n",
    "        # Could not find a failing point\n",
    "        return hi\n",
    "\n",
    "    lo = delta_lo\n",
    "    # Bisection in log-space is typically more stable for delta\n",
    "    for _ in range(max_iter):\n",
    "        mid = 10 ** ((np.log10(lo) + np.log10(hi)) / 2.0)\n",
    "        if passes_tangent(eps, mid, s=s, **verify_kwargs):\n",
    "            lo = mid\n",
    "        else:\n",
    "            hi = mid\n",
    "\n",
    "    return lo  # largest delta found to pass (approx)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "d9cc0765",
   "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>eps</th>\n",
       "      <th>delta_star_est</th>\n",
       "      <th>u0_at_delta_star</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>0.10</td>\n",
       "      <td>0.990000</td>\n",
       "      <td>0.037182</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>0.25</td>\n",
       "      <td>0.736670</td>\n",
       "      <td>0.175853</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>0.50</td>\n",
       "      <td>0.706970</td>\n",
       "      <td>0.174354</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>1.00</td>\n",
       "      <td>0.649185</td>\n",
       "      <td>0.168789</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>2.00</td>\n",
       "      <td>0.549133</td>\n",
       "      <td>0.151499</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>4.00</td>\n",
       "      <td>0.416972</td>\n",
       "      <td>0.115643</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6</th>\n",
       "      <td>8.00</td>\n",
       "      <td>0.292170</td>\n",
       "      <td>0.072593</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>7</th>\n",
       "      <td>16.00</td>\n",
       "      <td>0.197615</td>\n",
       "      <td>0.039554</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "     eps  delta_star_est  u0_at_delta_star\n",
       "0   0.10        0.990000          0.037182\n",
       "1   0.25        0.736670          0.175853\n",
       "2   0.50        0.706970          0.174354\n",
       "3   1.00        0.649185          0.168789\n",
       "4   2.00        0.549133          0.151499\n",
       "5   4.00        0.416972          0.115643\n",
       "6   8.00        0.292170          0.072593\n",
       "7  16.00        0.197615          0.039554"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import pandas as pd\n",
    "\n",
    "eps_grid = [0.1,0.25, 0.5, 1.0, 2.0, 4.0, 8.0, 16.0]\n",
    "rows = []\n",
    "\n",
    "for eps in eps_grid:\n",
    "    dstar = estimate_delta_star(eps, s=1.0, n_left=4000, n_right=4000)\n",
    "    # also record the corresponding u0(dstar)\n",
    "    u0_star = find_u0_for_delta(eps, dstar, s=1.0)\n",
    "    rows.append({\"eps\": eps, \"delta_star_est\": dstar, \"u0_at_delta_star\": u0_star})\n",
    "\n",
    "df = pd.DataFrame(rows)\n",
    "df"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "dp_privl_venv",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.11.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
