{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "deadly-rolling",
   "metadata": {},
   "source": [
    "## Probability of existence of consistent attack in the well-specified case"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "together-ghost",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from scipy.special import erf"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4d3ef031",
   "metadata": {},
   "outputs": [],
   "source": [
    "d = 10"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "tender-wales",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_p_star(p):\n",
    "    if p==1:\n",
    "        return np.inf\n",
    "    elif p==np.inf:\n",
    "        return 1\n",
    "    else: \n",
    "        return p / (p - 1)\n",
    "    \n",
    "def lp_norm(x, p):\n",
    "    if p == np.inf:\n",
    "        return max(x)\n",
    "    else:\n",
    "        return np.sum(x**p)**(1/p)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "christian-scholar",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Construct theta_star and hat_theta\n",
    "m = 0.5\n",
    "theta_star = np.zeros(d)\n",
    "theta_star[0] = np.sqrt(d)\n",
    "# hat_theta = m * theta_star + np.sqrt(d * (1 - m**2)) * np.array([0, 1] + [0] * (d - 2))\n",
    "v = np.sqrt(d * (1 - m**2)) / np.sqrt(d - 1) * np.concatenate(([0], np.ones(d - 1)))\n",
    "hat_theta = m * theta_star + v"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "consecutive-thumb",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_analytic(theta_star, hat_theta, eps, p=2):\n",
    "    d = len(theta_star)\n",
    "    p_star = get_p_star(p)\n",
    "    m = theta_star.dot(hat_theta)/d\n",
    "    theta_perp = hat_theta - m * theta_star\n",
    "        \n",
    "    return erf(eps * np.sqrt(d) *np.linalg.norm(theta_perp, ord=p_star) / np.linalg.norm(hat_theta) * 1/np.sqrt(2)) "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "tested-librarian",
   "metadata": {},
   "outputs": [],
   "source": [
    "p=np.inf\n",
    "# Parameters\n",
    "n = 10_000  # number of samples\n",
    "\n",
    "# Sample x\n",
    "x = np.random.normal(scale=1/np.sqrt(d), size=(n, d))\n",
    "\n",
    "# Compute L^{p*}-norm difference\n",
    "p_star = get_p_star(p)\n",
    "diff_norm_pstar = np.linalg.norm(hat_theta - m * theta_star, ord=p_star)\n",
    "\n",
    "# Estimate probabilities over epsilons\n",
    "epsilons = np.linspace(0, 3, 100)\n",
    "eps_num = np.linspace(0, 3, 20)\n",
    "probs = [np.mean(eps * diff_norm_pstar >= np.abs(x.dot(hat_theta))) for eps in eps_num]\n",
    "\n",
    "analytical_probs = get_analytic(theta_star, hat_theta, epsilons, p=p)\n",
    "\n",
    "# Plot\n",
    "plt.figure()\n",
    "plt.plot(eps_num, probs, 'x')\n",
    "plt.plot(epsilons, analytical_probs, '-')\n",
    "\n",
    "plt.xlabel(r'$\\varepsilon$')\n",
    "plt.ylabel(r'prob. of existence')\n",
    "plt.title(r'm=0.5, p=2, d=5')\n",
    "plt.grid(True)\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "54eaa7bd",
   "metadata": {},
   "outputs": [],
   "source": [
    "import sys\n",
    "import os\n",
    "\n",
    "# print the current working directory\n",
    "print(\"Current working directory:\", os.getcwd())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a57f4523",
   "metadata": {},
   "outputs": [],
   "source": [
    "width = 225.84377 * 1.25\n",
    "\n",
    "def set_size(width, fraction=1, subplots=(1, 1)):\n",
    "    if width == \"thesis\":\n",
    "        width_pt = 426.79135\n",
    "    elif width == \"beamer\":\n",
    "        width_pt = 307.28987\n",
    "    else:\n",
    "        width_pt = width\n",
    "\n",
    "    fig_width_pt = width_pt * fraction\n",
    "    inches_per_pt = 1 / 72.27\n",
    "\n",
    "    golden_ratio = (5**0.5 - 1) / 2\n",
    "\n",
    "    fig_width_in = fig_width_pt * inches_per_pt\n",
    "    fig_height_in = fig_width_in  * (subplots[0] / subplots[1]) #/ (golden_ratio)\n",
    "\n",
    "    return (fig_width_in, fig_height_in)\n",
    "\n",
    "tuple_size = set_size(width, fraction=0.33)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "weird-annual",
   "metadata": {},
   "outputs": [],
   "source": [
    "m = 0.5\n",
    "d = 5\n",
    "n = 1000\n",
    "\n",
    "plt.style.use(\"./latex_ready.mplstyle\")\n",
    "\n",
    "fig, ax = plt.subplots(\n",
    "    1, 1, sharex=True, figsize=(tuple_size[0], tuple_size[1]), gridspec_kw={\"hspace\": 0}\n",
    ")\n",
    "fig.subplots_adjust(left=0.20)\n",
    "fig.subplots_adjust(bottom=0.18)\n",
    "fig.subplots_adjust(top=0.98)\n",
    "fig.subplots_adjust(right=0.96)\n",
    "\n",
    "for p in [1, 2, 3, 4, 5, np.inf]:\n",
    "    p_star = get_p_star(p)\n",
    "    diff_norm_pstar = np.linalg.norm(hat_theta - m * theta_star, ord=p_star)\n",
    "    \n",
    "    epsilons = np.logspace(-1.7, 0.3, 100)\n",
    "    analytical_probs = get_analytic(theta_star, hat_theta, epsilons, p=p)\n",
    "\n",
    "    eps_num = np.logspace(-1.7, 0.3, 20)\n",
    "    probs = [np.mean(eps * diff_norm_pstar >= np.abs(x.dot(hat_theta))) for eps in eps_num]\n",
    "        \n",
    "    pl = ax.plot(epsilons, analytical_probs, '-', linewidth=1.5,label='$q={}$'.format(p) if p != np.inf else '$q=\\\\infty$')\n",
    "    color = pl[0].get_color()\n",
    "    \n",
    "    ax.plot(eps_num, probs, '.', c=color, markersize=3)\n",
    "    # ax.plot(eps_num * d**(1/p_star), probs, '.', markersize=3)\n",
    "    \n",
    "    ax.set_xlabel(r'$\\varepsilon$', labelpad=-2.0)\n",
    "    ax.set_ylabel(r'prob. of existence', labelpad=1.0)\n",
    "    ax.set_xscale('log')\n",
    "#     plt.yscale('log')\n",
    "\n",
    "plt.minorticks_off()\n",
    "legend = ax.legend(handlelength=1, ncols=2,frameon=False, loc='lower right', bbox_to_anchor=(1.02, 0.95), labelspacing=0.3, columnspacing=0.75)\n",
    "ax.grid(True)\n",
    "# plt.tight_layout()\n",
    "# legend = plt.legend(bbox_to_anchor=(1, 1), loc='lower right')\n",
    "# plt.savefig('myplot.png', bbox_extra_artists=[legend], bbox_inches='tight', dpi=300)\n",
    "plt.savefig('prob_norms_tiny.pdf',  bbox_extra_artists=[legend], bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "045a9c6d",
   "metadata": {},
   "outputs": [],
   "source": [
    "p"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "medieval-overview",
   "metadata": {},
   "outputs": [],
   "source": [
    "p = 2\n",
    "p_star = get_p_star(p)\n",
    "m = 0.5\n",
    "n = 1000  # number of samples\n",
    "\n",
    "plt.style.use(\"./latex_ready.mplstyle\")\n",
    "\n",
    "fig, ax = plt.subplots(\n",
    "    1, 1, sharex=True, figsize=(tuple_size[0], tuple_size[1]), gridspec_kw={\"hspace\": 0}\n",
    ")\n",
    "fig.subplots_adjust(left=0.20)\n",
    "fig.subplots_adjust(bottom=0.18)\n",
    "fig.subplots_adjust(top=0.98)\n",
    "fig.subplots_adjust(right=0.96)\n",
    "\n",
    "for d in [2 ** i for i in range(4, 16, 2)]:\n",
    "    x = np.random.normal(scale=1/np.sqrt(d), size=(n, d))\n",
    "    theta_star = np.zeros(d)\n",
    "    theta_star[0] = np.sqrt(d)\n",
    "    v = np.sqrt(d * (1 - m**2)) / np.sqrt(d - 1) * np.concatenate(([0], np.ones(d - 1)))\n",
    "    hat_theta = m * theta_star + v    \n",
    "    \n",
    "    diff_norm_pstar = np.linalg.norm(hat_theta - m * theta_star, ord=p_star)\n",
    "    \n",
    "    epsilons = np.logspace(-3.5, 0.1, 100)\n",
    "    analytical_probs = get_analytic(theta_star, hat_theta, epsilons, p=p)\n",
    "\n",
    "    eps_num = np.logspace(-3.5, 0.1, 20)\n",
    "    probs = [np.mean(eps * diff_norm_pstar >= np.abs(x.dot(hat_theta))) for eps in eps_num]\n",
    "        \n",
    "    pl = ax.plot(epsilons, analytical_probs,  '-',linewidth=1.5, label='$d$={}'.format(d))\n",
    "    color = pl[0].get_color()\n",
    "        \n",
    "    ax.plot(eps_num, probs, '.', c=color, markersize=3)\n",
    "    \n",
    "    ax.set_xlabel(r'$\\varepsilon$', labelpad=-2.0)\n",
    "    ax.set_ylabel(r'prob. of existence', labelpad=1.0)\n",
    "    ax.set_xscale('log')\n",
    "#     plt.yscale('log')\n",
    "\n",
    "legend = ax.legend(handlelength=1, ncols=2,frameon=False,loc='lower right', bbox_to_anchor=(1.05, 0.95), labelspacing=0.3, columnspacing=0.75)\n",
    "ax.grid(True)\n",
    "    \n",
    "plt.savefig('prob_ds_p={}_tiny.pdf'.format(p), bbox_extra_artists=[legend], bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "alone-apparatus",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "honey-television",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e93e0b96",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": ".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.13.11"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
