{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Papers that write about linear regression under differential privacy:\n",
    "\n",
    "\"Old Techniques in Differentially Private Linear Regression\"\n",
    "https://proceedings.mlr.press/v98/sheffet19a/sheffet19a.pdf\n",
    "\n",
    "\"Revisiting differentially private linear regression\"\n",
    "https://auai.org/uai2018/proceedings/papers/40.pdf \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "TORCH: Using device: cuda:0\n",
      "A shape: torch.Size([60000, 784])\n",
      "A type: <class 'torch.Tensor'>\n",
      "Number of sign patterns generated: 100\n"
     ]
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import random\n",
    "import sys\n",
    "import torch\n",
    "import torchvision.transforms as transforms\n",
    "import torchvision.datasets as datasets\n",
    "import tqdm\n",
    "\n",
    "import Conv_relus_MNIST as cvxnn\n",
    "import dp_linear\n",
    "\n",
    "seed = 123\n",
    "random.seed(seed)\n",
    "np.random.seed(seed)\n",
    "torch.manual_seed(seed)\n",
    "torch.cuda.manual_seed(seed)\n",
    "\n",
    "import warnings\n",
    "warnings.filterwarnings(\"ignore\", category=UserWarning)\n",
    "warnings.filterwarnings(\"ignore\", category=FutureWarning)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_neurons = num_rff = 100  # Number of neurons, or number of random fourier features - 64 for CIFAR, 100 for MNIST and FMNIST\n",
    "beta = 0.01  # coefficient for the regularization term, lower is fewer regularization, 0 is no regularization\n",
    "do_extend = False  #\n",
    "\n",
    "eps_target = 8.91\n",
    "eps_target = 2.88\n",
    "eps_target = 1.33\n",
    "eps_target = 4.76"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([60000, 28, 28])\n",
      "torch.Size([10000, 28, 28])\n",
      "A shape: torch.Size([60000, 784])\n",
      "A type: <class 'torch.Tensor'>\n",
      "Number of sign patterns generated: 100\n"
     ]
    }
   ],
   "source": [
    "directory = '../datasets/'\n",
    "device = \"cpu\"\n",
    "\n",
    "beta = torch.tensor(beta).to(device)\n",
    "\n",
    "transform_normalize = transforms.Lambda(lambda x: (x / 255.0) * 2 - 1)\n",
    "# transform_normalize = lambda x: x\n",
    "\n",
    "def transform_normalize_cifar(X):\n",
    "    renormalize = transforms.Normalize(mean=[0.507, 0.487, 0.441], std=[0.267, 0.256, 0.276])\n",
    "    X = X/255.0\n",
    "    X = X.transpose(1, 3)\n",
    "    X = renormalize(X)\n",
    "    X = X.transpose(1, 3)\n",
    "    return X\n",
    "\n",
    "\n",
    "dataset_name = \"FMNIST\"\n",
    "if dataset_name == \"MNIST\":\n",
    "    train_dataset = datasets.MNIST(directory, train=True, download=True,transform=transforms.Compose([transforms.ToTensor()]))\n",
    "    test_dataset =  datasets.MNIST(directory, train=False, transform=transforms.Compose([transforms.ToTensor()]))\n",
    "elif dataset_name == \"CIFAR10\":\n",
    "    train_dataset = datasets.CIFAR10(directory, train=True, download=True,transform=transforms.Compose([transforms.ToTensor()]))\n",
    "    test_dataset =  datasets.CIFAR10(directory, train=False, transform=transforms.Compose([transforms.ToTensor()]))\n",
    "elif dataset_name == \"FMNIST\":\n",
    "    train_dataset = datasets.FashionMNIST(directory, train=True, download=True,transform=transforms.Compose([transforms.ToTensor()]))\n",
    "    test_dataset =  datasets.FashionMNIST(directory, train=False, transform=transforms.Compose([transforms.ToTensor()]))\n",
    "else:\n",
    "    raise ValueError(f\"Dataset {dataset_name} not supported\")\n",
    "\n",
    "RAW_train = transform_normalize(torch.tensor(train_dataset.data).to(torch.float))\n",
    "RAW_test = transform_normalize(torch.tensor(test_dataset.data).to(torch.float))\n",
    "print(RAW_train.shape)\n",
    "print(RAW_test.shape)\n",
    "# flatten along the 28x28 gray scale images\n",
    "RAW_train = RAW_train.reshape(RAW_train.shape[0], -1)\n",
    "RAW_test = RAW_test.reshape(RAW_test.shape[0], -1)\n",
    "\n",
    "y_train = torch.tensor(train_dataset.targets).to(torch.int64)\n",
    "y_test = torch.tensor(test_dataset.targets).to(torch.int64)\n",
    "\n",
    "y_hot_train = torch.zeros(y_train.shape[0], 10)\n",
    "y_hot_train.scatter_(1, y_train.unsqueeze(1), 1)\n",
    "y_hot_train = y_hot_train.to(device)\n",
    "\n",
    "num_train_samples, num_features = RAW_train.size()\n",
    "print('A shape: ' + str(RAW_train.shape))\n",
    "print('A type: ' + str(type(RAW_train)))\n",
    "\n",
    "X_relu_train, X_relu_test = dp_linear.generate_relu_features(RAW_train, RAW_test, num_neurons, do_extend=do_extend)\n",
    "X_relu_train, X_relu_test = X_relu_train.to(device), X_relu_test.to(device)\n",
    "\n",
    "# Generate RFF features for training data\n",
    "gamma = 1.0 / (num_features * RAW_train.std()) # Heuristic for choosing gamma\n",
    "\n",
    "X_rff, W, b = dp_linear.generate_rff(RAW_train, n_components=num_rff, gamma=gamma, do_extend=do_extend)\n",
    "X_rff = torch.hstack([X_rff, torch.ones((X_rff.shape[0], 1))]).to(device)\n",
    "\n",
    "X_rff_test = dp_linear.generate_rff_test(RAW_test, W, b, num_rff, do_extend=do_extend)\n",
    "X_rff_test = torch.hstack([X_rff_test, torch.ones((X_rff_test.shape[0], 1))]).to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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0XXXXafCwkLZbDatWLHitOusXr1aF154oZxOp0aMGKFly5a1axPt+7tL5jz3wAMPmJ///OdmwYIFxu12d2udRx991LjdbrNixQqzZcsW86UvfckMHTrUNDQ0hNpce+21ZuLEieb99983//jHP8yIESPMLbfc0kej6Fq0fWltbTUVFRURt4ceesikp6ebmpqaUDtJZunSpRHtwl+DWOnJaz1z5kxz5513RvTd4/GE7m9tbTXjxo0zs2fPNps2bTKvv/66ycnJMQsXLuzr4XQo2jFu3brVfPnLXzZ//vOfTUlJiVm1apUZOXKk+ed//ueIdlZuw+XLl5ukpCSzZMkSs337dnPnnXeazMxMU1VV1WH79957zzgcDvNf//VfZseOHeZHP/qRSUxMNFu3bg216c57M1aiHd9XvvIVs3jxYrNp0yazc+dOc+uttxq3220OHToUajN//nxz7bXXRmyvEydOxGpI7UQ7xqVLlxqXyxXR/8rKyog2Z9M2NCb6MR4/fjxifNu2bTMOh8MsXbo01OZs246vv/66+c///E/zyiuvGEnm1Vdf7bL9vn37TGpqqlmwYIHZsWOH+eUvf2kcDod58803Q22ifd1O57wPI0FLly7tVhjx+/0mPz/fPP7446Fl1dXVxul0mj/84Q/GGGN27NhhJJmPPvoo1OaNN94wNpvNHD58uNf73pXe6sukSZPMv/3bv0Us684/2r7W0/HNnDnT3HPPPZ3e//rrrxu73R7xQfn0008bl8tlmpqaeqXv3dVb2/DFF180SUlJpqWlJbTMym04ZcoUc9ddd4X+9vl8prCw0CxatKjD9v/yL/9i5s6dG7Fs6tSp5t///d+NMd17b8ZStOP7rNbWVpORkWH+53/+J7Rs/vz55vrrr+/trvZYtGM83efs2bYNjTnz7fiLX/zCZGRkmNra2tCys207huvOZ8L3v/99c8EFF0Qsu+mmm8w111wT+vtMX7fPOu9/ponW/v37VVlZqdmzZ4eWud1uTZ06VevXr5ckrV+/XpmZmbr44otDbWbPni273a4PPvggpv3tjb5s3LhRmzdv1u23397uvrvuuks5OTmaMmWKlixZctrLQPe2Mxnf73//e+Xk5GjcuHFauHCh6uvrIx53/PjxysvLCy275ppr5PV6tX379t4fSBd669+Tx+ORy+VSQkLkJaes2IbNzc3auHFjxPvIbrdr9uzZoffRZ61fvz6ivdS2TYLtu/PejJWejO+z6uvr1dLSouzs7Ijlq1evVm5uroqLi/XNb35Tx48f79W+d1dPx1hbW6vBgwerqKhI119/fcT76WzahlLvbMfnn39eN998s9LS0iKWny3bsSdO917sjdfts86JC+XFUmVlpSRF7KSCfwfvq6ysVG5ubsT9CQkJys7ODrWJld7oy/PPP68xY8ZoxowZEcsffvhhfe5zn1Nqaqr++te/6j/+4z9UW1urb3/7273W/9Pp6fi+8pWvaPDgwSosLNQnn3yiH/zgB9q9e7deeeWV0ON2tI2D98VSb2zDY8eO6ZFHHtHXv/71iOVWbcNjx47J5/N1+Brv2rWrw3U62ybh77vgss7axEpPxvdZP/jBD1RYWBjxgX7ttdfqy1/+soYOHaq9e/fqhz/8oebMmaP169fL4XD06hhOpydjLC4u1pIlSzRhwgR5PB498cQTmjFjhrZv366BAweeVdtQOvPt+OGHH2rbtm16/vnnI5afTduxJzp7L3q9XjU0NOjkyZNn/O//s87JMHL//ffrscce67LNzp07NXr06Bj1qPd1d4xnqqGhQS+88IJ+/OMft7svfNnkyZNVV1enxx9/vFd2ZH09vvCd8vjx41VQUKCrrrpKe/fu1fDhw3v8uNGI1Tb0er2aO3euxo4dq5/85CcR9/XlNkTPPfroo1q+fLlWr14dcYDnzTffHPr/8ePHa8KECRo+fLhWr16tq666yoquRmX69OmaPn166O8ZM2ZozJgxevbZZ/XII49Y2LO+8fzzz2v8+PGaMmVKxPJzfTta4ZwMI9/97nd16623dtlm2LBhPXrs/Px8SVJVVZUKCgpCy6uqqjRp0qRQmyNHjkSs19raqhMnToTWP1PdHeOZ9uXll19WfX295s2bd9q2U6dO1SOPPKKmpqYzvl5BrMYXNHXqVElSSUmJhg8frvz8/HZHfldVVUnSObUNa2pqdO211yojI0OvvvqqEhMTu2zfm9uwKzk5OXI4HKHXNKiqqqrTMeXn53fZvjvvzVjpyfiCnnjiCT366KN65513NGHChC7bDhs2TDk5OSopKYn5TuxMxhiUmJioyZMnq6SkRNLZtQ2lMxtjXV2dli9frocffvi0z2PlduyJzt6LLpdLKSkpcjgcZ/xvo50eHWlyDor2ANYnnngitMzj8XR4AOuGDRtCbd566y1LD2DtaV9mzpzZ7gyMzvz0pz81WVlZPe5rT/TWa7127VojyWzZssUYc+oA1vAjv5999lnjcrlMY2Nj7w2gG3o6Ro/HY6ZNm2Zmzpxp6urquvVcsdyGU6ZMMXfffXfob5/PZwYMGNDlAaxf/OIXI5ZNnz693QGsXb03Yyna8RljzGOPPWZcLpdZv359t56jrKzM2Gw2s3LlyjPub0/0ZIzhWltbTXFxsfnOd75jjDn7tqExPR/j0qVLjdPpNMeOHTvtc1i9HcOpmwewjhs3LmLZLbfc0u4A1jP5t9GuXz1a6xxy8OBBs2nTptCpq5s2bTKbNm2KOIW1uLjYvPLKK6G/H330UZOZmWlWrlxpPvnkE3P99dd3eGrv5MmTzQcffGDWrl1rRo4caempvV315dChQ6a4uNh88MEHEevt2bPH2Gw288Ybb7R7zD//+c/mN7/5jdm6davZs2eP+dWvfmVSU1PNAw880Ofj+axox1dSUmIefvhhs2HDBrN//36zcuVKM2zYMHPFFVeE1gme2nv11VebzZs3mzfffNP079/f0lN7oxmjx+MxU6dONePHjzclJSURpxC2trYaY6zfhsuXLzdOp9MsW7bM7Nixw3z96183mZmZoTOYvva1r5n7778/1P69994zCQkJ5oknnjA7d+40Dz74YIen9p7uvRkr0Y7v0UcfNUlJSebll1+O2F7Bz6Kamhpz3333mfXr15v9+/ebd955x1x44YVm5MiRMQ/IPR3jQw89ZN566y2zd+9es3HjRnPzzTeb5ORks3379lCbs2kbGhP9GIMuu+wyc9NNN7VbfjZux5qamtC+T5L5+c9/bjZt2mQOHjxojDHm/vvvN1/72tdC7YOn9n7ve98zO3fuNIsXL+7w1N6uXrdonfdhZP78+UZSu9vf//73UBsF5mII8vv95sc//rHJy8szTqfTXHXVVWb37t0Rj3v8+HFzyy23mPT0dONyucxtt90WEXBi6XR92b9/f7sxG2PMwoULTVFRkfH5fO0e84033jCTJk0y6enpJi0tzUycONE888wzHbbta9GOr7S01FxxxRUmOzvbOJ1OM2LECPO9730vYp4RY4w5cOCAmTNnjklJSTE5OTnmu9/9bsRpsbEU7Rj//ve/d/jvWpLZv3+/Mebs2Ia//OUvzaBBg0xSUpKZMmWKef/990P3zZw508yfPz+i/YsvvmhGjRplkpKSzAUXXGD+8pe/RNzfnfdmLEUzvsGDB3e4vR588EFjjDH19fXm6quvNv379zeJiYlm8ODB5s477+zxh3tviWaM9957b6htXl6e+cIXvmA+/vjjiMc727ahMdH/O921a5eRZP7617+2e6yzcTt29nkRHNf8+fPNzJkz260zadIkk5SUZIYNGxaxjwzq6nWLls2YGJ+rCQAAEIZ5RgAAgKUIIwAAwFKEEQAAYCnCCAAAsBRhBAAAWIowAgAALEUYAQAAliKMAAAASxFGAACApQgjAADAUoQRAABgKcIIAACw1P8P5BEtVrLm8XsAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "hist, bins = np.histogram(RAW_train, bins=100)\n",
    "plt.plot(bins[:-1], hist)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "ADASPP ReLU non-private accuracy: 66.93%\n"
     ]
    }
   ],
   "source": [
    "idx_subset = np.random.choice(range(X_relu_train.shape[0]), size=1000, replace=False)\n",
    "X_relu_train_subset = X_relu_train[idx_subset]\n",
    "y_hot_train_subset = y_hot_train[idx_subset]\n",
    "\n",
    "w_nonpriv_ada_relu = dp_linear.adassp(X_relu_train_subset, y_hot_train_subset, epsilon=10000, delta=0.5)\n",
    "\n",
    "# ADASPP evaluation\n",
    "accuracy_adassp_relu_nonpriv = torch.mean((torch.argmax(X_relu_test @ w_nonpriv_ada_relu, axis=1).cpu() == y_test).float())\n",
    "print(f\"ADASPP ReLU non-private accuracy: {100*accuracy_adassp_relu_nonpriv.item():.2f}%\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "ADASPP RFF non-private accuracy: 73.98%\n"
     ]
    }
   ],
   "source": [
    "X_rff_subset = X_rff[idx_subset]\n",
    "y_hot_train_subset = y_hot_train[idx_subset]\n",
    "\n",
    "w_nonpriv_ada_rff = dp_linear.adassp(X_rff_subset, y_hot_train_subset, epsilon=10000, delta=0.5)\n",
    "\n",
    "# ADASPP evaluation\n",
    "accuracy_adassp_rff_nonpriv = torch.mean((torch.argmax(X_rff_test @ w_nonpriv_ada_rff, axis=1).cpu() == y_test).float())\n",
    "print(f\"ADASPP RFF non-private accuracy: {100*accuracy_adassp_rff_nonpriv.item():.2f}%\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "# bins = np.linspace(0, 10, 100)\n",
    "# hist_rff, _ = np.histogram(np.linalg.norm(X_rff.cpu().numpy(), axis=1), bins=bins)\n",
    "# hist_relu, _ = np.histogram(np.linalg.norm(cvxnn.to_numpy(sign_patterns), axis=1), bins=bins)\n",
    "\n",
    "# plt.plot(bins[:-1], hist_rff, label='RFF')\n",
    "# plt.plot(bins[:-1], hist_relu, label='ReLU')\n",
    "# plt.legend()\n",
    "# plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "# X_rff_outer = torch.einsum('ij,ik->ijk', X_rff, X_rff).reshape(X_rff.shape[0], -1)\n",
    "# X_relu_train_outer = torch.einsum('ij,ik->ijk', X_relu_train, X_relu_train).reshape(X_relu_train.shape[0], -1)\n",
    "# print(X_rff_outer.shape)\n",
    "\n",
    "# bins = np.linspace(0, 60, 100)\n",
    "# hist_rff, _ = np.histogram(np.linalg.norm(X_rff_outer.cpu().numpy(), axis=1), bins=bins)\n",
    "# hist_relu, _ = np.histogram(np.linalg.norm(X_relu_train_outer.cpu().numpy(), axis=1), bins=bins)\n",
    "\n",
    "# plt.plot(bins[:-1], hist_rff, label='RFF')\n",
    "# plt.plot(bins[:-1], hist_relu, label='ReLU')\n",
    "# plt.legend()\n",
    "# plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|          | 0/5 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 60%|██████    | 3/5 [00:00<00:00,  5.78it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Priv ReLU 67.4% +- 73.7%\n",
      "Priv ReLU 67.4% +- 73.2%\n",
      "Priv ReLU 68.4% +- 73.2%\n",
      "Priv ReLU 67.9% +- 73.1%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 5/5 [00:00<00:00,  6.85it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Priv ReLU 68.1% +- 72.9%\n",
      "Priv ReLU 67.9% +- 0.4% - Priv RFF: 73.2% +- 0.3%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "clip_rff = 2\n",
    "clip_relu = 8\n",
    "\n",
    "results_df = pd.DataFrame(columns=[\n",
    "    \"seed\",\n",
    "    \"adassp_relu_train_acc\",\n",
    "    \"adassp_relu_test_acc\",\n",
    "    \"adassp_rff_train_acc\",\n",
    "    \"adassp_rff_test_acc\",\n",
    "    # \"relu_train_acc\",\n",
    "    # \"relu_test_acc\",\n",
    "    # \"rff_train_acc\",\n",
    "    # \"rff_test_acc\"\n",
    "])\n",
    "\n",
    "num_monte_carlo = 5\n",
    "for i in tqdm.tqdm(range(num_monte_carlo), total=num_monte_carlo):\n",
    "    w_priv_ada_relu = dp_linear.adassp(X_relu_train, y_hot_train, eps_target, delta=1e-6)\n",
    "    w_priv_ada_rff = dp_linear.adassp(X_rff, y_hot_train, eps_target, delta=1e-6)\n",
    "\n",
    "    # ADASPP evaluation\n",
    "    accuracy_adassp_relu = torch.mean((torch.argmax(X_relu_train @ w_priv_ada_relu, axis=1).cpu() == y_train).float())\n",
    "    accuracy_adassp_relu_test = torch.mean((torch.argmax(X_relu_test @ w_priv_ada_relu, axis=1).cpu() == y_test).float())\n",
    "    accuracy_adassp_rff = torch.mean((torch.argmax(X_rff @ w_priv_ada_rff, axis=1).cpu() == y_train).float())\n",
    "    accuracy_adassp_rff_test = torch.mean((torch.argmax(X_rff_test @ w_priv_ada_rff, axis=1).cpu() == y_test).float())\n",
    "\n",
    "    results_df.loc[i] = [\n",
    "       i,\n",
    "       100 * accuracy_adassp_relu.item(),\n",
    "       100 * accuracy_adassp_relu_test.item(),\n",
    "       100 * accuracy_adassp_rff.item(),\n",
    "       100 * accuracy_adassp_rff_test.item()]\n",
    "    print(f\"Priv ReLU {100*accuracy_adassp_relu_test:.1f}% +- {100*accuracy_adassp_rff_test:.1f}%\")\n",
    "\n",
    "print(f\"Priv ReLU {results_df.adassp_relu_test_acc.mean():.1f}% +- {results_df.adassp_relu_test_acc.std()*1.96/np.sqrt(num_monte_carlo):.1f}% - \"\n",
    "      f\"Priv RFF: {results_df.adassp_rff_test_acc.mean():.1f}% +- {results_df.adassp_rff_test_acc.std()*1.96/np.sqrt(num_monte_carlo):.1f}%\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "pyt310",
   "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": 2
}
