{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0e32b959-e74a-496b-a02a-4417891f6bee",
   "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",
    "\n",
    "# ===== Data Deneration =====\n",
    "def generate_data(task_id, num):\n",
    "    torch.manual_seed(task_id)\n",
    "    online_yx = torch.rand(num, 1) * 4 * np.pi - 2 * np.pi\n",
    "    online_yy = torch.sin(online_yx) + torch.randn(num, 1) * 0.1\n",
    "    return online_yx,online_yy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d688af77-e1af-4eda-b71b-d7a875c7b648",
   "metadata": {},
   "outputs": [],
   "source": [
    "from gpytorch.mlls import ExactMarginalLogLikelihood\n",
    "import torch\n",
    "import gpytorch\n",
    "\n",
    "# ===== Constructing the Gaussian regression process model to be optimized =====\n",
    "class ExactGPModel(gpytorch.models.ExactGP):\n",
    "    def __init__(self, train_x, train_y, likelihood, lengthscale, outputscale):\n",
    "        super().__init__(train_x, train_y, likelihood)\n",
    "        self.mean_module = gpytorch.means.ConstantMean()\n",
    "        self.covar_module = gpytorch.kernels.ScaleKernel(gpytorch.kernels.RBFKernel())\n",
    "        self.covar_module.base_kernel.lengthscale = lengthscale\n",
    "        self.covar_module.outputscale = outputscale\n",
    "\n",
    "    def forward(self, x):\n",
    "        mean_x = self.mean_module(x)\n",
    "        covar_x = self.covar_module(x)\n",
    "        return gpytorch.distributions.MultivariateNormal(mean_x, covar_x)\n",
    "\n",
    "    def set_train_data(self, x, y, strict=False):\n",
    "        self.train_inputs = (x,)\n",
    "        self.train_targets = y\n",
    "\n",
    "\n",
    "class ObjectiveWithSlidingWindow:\n",
    "    def __init__(self, initial_x=None, initial_y=None, max_size=100):\n",
    "        self.max_size = max_size\n",
    "        \n",
    "        if initial_x is not None and initial_y is not None:\n",
    "            self.all_train_x = initial_x.clone().detach()\n",
    "            self.all_train_y = initial_y.clone().detach()\n",
    "        else:\n",
    "            self.all_train_x = torch.empty(0)\n",
    "            self.all_train_y = torch.empty(0)\n",
    "\n",
    "    def update_and_get_data(self, new_x, new_y):\n",
    "        new_x = new_x.clone().detach().reshape(-1, 1)  \n",
    "        new_y = new_y.clone().detach()\n",
    "\n",
    "        if len(self.all_train_x) == 0:\n",
    "            self.all_train_x = new_x\n",
    "            self.all_train_y = new_y\n",
    "        else:\n",
    "            self.all_train_x = torch.cat([self.all_train_x, new_x], dim=0)\n",
    "            self.all_train_y = torch.cat([self.all_train_y, new_y], dim=0)\n",
    "\n",
    "        if len(self.all_train_x) > self.max_size:\n",
    "            self.all_train_x = self.all_train_x[-self.max_size:]\n",
    "            self.all_train_y = self.all_train_y[-self.max_size:]\n",
    "\n",
    "        return self.all_train_x, self.all_train_y\n",
    "\n",
    "    def __call__(self, hyperparams, new_x, new_y):\n",
    "        new_x = new_x.clone().detach()\n",
    "        new_y = new_y.clone().detach()\n",
    "    \n",
    "        if new_x.dim() == 1:\n",
    "            new_x = new_x.unsqueeze(-1)  # (M,) -> (M,1)\n",
    "        if len(self.all_train_x) > 0:\n",
    "            lengthscale = torch.exp(hyperparams[0][0])\n",
    "            outputscale = torch.exp(hyperparams[0][1])\n",
    "\n",
    "\n",
    "            likelihood = gpytorch.likelihoods.GaussianLikelihood()\n",
    "            likelihood.noise = 1e-4\n",
    "    \n",
    "            eval_model = ExactGPModel(\n",
    "                train_x=self.all_train_x,\n",
    "                train_y=self.all_train_y,\n",
    "                likelihood=likelihood,\n",
    "                lengthscale=lengthscale,\n",
    "                outputscale=outputscale  \n",
    "            )\n",
    "            eval_model.mean_module.constant.data.fill_(self.all_train_y.mean().item())\n",
    "    \n",
    "            eval_model.eval()\n",
    "            likelihood.eval()\n",
    "    \n",
    "            with torch.no_grad():\n",
    "                pred_dist = eval_model(new_x)\n",
    "                pred_mean_new = pred_dist.mean\n",
    "                mse = torch.nn.functional.mse_loss(pred_mean_new, new_y)\n",
    "        else:\n",
    "            pred_mean_new = torch.zeros_like(new_y)\n",
    "            mse = torch.tensor(0)\n",
    "    \n",
    "        train_x, train_y = self.update_and_get_data(new_x, new_y)\n",
    "    \n",
    "        lengthscale = torch.exp(hyperparams[0][0])\n",
    "        outputscale = torch.exp(hyperparams[0][1])\n",
    "\n",
    "        likelihood = gpytorch.likelihoods.GaussianLikelihood()\n",
    "        likelihood.noise = 1e-4\n",
    "        train_model = ExactGPModel(\n",
    "            train_x=train_x,\n",
    "            train_y=train_y,\n",
    "            likelihood=likelihood,\n",
    "            lengthscale=lengthscale,\n",
    "            outputscale=outputscale  \n",
    "        )\n",
    "        train_model.mean_module.constant.data.fill_(train_y.mean().item())\n",
    "    \n",
    "        train_model.eval()\n",
    "        likelihood.eval()\n",
    "    \n",
    "        with torch.no_grad():\n",
    "            output = train_model(train_x)\n",
    "            mll = ExactMarginalLogLikelihood(likelihood, train_model)\n",
    "            loss = -mll(output, train_y)  \n",
    "    \n",
    "        return loss.unsqueeze(0), pred_mean_new, mse"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "21d5f7e9-6cda-4725-96ed-3dcd61a90737",
   "metadata": {},
   "outputs": [],
   "source": [
    "# ===== Execute the LDP-BO algorithm, see Algorithm 1 =====\n",
    "def ldp_bo(task_id, num, delta_0, B, epsilon, delta, kappa):\n",
    "    online_yx,online_yy = generate_data(task_id, num)\n",
    "    params = torch.ones(1,dim)\n",
    "    params_bar_sin = params\n",
    "    model = DerivativeExactGPSEModel(dim)\n",
    "    objective_sin = ObjectiveWithSlidingWindow(max_size=100)\n",
    "    res = objective_sin(params,online_yx[0],online_yy[0])\n",
    "    f_params = res[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], [delta]]) + torch.ones(dim)\n",
    "    mse = torch.ones(num, 1)\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",
    "        res = objective_sin(new_x,online_yx[i],online_yy[i])\n",
    "        new_y = res[0]\n",
    "        model.append_train_data(new_x, new_y)\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",
    "                epsilon = 1\n",
    "                noise = torch.randn(dim) * (2 * B / epsilon) * math.sqrt(2 * math.log(1.25 / delta_0))\n",
    "                params = params-0.02*(i+1)**(-0.505) *( params_grad_clipped+noise)\n",
    "                acquisition_fcn.update_theta_i(params)\n",
    "                params_bar_sin = (i * params_bar_sin + params) / (i+1)\n",
    "                mse[i] = res[2]\n",
    "                bounds = torch.tensor([[-delta], [delta]]) + params\n",
    "    return mse"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f70d8146-f54e-4ee6-ac2b-fef2a8d46780",
   "metadata": {},
   "outputs": [],
   "source": [
    "from joblib import Parallel, delayed\n",
    "import torch\n",
    "import math\n",
    "import warnings\n",
    "warnings.simplefilter('ignore')\n",
    "\n",
    "num = 10000 # Sample size\n",
    "B = 1\n",
    "epsilon = 1 # 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"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3eabbb85-cfdc-47c7-bab0-88fbbb250b88",
   "metadata": {},
   "outputs": [],
   "source": [
    "# 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, num=num, delta_0=delta_0, B=B, epsilon=epsilon, delta=delta,kappa=kappa) for x in range(iter))\n",
    "results_dp_bayes = torch.stack(result_dp_bayes) \n",
    "\n",
    "# Calculate the estimated mean and standard deviation\n",
    "def compute_mean_and_std(results, iter):\n",
    "    mean_result = torch.zeros(num, dim)\n",
    "    for i in range(iter):\n",
    "        mean_result += results[i]\n",
    "    mean_result /= iter\n",
    "    variance = torch.zeros(num, dim)\n",
    "    for i in range(iter):\n",
    "        variance += (results[i] - mean_result) ** 2\n",
    "    variance /= iter  \n",
    "    std_result = torch.sqrt(variance)\n",
    "    return mean_result, std_result\n",
    "\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"
   ]
  }
 ],
 "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
}
