{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "f72f8e35-2d08-4ecf-ae92-cc93bf9ec433",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import scipy.special as special\n",
    "\n",
    "from sklearn.model_selection import train_test_split\n",
    "\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "import torch.optim as optim\n",
    "\n",
    "from torch.utils.data import Subset, DataLoader\n",
    "\n",
    "from torchvision.datasets import CIFAR10, MNIST\n",
    "from torchvision.transforms import Compose, ToTensor, Resize, Normalize"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "47aabe63-a653-4760-895b-2bb09aaeee5a",
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "\n",
    "sns.set(rc={'figure.figsize': (8,5)}, font_scale=1.2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "e4dcb5bd-68bb-4d42-9c05-6f1d0f2cd61a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "device(type='cuda', index=2)"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "device = torch.device('cuda:2')\n",
    "device"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "7d483803-6ad0-4899-8472-2d4ab2313b81",
   "metadata": {},
   "outputs": [],
   "source": [
    "im_size = 7\n",
    "dataset_size = 59990\n",
    "batch_size = dataset_size\n",
    "num_classes = 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "2a6da8b8-5433-4412-948b-ee7b714f95c7",
   "metadata": {},
   "outputs": [],
   "source": [
    "dataset_name = 'MNIST binary'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "63a8fddf-d038-4a0e-99cb-35112dbea78e",
   "metadata": {},
   "outputs": [],
   "source": [
    "train_data = MNIST(\n",
    "    'data/', train=True, download=True, transform=Compose(\n",
    "        [\n",
    "            ToTensor(), \n",
    "            Normalize((0.1307,), (0.3081,)),\n",
    "            # Normalize((0.4914, 0.4822, 0.4465), (0.247, 0.243, 0.261)),\n",
    "            Resize(im_size)\n",
    "        ]\n",
    "    )\n",
    ")\n",
    "test_data = MNIST(\n",
    "    'data/', train=False, download=True, transform=Compose(\n",
    "        [\n",
    "            ToTensor(), \n",
    "            Normalize((0.1307,), (0.3081,)),\n",
    "            # Normalize((0.4914, 0.4822, 0.4465), (0.247, 0.243, 0.261)),\n",
    "            Resize(im_size)\n",
    "        ]\n",
    "    )\n",
    ")\n",
    "\n",
    "# input_shape = (3, im_size, im_size)\n",
    "input_shape = (1, im_size, im_size)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "e4fdc482-d57e-4ceb-ac41-f3942f47c1cc",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1 7 9 9 6 0 6 1 7 6 0 9 2 8 0 0 2 3 1 9 6 2 8 4 7 2 9 1 2 6 0 1 0 2 9 2 2\n",
      " 9 2 4 3 2 7 5 2 2 9 2 0 0 7 8 5 9 8 2 1 1 9 4 3 2 1 7 1 6 4 5 8 1 8 4 9 9\n",
      " 7 9 6 0 4 9 9 3 5 8 9 2 5 0 7 9 7 2 6 9 1 3 1 1 8 1]\n"
     ]
    }
   ],
   "source": [
    "np.random.seed(0)\n",
    "\n",
    "targets = np.array(train_data.targets)\n",
    "train_idx, _ = train_test_split(\n",
    "    np.arange(len(train_data.data)), train_size=dataset_size, stratify=targets\n",
    ")\n",
    "print(targets[train_idx][:100])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "fa101398-6ada-4a83-8e42-1eb655ee5e38",
   "metadata": {},
   "outputs": [],
   "source": [
    "train_data = Subset(train_data, train_idx)\n",
    "train_loader = DataLoader(train_data, batch_size=batch_size, shuffle=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "dde13028-ebae-49b1-bbbf-6004bc332180",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "10000\n"
     ]
    }
   ],
   "source": [
    "test_idx = np.arange(len(test_data.data))\n",
    "test_dataset_size = len(test_idx)\n",
    "print(test_dataset_size)\n",
    "\n",
    "test_data = Subset(test_data, test_idx)\n",
    "test_loader = DataLoader(test_data, batch_size=batch_size, shuffle=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "89bcf34e-4dd8-48ed-9202-c4c721a67e0d",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([59990, 1, 7, 7])\n",
      "torch.Size([59990])\n",
      "tensor([79.1850, 30.4912, 24.5800, 21.5613, 20.3418, 16.3136, 15.3170, 13.9628,\n",
      "        12.8406, 12.1691, 11.3322, 10.4198,  9.4705,  8.6260,  8.2682,  7.5432,\n",
      "         6.8728,  6.2907,  6.1489,  5.9556,  5.0747,  4.8428,  4.5955,  4.2184,\n",
      "         3.8834,  3.5765,  3.4975,  3.0836,  2.6887,  2.6316,  2.3903,  2.3161,\n",
      "         2.2716,  1.7109,  1.6743,  1.5192,  1.3588,  1.3296,  1.2605,  1.1576,\n",
      "         0.9907,  0.9034,  0.7244,  0.6586,  0.4404,  0.3120,  0.2264,  0.1948,\n",
      "         0.1063], device='cuda:2')\n"
     ]
    }
   ],
   "source": [
    "X_train = []\n",
    "y_train = []\n",
    "for X, y in train_loader:\n",
    "    X_train.append(X)\n",
    "    y_train.append(y)\n",
    "    \n",
    "X_train = torch.cat(X_train).to(device)\n",
    "y_train = torch.cat(y_train).to(device)\n",
    "print(X_train.shape)\n",
    "\n",
    "X_train_flat = X_train.view(X_train.shape[0], -1)\n",
    "X_train_norms = torch.norm(X_train_flat, dim=1)\n",
    "print(X_train_norms.shape)\n",
    "X_train_flat = X_train_flat / torch.max(X_train_norms)\n",
    "U, S, Vt = torch.linalg.svd(X_train_flat, full_matrices=False)\n",
    "print(S)\n",
    "\n",
    "X_train_white = U @ torch.diag(torch.ones_like(S)) @ Vt * dataset_size**0.5\n",
    "inv_root_Sigma = Vt.T @ torch.diag(S**(-1)) @ Vt * dataset_size**0.5"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "428fd72b-f1fe-4859-82ca-bed46cb166a5",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "2303.096435546875 3.0931183105747997 6.422111988067627 7.0000141688612185 63.570823669433594\n"
     ]
    }
   ],
   "source": [
    "inv_root_Sigma_norm = (dataset_size**0.5 / S[-1]).item()\n",
    "root_Sigma_norm = (S[0] * dataset_size**(-0.5)).item()\n",
    "av_x_white_norm = torch.mean(torch.norm(X_train_white, dim=1)).item()\n",
    "av_x_white_sqr_norm = torch.mean(torch.norm(X_train_white, dim=1)**2).item()\n",
    "max_x_white_norm = torch.max(torch.norm(X_train_white, dim=1)).item()\n",
    "print(inv_root_Sigma_norm, 1/root_Sigma_norm, av_x_white_norm, av_x_white_sqr_norm**0.5, max_x_white_norm)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "64a550d3-72d9-4ae7-9306-e1c18c511b14",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(49, device='cuda:2')\n",
      "0.6479544639587402\n",
      "tensor(0.2901, device='cuda:2')\n"
     ]
    }
   ],
   "source": [
    "X_train_white_plus = torch.linalg.pinv(X_train_white)\n",
    "# X_train_white_plus = X_train_white.T\n",
    "y_train = 2 * (y_train >= 5).float() - 1\n",
    "print(torch.linalg.matrix_rank(X_train_white))\n",
    "s = torch.norm(X_train_white_plus @ y_train).item()\n",
    "print(s)\n",
    "train_loss_linreg = torch.mean((X_train_white @ X_train_white_plus @ y_train - y_train) ** 2) / 2\n",
    "print(train_loss_linreg)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "e0db27d5-6304-413b-a5bb-f32861a8d4a6",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.6479559744332388"
      ]
     },
     "execution_count": 35,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.norm(X_train_white.T @ y_train).item() / dataset_size"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "id": "091bad21-2a3a-4f95-9802-8a6f710864b4",
   "metadata": {
    "scrolled": true,
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([10000, 1, 7, 7])\n",
      "tensor([33.0610, 12.5206, 10.0092,  9.0002,  8.2585,  6.7732,  6.3750,  5.6673,\n",
      "         5.1219,  5.0890,  4.4916,  4.3418,  3.9325,  3.4930,  3.3863,  3.0801,\n",
      "         2.7669,  2.5761,  2.4496,  2.3932,  2.0744,  1.9566,  1.8427,  1.7397,\n",
      "         1.5595,  1.4289,  1.4027,  1.2842,  1.0831,  1.0555,  0.9794,  0.9273,\n",
      "         0.9105,  0.6759,  0.6526,  0.6392,  0.5443,  0.5320,  0.4976,  0.4549,\n",
      "         0.3889,  0.3420,  0.2884,  0.2520,  0.1876,  0.1238,  0.0848,  0.0636,\n",
      "         0.0357], device='cuda:2')\n",
      "tensor([116.2245, 112.0190, 110.3590, 109.6852, 108.9274, 108.2066, 107.4744,\n",
      "        106.3235, 105.8339, 105.4430, 104.6152, 104.1092, 104.0565, 103.7642,\n",
      "        103.1363, 103.0296, 102.1925, 101.9924, 101.4685, 101.2417, 101.0700,\n",
      "        100.3814,  99.5089,  99.1541,  99.0084,  98.6531,  97.8803,  97.7016,\n",
      "         97.2115,  96.6019,  96.3328,  95.7006,  95.2159,  94.8506,  94.3279,\n",
      "         94.0868,  93.5466,  93.2226,  92.5550,  91.6302,  91.0339,  90.7011,\n",
      "         90.2081,  88.5382,  88.2971,  86.1953,  85.1127,  83.1225,  71.3603],\n",
      "       device='cuda:2')\n"
     ]
    }
   ],
   "source": [
    "X_test = []\n",
    "y_test = []\n",
    "for X, y in test_loader:\n",
    "    X_test.append(X)\n",
    "    y_test.append(y)\n",
    "    \n",
    "X_test = torch.cat(X_test).to(device)\n",
    "y_test = torch.cat(y_test).to(device)\n",
    "print(X_test.shape)\n",
    "\n",
    "X_test_flat = X_test.view(X_test.shape[0], -1)\n",
    "X_test_flat = X_test_flat / torch.max(X_train_norms)\n",
    "print(torch.linalg.svdvals(X_test_flat))\n",
    "X_test_white = X_test_flat @ inv_root_Sigma\n",
    "print(torch.linalg.svdvals(X_test_white))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "id": "7214b0f1-9141-42c6-9489-bc8814ba26e7",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "63.570823669433594\n"
     ]
    }
   ],
   "source": [
    "max_x_white_norm = max(max_x_white_norm, torch.max(torch.norm(X_test_white, dim=1)).item())\n",
    "print(max_x_white_norm)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "id": "128cfe25-e837-4483-8ec1-2d3e9b232bb0",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.27772387862205505 0.8305999636650085\n",
      "torch.Size([10000])\n"
     ]
    }
   ],
   "source": [
    "y_test = 2 * (y_test >= 5).float() - 1\n",
    "logits_linear = X_test_white @ X_train_white_plus @ y_train\n",
    "test_loss_linreg = torch.mean((logits_linear - y_test) ** 2).item() / 2\n",
    "test_acc_linreg = torch.mean((logits_linear * y_test > 0).float()).item()\n",
    "print(test_loss_linreg, test_acc_linreg)\n",
    "print(logits_linear.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "id": "b8a35f22-695e-4a33-a656-233fe9e2e907",
   "metadata": {},
   "outputs": [],
   "source": [
    "class FCNet(nn.Module):\n",
    "    \n",
    "    def __init__(\n",
    "        self, input_shape, output_size, width, num_hidden, bias=False\n",
    "    ):\n",
    "        super(FCNet, self).__init__()\n",
    "        \n",
    "        assert num_hidden >= 1\n",
    "        \n",
    "        input_dim = int(np.prod(input_shape))\n",
    "        \n",
    "        self.linear_layers = nn.ModuleList()\n",
    "        self.input_layer = nn.Linear(input_dim, width, bias=bias)\n",
    "        for _ in range(num_hidden-1):\n",
    "            self.linear_layers.append(nn.Linear(width, width, bias=bias))\n",
    "        self.linear_layers.append(nn.Linear(width, output_size, bias=bias))\n",
    "            \n",
    "        \n",
    "    def forward(self, X, epsilon):\n",
    "        activation = lambda x, eps: F.leaky_relu(x, negative_slope=1.-eps)\n",
    "        X = X.view(X.shape[0], -1)\n",
    "        X = self.input_layer(X)\n",
    "        for layer in self.linear_layers:\n",
    "            X = activation(X, epsilon)\n",
    "            X = layer(X)\n",
    "        return X"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 49,
   "id": "7e956efe-08bb-4e55-9ba0-202855ff8a90",
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_acc(logits, y, gamma, soft_risk):\n",
    "    if soft_risk and gamma > 1e-8:\n",
    "        return torch.mean(torch.clamp(logits * y / gamma, min=0, max=1))\n",
    "    else:\n",
    "        return torch.mean((logits * y > gamma).float())\n",
    "\n",
    "\n",
    "def get_loss_acc(\n",
    "    model, epsilon, loader, optimizer=None, scheduler=None, inv_q=0., beta_L=None, soft_risk=False\n",
    "):\n",
    "    inv_q_is_list = isinstance(inv_q, list)\n",
    "    if not inv_q_is_list:\n",
    "        if inv_q < 1e-8:\n",
    "            beta_L = 0.\n",
    "        elif beta_L is not None:\n",
    "            raise ValueError(\"If inv_q > 0, beta_L should be specified\")\n",
    "        inv_q = [inv_q]\n",
    "        \n",
    "    cum_loss = 0\n",
    "    cum_proj_loss = 0\n",
    "    cum_acc = []\n",
    "    for _ in inv_q_list:\n",
    "        cum_acc.append(0)\n",
    "    data_size = 0\n",
    "    \n",
    "    loss_fun = nn.MSELoss()\n",
    "\n",
    "    for X, y in loader:\n",
    "        X = X.to(device)\n",
    "        y = y.to(device)\n",
    "\n",
    "        logits = model(X, epsilon=epsilon).view(-1)\n",
    "        loss = loss_fun(logits, y) / 2\n",
    "        cum_loss += loss.item() * len(y)\n",
    "\n",
    "        for j, g in enumerate(inv_q):\n",
    "            acc = compute_acc(logits, y, g*beta_L, soft_risk=soft_risk)\n",
    "            cum_acc[j] += acc.item() * len(y)\n",
    "            \n",
    "        cum_proj_loss += (0.5 * torch.norm(X.T @ (y - logits))**2).item()\n",
    "        \n",
    "        data_size += len(y)\n",
    "        \n",
    "        if optimizer is not None:\n",
    "            optimizer.zero_grad()\n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "            if scheduler is not None:\n",
    "                scheduler.step()\n",
    "                \n",
    "    mean_acc = {}\n",
    "    for j, g in enumerate(inv_q):\n",
    "        mean_acc[g] = cum_acc[j] / data_size\n",
    "        \n",
    "    if not inv_q_is_list:\n",
    "        mean_acc = mean_acc[inv_q[0]]\n",
    "\n",
    "    return cum_loss / data_size, mean_acc, cum_proj_loss / data_size"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "id": "3fcc5ecb-d4cb-4757-a359-3d9699230941",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_bar_s(L, beta, eps, s, rho, r=1):\n",
    "    return (1-eps) * (s + rho**0.5 * beta**L) + eps * (L-1) * (1 + rho**0.5 * beta**L) * r**0.5\n",
    "\n",
    "def get_bar_1(L, beta, rho):\n",
    "    return 1 + rho**0.5 * beta**L"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 50,
   "id": "c66941eb-a9d2-4088-a2d2-12d12497e651",
   "metadata": {},
   "outputs": [],
   "source": [
    "import mpmath\n",
    "\n",
    "def get_deviation_bound(t, L, s, beta, eps, delta, m, d, srk, loss_proj_decay=False):\n",
    "    def gamma_inc(x, a):\n",
    "        return mpmath.gammainc(x, a=a)\n",
    "    \n",
    "    bar_1 = get_bar_1(L=L, beta=beta, rho=srk)\n",
    "    \n",
    "    if loss_proj_decay:\n",
    "        bar_s_rho = get_bar_s(L=L, beta=beta, eps=eps, s=s, rho=srk, r=srk)\n",
    "        bar_s = get_bar_s(L=L, beta=beta, eps=eps, s=s, rho=srk, r=1)\n",
    "        \n",
    "        bar_beta = beta * bar_s_rho / bar_s\n",
    "        hat_s = bar_s * L / (1+(L-1) * srk**0.5)\n",
    "    \n",
    "        bar_s_rho_eps0 = get_bar_s(L=L, beta=beta, eps=0, s=s, rho=srk, r=srk)\n",
    "        bar_s_eps0 = get_bar_s(L=L, beta=beta, eps=0, s=s, rho=srk, r=1)\n",
    "        bar_beta_eps0 = beta * bar_s_rho_eps0 / bar_s_eps0\n",
    "\n",
    "        if L == 2:\n",
    "            u = bar_beta * np.exp(bar_s * t)\n",
    "            u_eps0 = bar_beta_eps0 * np.exp(bar_s_eps0 * t)\n",
    "            # print(type(bar_1), type(bar_s), type(hat_s), type(srk))\n",
    "            w = lambda x: bar_1/bar_s * special.expi(-x**2/hat_s) - srk**0.5 * hat_s/bar_s * np.exp(-x**2/hat_s)\n",
    "        elif L >= 3:\n",
    "            u = (bar_beta**(2-L) - (L-2) * bar_s * t)**(1/(2-L))\n",
    "            u_eps0 = (bar_beta_eps0**(2-L) - (L-2) * bar_s_eps0 * t)**(1/(2-L))\n",
    "            w = lambda x: -bar_1/bar_s * gamma_inc(2/L-1, x**L/hat_s) - srk**0.5 * hat_s/bar_s * gamma_inc(2/L, x**L/hat_s)\n",
    "        else:\n",
    "            raise ValueError(\"L should be at least 2\")\n",
    "\n",
    "        v = (1-1/L) * hat_s**(2/L-1) * u**(L-1) * (w(u) - w(beta)) * np.exp(u**L/hat_s)\n",
    "    else:\n",
    "        sqrt_srk_1 = srk**0.5 - 1\n",
    "        a = beta * (1 + eps * sqrt_srk_1 / 2)\n",
    "        b = beta * sqrt_srk_1 * (1 - 2 * eps)\n",
    "        bar_a_2_div_1 = (a + eps * b)**2 / bar_1\n",
    "        a_div_bar_a = a / (a + eps * b)\n",
    "        \n",
    "        if L == 2:\n",
    "            z = np.exp(bar_1 * t)\n",
    "            u = a * z\n",
    "            w = lambda x: special.expi(-bar_a_2_div_1 * x**2) - a_div_bar_a * np.exp(-bar_a_2_div_1 * x**2)\n",
    "            v = (a+b)/2 * (w(z) - w(1)) * z * np.exp(bar_a_2_div_1 * z**2)\n",
    "        elif L >= 3:\n",
    "            raise NotImplementedError()\n",
    "        else:\n",
    "            raise ValueError(\"L should be at least 2\")\n",
    "                    \n",
    "    deviation = v * u**(L-1)\n",
    "    deviation_eps0 = u_eps0**L\n",
    "    mean_tilde_x = av_x_white_norm + (d / (m * delta))**0.5\n",
    "    \n",
    "    res = {\n",
    "        1: (1 + (L-1) * srk**0.5 + L * mean_tilde_x) * deviation*eps, \n",
    "        2: (L-1) * ((1 + (L-1)*srk**0.5) * 2 + (L+1 + (L-1)*srk**0.5) * mean_tilde_x) * deviation*eps**2\n",
    "    }\n",
    "    res['linear'] = res[1] + (L-1) * (srk**0.5 + mean_tilde_x) * deviation_eps0*eps\n",
    "    \n",
    "    return res, u**L"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 43,
   "id": "02e7cefb-f8fc-43f0-bf6b-d692347da736",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_linear_training_time(L, s, alpha, beta):\n",
    "    if L == 2:\n",
    "        return np.log(alpha * (s-beta**2) / ((1-alpha) * beta**2)) / (2 * s)\n",
    "    elif L >= 3:\n",
    "        return (\n",
    "            beta**(2-L) * special.hyp2f1(1, 2/L-1, 2/L, beta**L/s) \\\n",
    "            - (alpha * s)**(2/L-1) * special.hyp2f1(1, 2/L-1, 2/L, alpha)\n",
    "        ) / (s * (L-2))\n",
    "    else:\n",
    "        raise ValueError(\"L should be at least 2\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "id": "1c6f91ab-ead4-47c5-a02c-68295c7308dc",
   "metadata": {},
   "outputs": [],
   "source": [
    "from scipy.special import rel_entr\n",
    "from scipy.optimize import root_scalar\n",
    "\n",
    "def inv_kl(kl, q):\n",
    "    kl_div = lambda p: rel_entr(q, p) + rel_entr(1-q, 1-p) - kl\n",
    "    res = root_scalar(f=kl_div, method='toms748', bracket=[q, 0.99])\n",
    "    return res.root"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 45,
   "id": "f2bbc6e2-678f-4628-85f5-65f0973262aa",
   "metadata": {},
   "outputs": [],
   "source": [
    "from scipy.optimize import minimize_scalar\n",
    "\n",
    "def get_test_risk_bound(train_risk, dev_term, d, m, delta, p=32, alpha=1.1, mode='langford_seeger'):\n",
    "    bounds = {}\n",
    "    for k in [1,2]:\n",
    "        proxy_train_risk_bound = np.clip(train_risk + dev_term[k], 0, 0.9)\n",
    "        if mode == 'langford_seeger':\n",
    "            kl_bound = (np.log(2 * m / delta) + k * d * p * np.log(2)) / (m - 1)\n",
    "            proxy_test_risk_bound = inv_kl(kl=kl_bound, q=proxy_train_risk_bound)\n",
    "        elif mode == 'catoni':\n",
    "            phi_inv = lambda x, gamma: (1 - np.exp(-gamma * x)) / (1 - np.exp(-gamma))\n",
    "            test_risk_f = lambda la: phi_inv(\n",
    "                proxy_train_risk_bound + \\\n",
    "                alpha / la * (k*d*p*np.log(2) - np.log(delta) + 2 * np.log(np.log(la*alpha**2) / np.log(alpha))),\n",
    "                la / m\n",
    "            )\n",
    "            minimiziation_res = minimize_scalar(fun=test_risk_f, method='bounded', bounds=[alpha,1e6])\n",
    "            proxy_test_risk_bound = minimiziation_res.fun\n",
    "            # print(minimiziation_res.x, minimiziation_res.fun)\n",
    "        else:\n",
    "            raise ValueError(\"Unrecognized mode: '{}'\".format(mode))\n",
    "        test_risk_bound = proxy_test_risk_bound + dev_term[k]\n",
    "        bounds[k] = test_risk_bound\n",
    "    return bounds"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "id": "c4cd4d87-8896-43c7-b5c3-81950a374993",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_proxy_generalization_gap_bound(d, m, delta, mode='finite_union', p=32):\n",
    "    bounds = {}\n",
    "    for k in [1,2]:\n",
    "        if mode == 'vc':\n",
    "            bounds[k] = (8 * k * d / m * (1 - np.log(k * d / m)))**0.5 + (-np.log(delta) / (2 * m))**0.5\n",
    "        elif mode == 'finite_union':\n",
    "            bounds[k] = ((np.log(1/delta) + k * d * p * np.log(2)) / (2 * m))**0.5\n",
    "        else:\n",
    "            raise ValueError(\"unsupported mode '{}'\".format(mode))\n",
    "    return bounds"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "id": "9cbb9e18-080f-4701-aab6-cd6f9aa5a6ba",
   "metadata": {},
   "outputs": [],
   "source": [
    "epsilon_list = [1e-3]\n",
    "# epsilon = [1e-2, 1e0]\n",
    "beta_list = [1e-3]\n",
    "inv_q_list = [0., 1e-3, 1e-2, 1e-1]\n",
    "# inv_q_list = [0, 1e-2]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 51,
   "id": "357ef25b-cd24-4741-923c-5c29074b2a73",
   "metadata": {
    "scrolled": true,
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "beta = 0.001, epsilon = 0.001\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_646844/3279153538.py:154: RuntimeWarning: overflow encountered in scalar power\n",
      "  (r_bound * r_X_bound * (1+cos_bound) + (1+3*cos_bound))**2 +\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "epoch 500\n",
      "train loss = 0.5000, train acc = 60.79 | 0.0; 60.62 | 0.001; 58.83 | 0.01; 41.25 | 0.1; \n",
      "test loss = 0.5000, test acc = 62.31\n",
      "(normalized) proj loss = 0.4198469595463221, train loss = 1.0\n",
      "cos = 0.64795598580947, cos bound = 0.6479544600113\n",
      "eps = 0.001, eps bound = -inf\n",
      "deviation bound = 1.48464684025288e-11; 9.96684672630811e-14; 4.37444463396103e-11; \n",
      "proxy deviation = 1.1495333493687782e-13; 1.10183799401738e-16; 4.946324925168832e-13; \n",
      "proxy generalization gap = 1.51 | 0.0; 1.49 | 0.001; 1.46 | 0.01; 2.02 | 0.1; \n",
      "1.51 | 0.0; 1.49 | 0.001; 1.47 | 0.01; 2.02 | 0.1; \n",
      "1.50 | 0.0; 1.47 | 0.001; 1.47 | 0.01; 2.05 | 0.1; \n",
      "\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_646844/3279153538.py:152: RuntimeWarning: overflow encountered in scalar multiply\n",
      "  r_bound * r_X_bound * (1+cos_bound) + (1+3*cos_bound) -\n",
      "/tmp/ipykernel_646844/3279153538.py:154: RuntimeWarning: overflow encountered in scalar multiply\n",
      "  (r_bound * r_X_bound * (1+cos_bound) + (1+3*cos_bound))**2 +\n",
      "/tmp/ipykernel_646844/3279153538.py:155: RuntimeWarning: overflow encountered in scalar multiply\n",
      "  4*cos_bound * (r_X_bound**2 - r_bound * r_X_bound * (1+cos_bound) - 2*cos_bound)\n",
      "/tmp/ipykernel_646844/3279153538.py:154: RuntimeWarning: invalid value encountered in scalar add\n",
      "  (r_bound * r_X_bound * (1+cos_bound) + (1+3*cos_bound))**2 +\n",
      "/tmp/ipykernel_646844/3279153538.py:157: RuntimeWarning: overflow encountered in scalar multiply\n",
      "  ) / (2 * (2*cos_bound + r_bound * r_X_bound * (1+cos_bound) - r_X_bound**2))\n",
      "/tmp/ipykernel_646844/3279153538.py:155: RuntimeWarning: overflow encountered in scalar power\n",
      "  4*cos_bound * (r_X_bound**2 - r_bound * r_X_bound * (1+cos_bound) - 2*cos_bound)\n",
      "/tmp/ipykernel_646844/3279153538.py:155: RuntimeWarning: invalid value encountered in scalar subtract\n",
      "  4*cos_bound * (r_X_bound**2 - r_bound * r_X_bound * (1+cos_bound) - 2*cos_bound)\n",
      "/tmp/ipykernel_646844/3279153538.py:157: RuntimeWarning: overflow encountered in scalar power\n",
      "  ) / (2 * (2*cos_bound + r_bound * r_X_bound * (1+cos_bound) - r_X_bound**2))\n",
      "/tmp/ipykernel_646844/3279153538.py:157: RuntimeWarning: invalid value encountered in scalar subtract\n",
      "  ) / (2 * (2*cos_bound + r_bound * r_X_bound * (1+cos_bound) - r_X_bound**2))\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "epoch 1000\n",
      "train loss = 0.5000, train acc = 66.42 | 0.0; 66.29 | 0.001; 64.92 | 0.01; 49.72 | 0.1; \n",
      "test loss = 0.5000, test acc = 68.02\n",
      "(normalized) proj loss = 0.4198469595463221, train loss = 1.0\n",
      "cos = 0.64795598580947, cos bound = 0.6479544594322261\n",
      "eps = 0.001, eps bound = nan\n",
      "deviation bound = 4.5163055546528e-11; 3.03192139786802e-13; 8.58296366636249e-11; \n",
      "proxy deviation = 2.554832430681775e-13; 1.2192127391818375e-16; 6.874971512221828e-13; \n",
      "proxy generalization gap = 1.58 | 0.0; 1.61 | 0.001; 1.67 | 0.01; 1.99 | 0.1; \n",
      "1.58 | 0.0; 1.60 | 0.001; 1.67 | 0.01; 1.97 | 0.1; \n",
      "1.59 | 0.0; 1.59 | 0.001; 1.69 | 0.01; 2.01 | 0.1; \n",
      "\n",
      "\n",
      "epoch 1500\n",
      "train loss = 0.5000, train acc = 70.40 | 0.0; 70.30 | 0.001; 69.20 | 0.01; 57.25 | 0.1; \n",
      "test loss = 0.5000, test acc = 72.14\n",
      "(normalized) proj loss = 0.4198469595463221, train loss = 1.0\n",
      "cos = 0.64795598580947, cos bound = 0.6479544584843071\n",
      "eps = 0.001, eps bound = nan\n",
      "deviation bound = 1.1251070461127e-10; 7.55315619530404e-13; 1.72310770900017e-10; \n",
      "proxy deviation = 4.50659583435542e-13; 1.4212670224445325e-16; 9.850067810013297e-13; \n",
      "proxy generalization gap = 1.75 | 0.0; 1.75 | 0.001; 1.76 | 0.01; 1.63 | 0.1; \n",
      "1.73 | 0.0; 1.74 | 0.001; 1.75 | 0.01; 1.59 | 0.1; \n",
      "1.73 | 0.0; 1.75 | 0.001; 1.75 | 0.01; 1.62 | 0.1; \n",
      "\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_646844/3279153538.py:149: RuntimeWarning: overflow encountered in exp\n",
      "  r_bound = 3 + epsilon / bar_s * (np.exp(2 * bar_s * epoch * lr) - 1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "epoch 2000\n",
      "train loss = 0.5000, train acc = 73.29 | 0.0; 73.17 | 0.001; 72.35 | 0.01; 63.10 | 0.1; \n",
      "test loss = 0.5000, test acc = 75.03\n",
      "(normalized) proj loss = 0.4198469595463221, train loss = 1.0\n",
      "cos = 0.64795598580947, cos bound = 0.6479544568202037\n",
      "eps = 0.001, eps bound = nan\n",
      "deviation bound = 2.71397063037916e-10; 1.8219638879293e-12; 3.64488536447957e-10; \n",
      "proxy deviation = 7.43857774855583e-13; 1.7110209794835696e-16; 1.423493592965508e-12; \n",
      "proxy generalization gap = 1.72 | 0.0; 1.78 | 0.001; 1.81 | 0.01; 2.15 | 0.1; \n",
      "1.73 | 0.0; 1.80 | 0.001; 1.81 | 0.01; 2.12 | 0.1; \n",
      "1.72 | 0.0; 1.77 | 0.001; 1.81 | 0.01; 2.12 | 0.1; \n",
      "\n",
      "\n",
      "epoch 2500\n",
      "train loss = 0.5000, train acc = 75.32 | 0.0; 75.25 | 0.001; 74.63 | 0.01; 67.77 | 0.1; \n",
      "test loss = 0.5000, test acc = 77.04\n",
      "(normalized) proj loss = 0.4198469595463221, train loss = 1.0\n",
      "cos = 0.64795598580947, cos bound = 0.6479544536129247\n",
      "eps = 0.001, eps bound = nan\n",
      "deviation bound = 6.84499601999398e-10; 4.59523600655417e-12; 8.40981441462056e-10; \n",
      "proxy deviation = 1.20333398193756e-12; 2.2193584060064794e-16; 2.078937716420981e-12; \n",
      "proxy generalization gap = 1.70 | 0.0; 1.69 | 0.001; 1.60 | 0.01; 1.78 | 0.1; \n",
      "1.72 | 0.0; 1.70 | 0.001; 1.61 | 0.01; 1.79 | 0.1; \n",
      "1.69 | 0.0; 1.70 | 0.001; 1.62 | 0.01; 1.79 | 0.1; \n",
      "\n",
      "\n",
      "epoch 3000\n",
      "train loss = 0.5000, train acc = 76.79 | 0.0; 76.74 | 0.001; 76.30 | 0.01; 71.23 | 0.1; \n",
      "test loss = 0.5000, test acc = 78.39\n",
      "(normalized) proj loss = 0.4198469595463221, train loss = 1.0\n",
      "cos = 0.64795598580947, cos bound = 0.6479544465648751\n",
      "eps = 0.001, eps bound = nan\n",
      "deviation bound = 1.9323666894126e-9; 1.29725144662137e-11; 2.22573540854679e-9; \n",
      "proxy deviation = 1.946391398749414e-12; 3.0142710337360584e-16; 3.0877574802579133e-12; \n",
      "proxy generalization gap = 1.60 | 0.0; 1.62 | 0.001; 1.49 | 0.01; 1.79 | 0.1; \n",
      "1.59 | 0.0; 1.61 | 0.001; 1.49 | 0.01; 1.81 | 0.1; \n",
      "1.61 | 0.0; 1.62 | 0.001; 1.51 | 0.01; 1.79 | 0.1; \n",
      "\n",
      "\n",
      "epoch 3500\n",
      "train loss = 0.5000, train acc = 77.98 | 0.0; 77.94 | 0.001; 77.58 | 0.01; 73.84 | 0.1; \n",
      "test loss = 0.5000, test acc = 79.19\n",
      "(normalized) proj loss = 0.4198469595463221, train loss = 1.0\n",
      "cos = 0.64795598580947, cos bound = 0.6479544276733181\n",
      "eps = 0.001, eps bound = nan\n",
      "deviation bound = 6.72115750629236e-9; 4.51209977163196e-11; 7.37147462873915e-9; \n",
      "proxy deviation = 3.1861744145822435e-12; 4.2875595474106925e-16; 4.698552311110937e-12; \n",
      "proxy generalization gap = 1.21 | 0.0; 1.15 | 0.001; 1.27 | 0.01; 1.59 | 0.1; \n",
      "1.21 | 0.0; 1.15 | 0.001; 1.26 | 0.01; 1.60 | 0.1; \n",
      "1.21 | 0.0; 1.18 | 0.001; 1.27 | 0.01; 1.61 | 0.1; \n",
      "\n",
      "\n",
      "epoch 4000\n",
      "train loss = 0.5000, train acc = 78.85 | 0.0; 78.83 | 0.001; 78.59 | 0.01; 75.91 | 0.1; \n",
      "test loss = 0.5000, test acc = 79.80\n",
      "(normalized) proj loss = 0.4198469595463221, train loss = 1.0\n",
      "cos = 0.64795598580947, cos bound = 0.6479543565767681\n",
      "eps = 0.001, eps bound = nan\n",
      "deviation bound = 3.47742921142077e-8; 2.334494844977e-10; 3.67021280372377e-8; \n",
      "proxy deviation = 5.330298268740874e-12; 6.311186374059036e-16; 7.380363681308566e-12; \n",
      "proxy generalization gap = 0.96 | 0.0; 0.97 | 0.001; 0.92 | 0.01; 1.32 | 0.1; \n",
      "0.95 | 0.0; 0.96 | 0.001; 0.90 | 0.01; 1.33 | 0.1; \n",
      "0.98 | 0.0; 0.97 | 0.001; 0.91 | 0.01; 1.32 | 0.1; \n",
      "\n",
      "\n",
      "epoch 4500\n",
      "train loss = 0.5000, train acc = 79.60 | 0.0; 79.58 | 0.001; 79.43 | 0.01; 77.60 | 0.1; \n",
      "test loss = 0.5000, test acc = 80.57\n",
      "(normalized) proj loss = 0.4198469595463221, train loss = 1.0\n",
      "cos = 0.64795598580947, cos bound = 0.6479537984327912\n",
      "eps = 0.001, eps bound = nan\n",
      "deviation bound = 4.56949108128971e-7; 3.06762631958245e-9; 4.67725159681103e-7; \n",
      "proxy deviation = 9.212721731322038e-12; 9.676940611400963e-16; 1.2075760530416701e-11; \n",
      "proxy generalization gap = 0.97 | 0.0; 0.97 | 0.001; 0.96 | 0.01; 0.93 | 0.1; \n",
      "0.97 | 0.0; 0.97 | 0.001; 0.96 | 0.01; 0.94 | 0.1; \n",
      "0.97 | 0.0; 0.97 | 0.001; 0.96 | 0.01; 0.92 | 0.1; \n",
      "\n",
      "\n",
      "epoch 5000\n",
      "train loss = 0.5000, train acc = 80.26 | 0.0; 80.24 | 0.001; 80.12 | 0.01; 78.87 | 0.1; \n",
      "test loss = 0.5000, test acc = 81.37\n",
      "(normalized) proj loss = 0.4198469595463221, train loss = 1.0\n",
      "cos = 0.64795598580947, cos bound = 0.6477609004354823\n",
      "eps = 0.001, eps bound = nan\n",
      "deviation bound = 0.000983257690906963; 6.60088206301245e-6; 0.000984200214497877; \n",
      "proxy deviation = 1.666067457606335e-11; 1.5626584736209894e-15; 2.0816823959046715e-11; \n",
      "proxy generalization gap = 1.10 | 0.0; 1.08 | 0.001; 1.08 | 0.01; 0.90 | 0.1; \n",
      "1.11 | 0.0; 1.10 | 0.001; 1.06 | 0.01; 0.92 | 0.1; \n",
      "1.10 | 0.0; 1.08 | 0.001; 1.08 | 0.01; 0.90 | 0.1; \n",
      "\n",
      "\n"
     ]
    }
   ],
   "source": [
    "from copy import deepcopy\n",
    "\n",
    "num_epochs = 5000\n",
    "print_every = 500\n",
    "width = 64\n",
    "num_hidden = 2\n",
    "kappas = [1,2, 'linear']\n",
    "delta = 1e-2\n",
    "loss_proj_decay = True\n",
    "rank1_input = False\n",
    "\n",
    "train_accs = {}\n",
    "test_accs = {}\n",
    "dev_bounds = {}\n",
    "dev_proxies = {}\n",
    "gen_gap_proxies = {}\n",
    "proj_losses = {}\n",
    "train_losses = {}\n",
    "for k in kappas:\n",
    "    dev_bounds[k] = {}\n",
    "    dev_proxies[k] = {}\n",
    "    gen_gap_proxies[k] = {}\n",
    "cos_bounds = {}\n",
    "eps_bounds = {}\n",
    "\n",
    "for beta in beta_list:\n",
    "    train_accs[beta] = {}\n",
    "    test_accs[beta] = {}\n",
    "    proj_losses[beta] = {}\n",
    "    train_losses[beta] = {}\n",
    "    for k in kappas:\n",
    "        dev_bounds[k][beta] = {}\n",
    "        dev_proxies[k][beta] = {}\n",
    "        gen_gap_proxies[k][beta] = {}\n",
    "    cos_bounds[beta] = {}\n",
    "    eps_bounds[beta] = {}\n",
    "    \n",
    "    lr = 3e-1\n",
    "    # print(lr)\n",
    "    \n",
    "    for epsilon in epsilon_list:\n",
    "        print('beta = {}, epsilon = {}'.format(beta,epsilon))\n",
    "        train_accs[beta][epsilon] = {}\n",
    "        for inv_q in inv_q_list:\n",
    "            train_accs[beta][epsilon][inv_q] = []\n",
    "        test_accs[beta][epsilon] = []\n",
    "        proj_losses[beta][epsilon] = []\n",
    "        train_losses[beta][epsilon] = []\n",
    "        for k in kappas:\n",
    "            dev_bounds[k][beta][epsilon] = []\n",
    "            dev_proxies[k][beta][epsilon] = []\n",
    "            gen_gap_proxies[k][beta][epsilon] = {}\n",
    "            for inv_q in inv_q_list:\n",
    "                gen_gap_proxies[k][beta][epsilon][inv_q] = []\n",
    "        cos_bounds[beta][epsilon] = []\n",
    "        eps_bounds[beta][epsilon] = []\n",
    "    \n",
    "        torch.manual_seed(0)\n",
    "        torch.cuda.manual_seed(0)\n",
    "\n",
    "        model = FCNet(input_shape, 1, width, num_hidden)\n",
    "        for layer in model.linear_layers:\n",
    "            param_norm = torch.linalg.norm(layer.weight.data, ord=2)\n",
    "            layer.weight.data = layer.weight.data / param_norm * beta\n",
    "\n",
    "        U, D, Vt = torch.linalg.svd(model.input_layer.weight.data, full_matrices=False)\n",
    "        if rank1_input:\n",
    "            srk = 1.\n",
    "            D = torch.zeros_like(D)\n",
    "            D[0] = beta\n",
    "        else:\n",
    "            srk = (torch.sum(D**2) / D[0]**2).item()\n",
    "            D = D * beta / D[0]\n",
    "        model.input_layer.weight.data = U @ torch.diag(D) @ Vt\n",
    "        \n",
    "        model_lin = deepcopy(model)\n",
    "        \n",
    "        model = model.to(device)\n",
    "        model_lin = model_lin.to(device)\n",
    "\n",
    "        optimizer = optim.SGD(model.parameters(), lr=lr)\n",
    "        optimizer_lin = optim.SGD(model_lin.parameters(), lr=lr)\n",
    "\n",
    "        for epoch in range(num_epochs):\n",
    "            if (epoch+1) % print_every == 0:\n",
    "                print('epoch {}'.format(epoch+1))\n",
    "\n",
    "            train_loss, train_acc, proj_loss = get_loss_acc(\n",
    "                model, epsilon=epsilon, loader=[(X_train_white, y_train)], \n",
    "                optimizer=optimizer, inv_q=inv_q_list, beta_L=beta**(num_hidden+1)\n",
    "            )\n",
    "            test_loss, test_acc, _ = get_loss_acc(\n",
    "                model, epsilon=epsilon, loader=[(X_test_white, y_test)]\n",
    "            )\n",
    "            \n",
    "            _, _, _ = get_loss_acc(\n",
    "                model_lin, epsilon=0, loader=[(X_train_white, y_train)], \n",
    "                optimizer=optimizer_lin\n",
    "            )\n",
    "            \n",
    "            gen_gap_proxy = {}\n",
    "            dev_proxy = {}\n",
    "            with torch.no_grad():\n",
    "                for k in kappas:\n",
    "                    model_logits_train = model(X_train_white, epsilon=epsilon).view(-1)\n",
    "                    model_logits_test = model(X_test_white, epsilon=epsilon).view(-1)\n",
    "\n",
    "                    if k == 1:\n",
    "                        model_proxy = lambda x, epsilon: model_lin(x, epsilon=epsilon)\n",
    "                    elif k == 2:\n",
    "                        model_lin_logits_train = model_lin(X_train_white, epsilon=epsilon).view(-1)\n",
    "                        # print(X_train_white_plus.device, model_logits_train.device, model_lin_logits_train.device)\n",
    "                        correction = (X_train_white_plus @ (model_logits_train - model_lin_logits_train)).view(-1)\n",
    "                        model_proxy = lambda x, epsilon: model_lin(x, epsilon=epsilon).view(-1) + x @ correction\n",
    "                    elif k == 'linear':\n",
    "                        model_proxy = lambda x, epsilon: model_lin(x, epsilon=0)\n",
    "\n",
    "                    model_proxy_logits_train = model_proxy(X_train_white, epsilon).view(-1)\n",
    "                    model_proxy_logits_test = model_proxy(X_test_white, epsilon).view(-1)\n",
    "\n",
    "                    dev_proxy[k] = torch.mean(\n",
    "                        torch.abs(model_logits_train - model_proxy_logits_train)\n",
    "                    ) + torch.mean(\n",
    "                        torch.abs(model_logits_test - model_proxy_logits_test)\n",
    "                    )\n",
    "\n",
    "                    _, train_acc_proxy, _ = get_loss_acc(\n",
    "                        model_proxy, epsilon=epsilon, loader=[(X_train_white, y_train)], \n",
    "                        inv_q=inv_q_list, beta_L=beta**(num_hidden+1), soft_risk=True\n",
    "                    )\n",
    "                    _, test_acc_proxy, _ = get_loss_acc(\n",
    "                        model_proxy, epsilon=epsilon, loader=[(X_test_white, y_test)], \n",
    "                        inv_q=inv_q_list, beta_L=beta**(num_hidden+1), soft_risk=True\n",
    "                    )\n",
    "                    gen_gap_proxy[k] = {}\n",
    "                    for inv_q in inv_q_list:\n",
    "                        gen_gap_proxy[k][inv_q] = np.abs(train_acc_proxy[inv_q] - test_acc_proxy[inv_q])\n",
    "            \n",
    "            \n",
    "            dev_bound, lip_factor = get_deviation_bound(\n",
    "                t=epoch*lr, L=num_hidden+1, s=s, beta=beta, eps=epsilon, srk=srk,\n",
    "                delta=delta, m=dataset_size, d=im_size**2, loss_proj_decay=loss_proj_decay\n",
    "            )\n",
    "            \n",
    "            cos_bound = (s - lip_factor) / (1 + beta**(num_hidden+1) * srk**0.5)\n",
    "            cos = (proj_loss / (dataset_size * train_loss))**0.5\n",
    "            \n",
    "            bar_s = get_bar_s(beta=beta, eps=epsilon, L=num_hidden+1, s=s, rho=srk)\n",
    "            r_bound = 3 + epsilon / bar_s * (np.exp(2 * bar_s * epoch * lr) - 1)\n",
    "            r_X_bound = r_bound\n",
    "            eps_bound = (\n",
    "                r_bound * r_X_bound * (1+cos_bound) + (1+3*cos_bound) - \n",
    "                (\n",
    "                    (r_bound * r_X_bound * (1+cos_bound) + (1+3*cos_bound))**2 + \n",
    "                    4*cos_bound * (r_X_bound**2 - r_bound * r_X_bound * (1+cos_bound) - 2*cos_bound)\n",
    "                )**0.5\n",
    "            ) / (2 * (2*cos_bound + r_bound * r_X_bound * (1+cos_bound) - r_X_bound**2))\n",
    "            if cos_bound < 0:\n",
    "                eps_bound = 0.\n",
    "            \n",
    "            for inv_q in inv_q_list:\n",
    "                train_accs[beta][epsilon][inv_q].append(train_acc[inv_q])\n",
    "            test_accs[beta][epsilon].append(test_acc)\n",
    "            proj_losses[beta][epsilon].append(proj_loss)\n",
    "            train_losses[beta][epsilon].append(train_loss)\n",
    "            for k in kappas:\n",
    "                dev_bounds[k][beta][epsilon].append(dev_bound[k])\n",
    "                dev_proxies[k][beta][epsilon].append(dev_proxy[k].item())\n",
    "                for inv_q in inv_q_list:\n",
    "                    gen_gap_proxies[k][beta][epsilon][inv_q].append(gen_gap_proxy[k][inv_q])\n",
    "            cos_bounds[beta][epsilon].append(cos_bound)\n",
    "            eps_bounds[beta][epsilon].append(eps_bound)\n",
    "                \n",
    "            if (epoch+1) % print_every == 0:\n",
    "                print('train loss = {:.4f}, train acc = '.format(train_loss), end='')\n",
    "                for inv_q in inv_q_list:\n",
    "                    print('{:.2f} | {}; '.format(train_acc[inv_q]*100, inv_q), end='')\n",
    "                print()\n",
    "                print('test loss = {:.4f}, test acc = {:.2f}'.format(test_loss, test_acc * 100))\n",
    "                print('(normalized) proj loss = {}, train loss = {}'.format(2*proj_loss / dataset_size, 2*train_loss))\n",
    "                print('cos = {}, cos bound = {}'.format(cos, cos_bound))\n",
    "                print('eps = {}, eps bound = {}'.format(epsilon, eps_bound))\n",
    "                print('deviation bound = ', end='')\n",
    "                for k in kappas:\n",
    "                    print('{}; '.format(dev_bound[k]), end='')\n",
    "                print()\n",
    "                print('proxy deviation = ', end='')\n",
    "                for k in kappas:\n",
    "                    print('{}; '.format(dev_proxy[k]), end='')\n",
    "                print()\n",
    "                print('proxy generalization gap = ', end='')\n",
    "                for k in kappas:\n",
    "                    for inv_q in inv_q_list:\n",
    "                        print('{:.2f} | {}; '.format(gen_gap_proxy[k][inv_q]*100, inv_q), end='')\n",
    "                    print()\n",
    "                print('\\n')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 52,
   "id": "0eb14d3a-3c60-431c-a47a-ec8a2b51131c",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "image/png": "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TJjA+++/Xxe7XK0WLlzI//73P4qKiioUzjqdjrvvvpt9+/YxduxYunbtyunTp1myZAldu3Zl4cKFWFn980uo1NRUJk6cSH5+PnfddReBgYFs2rSJv//+m8cee4xHH320DvbwxqWlpTFjxgyys7MZP3484eHhFBcXc/HiRby9vXnooYcAyQmMBd/kyZM5efIk48aNo3379mg0GjZu3Mj+/fsZM2YMc+bMARpXXpGRkbi6utKyZUtiY2PJysq6ZoFTk+en06dPM3XqVGxtbbnrrrvw8PBg1apVHDx4kPfff58JEybUWA5KVCWrSZMmkZyczNChQ4mMjKSsrIxNmzaxe/duWrduzZIlS7CzszO1b8xZ/dtDDz3Enj17KCoqqrRwrurnbfv27Tz44IP4+vpyxx13YGtryy+//EJsbCxff/01vXr1qrb9rrcMwuLk5OQYOnXqZOjXr58hPz/ftDwpKcnQoUMHw/Tp0+uwdzXr6NGjhtzc3ArLly5daoiIiDDMnj3btGzXrl2GiIgIw4MPPmjWdv369YaIiAjDZ599Zrb8008/NURERBgWLFhgtvytt94yREREGPbt21eNe1L74uPjDe3btzd89913hoiICMNdd91ltn7ZsmWGiIgIw9tvv222/NtvvzVEREQYVqxYYbb82WefNURERBg2bNhgtnzWrFmGVq1aGeLj42tiN2rMjBkzDL179zYkJydfs11jz8lgMBh27txpiIiIMHzwwQdmy7VarWHs2LGGqKgo07mpMeUVFxdn+vedd95paNmy5VXb1vT5adq0aYbIyEjD0aNHTcs0Go1h3Lhxhi5dupj931EXqpLVnj17DGVlZRWWP/XUU4aIiAjDokWLzJY35qyutHr1akPLli0NCxYsMERERBi++OKLCm2q8nnTarWGgQMHGjp06GBISkoyLc/LyzP07dvXMHToUINOp6vq7jU4MsbZAm3atImCggJuv/12nJ2dTcubNm3K8OHD2bt3LykpKXXYw5rTtm1bXF1dKywfNWoUAGfOnDEtW7lyJQAzZ840azt8+HACAgJM669s7+DgwNSpU82Wl7/+999/v+n+16VXXnmF8PBwpk+fXun6q+U1bdo07O3tzfa/uLiYDRs2EBgYyLBhw8zaz5w5E61Wyx9//FG9O1CDoqOj2bNnD/fddx/+/v7odDoKCwsrbduYcyqX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",
      "text/plain": [
       "<Figure size 800x500 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "cmap = plt.get_cmap('tab10')\n",
    "linestyles = ['solid', 'dashed']\n",
    "p = 16\n",
    "\n",
    "proxy_bound = get_proxy_generalization_gap_bound(d=im_size**2, m=dataset_size, delta=delta, p=p)\n",
    "\n",
    "for k in [2]:\n",
    "    for epsilon in epsilon_list:\n",
    "        for beta in beta_list:\n",
    "            t = get_linear_training_time(L=num_hidden+1, s=s, alpha=0.9, beta=beta)\n",
    "            xs = np.arange(num_epochs) * lr\n",
    "\n",
    "            train_risks = {}\n",
    "            for j, inv_q in enumerate(inv_q_list):\n",
    "                train_risks[inv_q] = 1. - np.array(train_accs[beta][epsilon][inv_q])\n",
    "                if j == 0:\n",
    "                    plt.plot(\n",
    "                        xs, train_risks[inv_q], label='train risk, $\\hat{R}$', \n",
    "                        color='black', linestyle='dashed'\n",
    "                    )\n",
    "\n",
    "            test_risks = 1. - np.array(test_accs[beta][epsilon])\n",
    "            plt.plot(xs, test_risks, label=r'test risk, $R$', color='black', linestyle='solid')\n",
    "\n",
    "            # gamma_to_search = beta**2 * 0.1\n",
    "            # gamma_res = 0\n",
    "            # diff = 1e8\n",
    "            # for gamma in gamma_list:\n",
    "            #     new_diff = np.abs(gamma_to_search / gamma - 1)\n",
    "            #     if new_diff < diff:\n",
    "            #         diff = new_diff\n",
    "            #         gamma_res = gamma\n",
    "            # gamma = gamma_res\n",
    "\n",
    "            dev_terms = {}\n",
    "            for j, inv_q in enumerate(inv_q_list[1:]):\n",
    "                dev_terms[k] = np.array(dev_bounds[k][beta][epsilon]) / (inv_q * beta**(num_hidden+1))\n",
    "                # plt.plot(xs, dev_terms[k], label='deviation term', linestyle=linestyles[i], color=cmap(3))\n",
    "                # plt.hlines(\n",
    "                #     proxy_bound[k], xs[0], xs[-1], label='bound for proxy', linestyle=linestyles[i], color=cmap(0)\n",
    "                # )\n",
    "                \n",
    "                # plt.plot(\n",
    "                #     xs, train_risks[inv_q] + proxy_bound[k], \n",
    "                #     label=r'$\\hat{R}_\\gamma + \\Upsilon_\\kappa$' if j==0 else None, color=cmap(0)\n",
    "                # )\n",
    "\n",
    "                # plt.plot(\n",
    "                #     xs, train_risks[inv_q] + dev_terms[k], \n",
    "                #     label=r'$\\hat{R}_\\gamma + \\frac{\\Delta_{\\kappa,\\beta} \\epsilon^\\kappa}{\\gamma}$' if j==0 else None, \n",
    "                #     color=cmap(1)\n",
    "                # )\n",
    "\n",
    "                full_bounds = train_risks[inv_q] + dev_terms[k] + proxy_bound[k]\n",
    "                plt.plot(\n",
    "                    xs, full_bounds, \n",
    "                    label=r'full bound, $\\lg\\gamma \\in \\lbrace{:.0f},{:.0f},{:.0f}\\rbrace$'.format(\n",
    "                        np.log10(inv_q_list[1]) + (num_hidden+1) * np.log10(beta),\n",
    "                        np.log10(inv_q_list[2]) + (num_hidden+1) * np.log10(beta),\n",
    "                        np.log10(inv_q_list[3]) + (num_hidden+1) * np.log10(beta)\n",
    "                    ) if j==0 else None, color=cmap(2)\n",
    "                )\n",
    "\n",
    "                # mode = 'catoni'\n",
    "                # full_bounds = []\n",
    "                # for epoch in range(num_epochs):\n",
    "                #     full_bounds.append(\n",
    "                #         get_test_risk_bound(\n",
    "                #             train_risk=train_risks[gamma][epoch], \n",
    "                #             dev_term={1: dev_terms[1][epoch]/2, 2: dev_terms[2][epoch]/2},\n",
    "                #             d=im_size**2, m=dataset_size, delta=delta, p=p, mode=mode\n",
    "                #         )[k]\n",
    "                #     )\n",
    "                # plt.plot(\n",
    "                #     xs, full_bounds, label='full bound, {}'.format(mode) if k == 1 else None,\n",
    "                #     linestyle=linestyles[i], color=cmap(2)\n",
    "                # )\n",
    "\n",
    "            # plt.vlines(t, 0, 10, color='black', linestyle='dotted')\n",
    "\n",
    "            plt.hlines(1 / num_classes, xs[0], xs[-1], label='random guess risk', color='black', linestyle='dotted')\n",
    "\n",
    "            plt.legend()\n",
    "            plt.title(\n",
    "                r'{}x{} MNIST; $\\beta$ = {}, $\\epsilon$ = {}, $\\rho$ = {:.2f}, $\\kappa$ = {}, $p$ = {}, $L$ = {}'.format(\n",
    "                    im_size, im_size, beta, epsilon, srk**0.5, k, p, num_hidden+1\n",
    "                )\n",
    "            )\n",
    "            # plt.yscale('log')\n",
    "            # plt.xscale('log')\n",
    "            plt.xlabel(r'training time, $t$')\n",
    "            plt.ylabel('risk = 1 - accuracy')\n",
    "            plt.ylim((0.2,1.2))\n",
    "            plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "cd322608-e14b-4d56-ad24-e77def1e35e9",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
