{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5b3cf885-b719-4253-a74f-641261bbf83a",
   "metadata": {},
   "outputs": [],
   "source": [
    "from typing import Tuple\n",
    "import torch\n",
    "import gpytorch\n",
    "import botorch\n",
    "\n",
    "%run SWC.ipynb\n",
    "%run LDP_BO.ipynb\n",
    "# ===== The Objective Function =====\n",
    "def huber_loss(residual, c=1.0):\n",
    "    abs_r = torch.abs(residual)\n",
    "    loss = torch.where(abs_r <= c, \n",
    "                       0.5 * residual.pow(2),\n",
    "                       c * abs_r - 0.5 * c**2)\n",
    "    return loss\n",
    "\n",
    "def compute_weights(X):\n",
    "    squared_norms = torch.sum(X.pow(2), dim=1)  \n",
    "    weights = torch.clamp(2.0 / squared_norms, max=1.0)  \n",
    "    return weights\n",
    "\n",
    "def objective(theta, X, y, c=1.0):\n",
    "    residuals = y - torch.matmul(X, theta.T).reshape(-1,1)  \n",
    "    huber = huber_loss(residuals, c)        \n",
    "    weights = compute_weights(X)            \n",
    "    weighted_loss = huber * weights.reshape(-1,1)         \n",
    "    return weighted_loss\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e7094178-4add-4317-b313-49a316dffffa",
   "metadata": {},
   "outputs": [],
   "source": [
    "# ===== Data Deneration =====\n",
    "def generate_data(task_id, dim, num):\n",
    "    torch.manual_seed(task_id)\n",
    "    theta = torch.ones(1,dim)\n",
    "    online_yx = torch.randn(num,dim)\n",
    "    online_yy = (online_yx @ theta.T).reshape(-1,1) + torch.randn(num).reshape(-1,1)\n",
    "    return online_yx,online_yy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9e70ba28-3540-48db-b5ef-58df5fde4f4b",
   "metadata": {},
   "outputs": [],
   "source": [
    "# ===== Execute the LDP-BO algorithm, see Algorithm 1 =====\n",
    "def ldp_bo(task_id, dim, num, delta_0, B, epsilon, delta):\n",
    "    online_yx,online_yy = generate_data(task_id, dim, num)\n",
    "    params = torch.ones(1,dim)-torch.ones(1,dim)\n",
    "    params_bar = params\n",
    "    model = DerivativeExactGPSEModel(dim)\n",
    "    f_params = objective(params,online_yx[0,].unsqueeze(0),online_yy[0])[0]\n",
    "    model.append_train_data(params, f_params)\n",
    "    acquisition_fcn = GradientInformation(model)\n",
    "    model.posterior(params) \n",
    "    acquisition_fcn.update_theta_i(params)\n",
    "    bounds = torch.tensor([[-delta_0], [delta_0]]) + torch.ones(dim)\n",
    "    dpbayes_matrix = torch.ones(num, dim)\n",
    "    for i in range(num):\n",
    "        new_x, acq_value = optimize_acqf_custom_bo(acquisition_fcn,bounds=bounds,q=1,num_restarts=2,raw_samples=32)\n",
    "        new_y = objective(new_x,online_yx[i,].unsqueeze(0),online_yy[i])\n",
    "        model.train_ys = objective(model.train_xs,online_yx[i,].unsqueeze(0),online_yy[i]).squeeze(-1)\n",
    "        model.append_train_data(new_x, new_y[0])\n",
    "        model.posterior(params)\n",
    "        acquisition_fcn.update_K_xX_dx()\n",
    "        index = dkmppf_sw(model,params,kappa)\n",
    "        train_x = model.train_xs[index,]\n",
    "        train_y = model.train_ys[index]\n",
    "        model.update_train_data(train_x, train_y)\n",
    "        model.posterior(params)\n",
    "        acquisition_fcn.update_K_xX_dx()\n",
    "        with torch.no_grad():\n",
    "                params_grad = model._get_KxX_dx(params) @ model.get_KXX_inv() @ model.train_ys\n",
    "                params_grad_clipped = params_grad * min(1, B / torch.norm(params_grad))\n",
    "                noise = torch.randn(dim)  * (2 * B / epsilon) * math.sqrt(2 * math.log(1.25 / delta_0))\n",
    "                params = params -  0.14*(i+1)**(-0.505)* ( params_grad_clipped+noise)\n",
    "                acquisition_fcn.update_theta_i(params)\n",
    "                params_bar = (i * params_bar + params) / (i+1)\n",
    "                dpbayes_matrix[i,] = params_bar\n",
    "                bounds = torch.tensor([[-delta], [delta]]) + params\n",
    "    return dpbayes_matrix"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a7afe9c5-f02c-4ca3-ac44-ec00a86eaebb",
   "metadata": {},
   "outputs": [],
   "source": [
    "from joblib import Parallel, delayed\n",
    "import torch\n",
    "import math\n",
    "import warnings\n",
    "warnings.simplefilter('ignore')\n",
    "\n",
    "num,dim = 20000,5 # Sample size and dimensions\n",
    "B = 1 # \n",
    "epsilon = 2 # Privacy parameters\n",
    "delta = 0.2 \n",
    "delta_0 = 0.2 # Acquisition function exploration bounds\n",
    "kappa = 0.1 # Compression budget\n",
    "iter = 100 # Number of replications\n",
    "\n",
    "# Note: n_jobs needs to be replaced with the number of cores on your computer\n",
    "result_dp_bayes = Parallel(n_jobs=50)(delayed(ldp_bo)(x, dim=dim, num=num, delta=delta, B=B, epsilon=epsilon, delta_0=delta_0) for x in range(iter))\n",
    "results_dp_bayes = torch.stack(result_dp_bayes) \n",
    "mean_dp_bayes,sd_dp_bayes = compute_mean_and_std(results_dp_bayes,iter)\n",
    "final_dp_bayes = pd.DataFrame(torch.cat([mean_dp_bayes,sd_dp_bayes],dim=1))\n",
    "final_dp_bayes.to_excel('dp_bayes.xlsx', index=False) # Output and save the results\n"
   ]
  }
 ],
 "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.12.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
