{
  "cells": [
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "Gl7SB4inD_s3"
      },
      "source": [
        "## Analytical compraison: our analysis (lemma 1) and the bounds from Sheffet '19, theorem 2"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 630
        },
        "id": "QzXKt09iEBQx",
        "outputId": "cb4cdf6f-9266-4374-c570-3310a40a59c1"
      },
      "outputs": [],
      "source": [
        "import numpy as np\n",
        "import matplotlib.pyplot as plt\n",
        "\n",
        "# Define parameters\n",
        "n = 10000\n",
        "d = 100\n",
        "target_delta = 1 / (n**2)\n",
        "\n",
        "k = 1.5 * d\n",
        "\n",
        "sigma_values = np.logspace(np.log10(4/3), 4.0, 25)\n",
        "\n",
        "linstyles = ['-', '--', '-.']\n",
        "\n",
        "eps_opt_Matrix = []\n",
        "eps_opt_Matrix_improved = []\n",
        "\n",
        "for curr_sigma in sigma_values:\n",
        "    if curr_sigma**2 < 2.00001:\n",
        "      alpha_near_1 = np.logspace(np.log10(1.001), np.log10(curr_sigma**2), 300000)  # very fine near 1\n",
        "      alpha_values_mixing = alpha_near_1\n",
        "    else:\n",
        "      alpha_near_1 = np.logspace(np.log10(1.001), np.log10(2), 300000)  # very fine near 1\n",
        "      alpha_far = np.linspace(2.00001, curr_sigma**2, 300000)            # linear spacing up to sigma^2\n",
        "      alpha_values_mixing = np.unique(np.concatenate([alpha_near_1, alpha_far]))\n",
        "\n",
        "    KLDA_Matrix = []\n",
        "    for curr_alpha in alpha_values_mixing:\n",
        "      KLDA_Matrix.append((k / (2 * (curr_alpha - 1))) * (\n",
        "          np.log(((1.0 - 1.0 / (curr_sigma**2)) ** curr_alpha) /\n",
        "                 (1.0 - curr_alpha / (curr_sigma**2)))\n",
        "        ))\n",
        "    # Compute final functions\n",
        "\n",
        "    # Improved conversion (via [Cannone '20])\n",
        "    eps_opt_Matrix_improved.append(np.min(KLDA_Matrix + np.log1p(-1 / alpha_values_mixing) - np.log(target_delta * alpha_values_mixing) / (alpha_values_mixing - 1)))\n",
        "\n",
        "    # Standard conversion (via [Mironov '17])\n",
        "    eps_opt_Matrix.append(np.min(KLDA_Matrix + np.log(1/target_delta) / (alpha_values_mixing - 1)))\n",
        "\n",
        "# Sheffet's bound: Sheffet '19 (https://proceedings.mlr.press/v98/sheffet19a/sheffet19a.pdf) , theorem 2\n",
        "eps_val_sheffet = 2.0 * ((np.log(4.0/target_delta) + np.sqrt(2.0*k*np.log(4.0/target_delta))))/(sigma_values**2)\n",
        "\n",
        "# Plot Sheffet's Bound\n",
        "plt.plot(sigma_values**2, eps_val_sheffet,\n",
        "         label=r\"Sheffet '19, Theorem 2\",\n",
        "         linewidth=4.0,\n",
        "         linestyle='-',\n",
        "         color='blue')\n",
        "\n",
        "# Plot Gaussian Mixing (ours)\n",
        "plt.plot(sigma_values**2, eps_opt_Matrix_improved,\n",
        "         label=rf\"Gaussian Mixing: RDP (ours), converted via proposition 1\",\n",
        "         linewidth=3.0,\n",
        "         marker='*',\n",
        "         markersize=10,\n",
        "         color='orangered')\n",
        "\n",
        "\n",
        "# Labels and Title\n",
        "plt.xlabel(r\"$\\overline{\\lambda}_{\\mathrm{min}} + \\sigma^2$\", fontsize=18)\n",
        "plt.ylabel(r\"$\\varepsilon_{\\mathrm{DP}}$\", fontsize=18)\n",
        "plt.yscale('log')\n",
        "plt.xscale('log')\n",
        "plt.xticks(fontsize=18)\n",
        "plt.yticks(fontsize=18)\n",
        "plt.title(\n",
        "    rf\"$\\delta_{{\\mathrm{{DP}}}} = \\frac{{1}}{{n^2}},\\quad n = {n},\\quad d = {d},\\quad \\frac{{k}}{{d}} = \\frac{{3}}{{2}}$\",\n",
        "    fontsize=18,\n",
        "    fontweight=\"bold\",\n",
        "    loc=\"center\",\n",
        ")\n",
        "\n",
        "plt.legend(\n",
        "    loc=\"upper center\",\n",
        "    bbox_to_anchor=(0.5, -0.2),  # move further down\n",
        "    fontsize=14,\n",
        "    frameon=False,\n",
        "    ncol=1\n",
        ")\n",
        "\n",
        "plt.savefig(\"Comparison_Sheffet_Ours_k_\" + str(k) + \"_d_\" + str(d) + \".pdf\", format=\"pdf\", bbox_inches=\"tight\")\n",
        "plt.show()"
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "NO-wg8S11Z19"
      },
      "source": [
        "# Ratio between bounds"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 582
        },
        "id": "YGVZqC-Y1bOg",
        "outputId": "f99bc695-ef27-4a74-fa9c-505fa17001f7"
      },
      "outputs": [],
      "source": [
        "# Plot Sheffet's Bound\n",
        "plt.plot(sigma_values**2, eps_val_sheffet/eps_opt_Matrix_improved,\n",
        "         linewidth=3.0,\n",
        "         marker='*',\n",
        "         markersize=10,\n",
        "         color='black')\n",
        "\n",
        "# Labels and Title\n",
        "plt.xlabel(r\"$\\overline{\\lambda}_{\\mathrm{min}} + \\sigma^2$\", fontsize=18)\n",
        "plt.ylabel(r\"Bounds Ratio\", fontsize=18)\n",
        "plt.xscale('log')\n",
        "plt.xticks(fontsize=18)\n",
        "plt.yticks(fontsize=18)\n",
        "plt.title(\n",
        "    rf\"$\\delta_{{\\mathrm{{DP}}}} = \\frac{{1}}{{n^2}},\\quad n = {n},\\quad d = {d},\\quad \\frac{{k}}{{d}} = \\frac{{3}}{{2}}$\",\n",
        "    fontsize=18,\n",
        "    fontweight=\"bold\",\n",
        "    loc=\"center\",\n",
        ")\n",
        "\n",
        "plt.legend(\n",
        "    loc=\"upper center\",\n",
        "    bbox_to_anchor=(0.5, -0.2),  # move further down\n",
        "    fontsize=14,\n",
        "    frameon=False,\n",
        "    ncol=1\n",
        ")\n",
        "\n",
        "plt.savefig(\"Comparison_Ratio_Sheffet_Ours_k_\" + str(k) + \"_d_\" + str(d) + \".pdf\", format=\"pdf\", bbox_inches=\"tight\")\n",
        "plt.show()"
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {},
      "source": [
        "## tCDP Approximation"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {},
      "outputs": [],
      "source": [
        "import numpy as np\n",
        "import matplotlib.pyplot as plt\n",
        "\n",
        "# Define parameters\n",
        "n = 10000\n",
        "d = 100\n",
        "target_delta = 1 / (n**2)\n",
        "\n",
        "k = 1.5 * d\n",
        "\n",
        "# Sheffet's bound\n",
        "# sigma_values = np.logspace(0.25, 2.5, 20)\n",
        "gamma2_value = 500.0\n",
        "\n",
        "alpha_near_1 = np.logspace(np.log10(1.0001), np.log10(2), 50000)  # High resolution near 1\n",
        "alpha_far = np.linspace(2.0001, gamma2_value, 50000)\n",
        "alpha_vals = np.unique(np.concatenate([alpha_near_1, alpha_far]))\n",
        "\n",
        "\n",
        "KLDA_Matrix = ((k / (2 * (alpha_vals - 1))) * (\n",
        "          np.log(((1.0 - 1.0 / (gamma2_value)) ** alpha_vals) /\n",
        "                 (1.0 - alpha_vals / (gamma2_value)))\n",
        "        ))\n",
        "\n",
        "\n",
        "# Plot Sheffet's Bound\n",
        "plt.plot(alpha_vals, k*alpha_vals/(2.0 * (gamma2_value**2)),\n",
        "         label=r\"tCDP Bound: $\\frac{k\\alpha}{2\\gamma^2}$\",\n",
        "         linewidth=4.0,\n",
        "         linestyle='-.',\n",
        "         color='red')\n",
        "\n",
        "plt.plot(alpha_vals, KLDA_Matrix,\n",
        "         label=r\"Closed form $\\varphi$\",\n",
        "         linewidth=4.0,\n",
        "         linestyle='-',\n",
        "         color='blue')\n",
        "\n",
        "plt.plot(alpha_vals, k*alpha_vals/(4.0 * (gamma2_value**2)),\n",
        "         label=r\"Taylor: $\\frac{k\\alpha}{4\\gamma^2}$\",\n",
        "         linewidth=4.0,\n",
        "         linestyle='--',\n",
        "         color='green')\n",
        "\n",
        "\n",
        "plt.axvline(x=(2.0 / 5.0) * gamma2_value,\n",
        "            color='black',\n",
        "            linestyle=':',\n",
        "            linewidth=2.0,\n",
        "            label=r\"$\\alpha = \\frac{2}{5}\\gamma$\")\n",
        "\n",
        "\n",
        "# Labels and Title\n",
        "plt.xlabel(r\"$\\alpha$\", fontsize=18)\n",
        "plt.ylabel(r\"$\\varphi$ and the tCDP Bound\", fontsize=18)\n",
        "plt.yscale('log')\n",
        "plt.xscale('log')\n",
        "plt.xticks(fontsize=18)\n",
        "plt.yticks(fontsize=18)\n",
        "plt.title(\n",
        "    rf\"$\\delta_{{\\mathrm{{DP}}}}=\\frac{{1}}{{n^2}},\\quad n={n},\\quad d={d},\\quad\\frac{{k}}{{d}}=\\frac{{3}}{{2}},\\quad\\gamma = {gamma2_value}$\",\n",
        "    fontsize=18,\n",
        "    fontweight=\"bold\",\n",
        "    loc=\"center\",\n",
        ")\n",
        "\n",
        "plt.legend(\n",
        "    loc=\"upper center\",\n",
        "    bbox_to_anchor=(0.5, -0.2),  # move further down\n",
        "    fontsize=14,\n",
        "    frameon=False,\n",
        "    ncol=1\n",
        ")\n",
        "\n",
        "plt.show()"
      ]
    }
  ],
  "metadata": {
    "colab": {
      "provenance": []
    },
    "kernelspec": {
      "display_name": "unlearning_layerwise_tuning",
      "language": "python",
      "name": "python3"
    },
    "language_info": {
      "codemirror_mode": {
        "name": "ipython",
        "version": 3
      },
      "file_extension": ".py",
      "mimetype": "text/x-python",
      "name": "python",
      "nbconvert_exporter": "python",
      "pygments_lexer": "ipython3",
      "version": "3.12.4"
    }
  },
  "nbformat": 4,
  "nbformat_minor": 0
}
