{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Run-script for the synthetic data and constant rate of change\n",
    "\n",
    "\n",
    "This run-script can be used to replicate the results in Table 1. Note that running this skript might take a while (up to 10h - depending on the number of trials) since some of the objective functions have to be recreated."
   ]
  },
  {
   "cell_type": "code",
   "metadata": {},
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ],
   "outputs": [],
   "execution_count": null
  },
  {
   "cell_type": "code",
   "source": [
    "import re\n",
    "import torch\n",
    "import os.path\n",
    "\n",
    "from src.triggers import get_trigger\n",
    "from src.optimize import EventTriggeredBO\n",
    "from utils.preprocessing_utils import get_environment_path\n",
    "from utils.postprocessing_utils import data2pickle, visualize_results\n",
    "from objective_functions.objective_functions import within_model_wrapper_ND\n",
    "\n",
    "import warnings\n",
    "warnings.filterwarnings('ignore')"
   ],
   "metadata": {
    "collapsed": false
   },
   "outputs": [],
   "execution_count": null
  },
  {
   "cell_type": "code",
   "source": [
    "# get path to this venv\n",
    "path = get_environment_path()\n",
    "\n",
    "# sprecify methods/algorithms\n",
    "methods = [\n",
    "    \"R_GP_UCB\",   # see Bogunovic et. al\n",
    "    \"TV_GP_UCB\",  # see Bogunovic et. al\n",
    "    \"GP_UCB\",     # see Srinivas et. al\n",
    "    \"ET_GP_UCB\",  # our method\n",
    "    \"ET_GP_UCB_backtracking\",  # our method\n",
    "]"
   ],
   "metadata": {
    "collapsed": false
   },
   "outputs": [],
   "execution_count": null
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Next, the parameter for the objective function are specified. Due to the size limitations of the upload, some of the objective functions have to be recreated when running this skript. "
   ]
  },
  {
   "cell_type": "code",
   "metadata": {},
   "source": [
    "TIME_HORIZON = 400\n",
    "FORGETTING_FACTORS = [0.01, 0.03, 0.05]   # as in the paper\n",
    "\n",
    "DIM = 4  # dimension of the objective function, choose accordingly\n",
    "\n",
    "# parameters for the objective function \n",
    "objective_function_parameters = {\n",
    "        \"dimension\": DIM,\n",
    "        \"noise\": 0.02,\n",
    "        \"time_horizon\": TIME_HORIZON,\n",
    "        \"lengthscale\": 0.2,\n",
    "        \"num_rff_features\": 1028,\n",
    "        \"compact_set\": [[0, 1]] * DIM,\n",
    "    }"
   ],
   "outputs": [],
   "execution_count": null
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": "## Get trigger and start optimization"
  },
  {
   "cell_type": "code",
   "metadata": {
    "jupyter": {
     "is_executing": true
    }
   },
   "source": [
    "MODEL_ID = 0  # artefact from earlier implementations\n",
    "TRIALS = 20\n",
    "VERBOSE = False  # turn on to see hyperparameters\n",
    "\n",
    "for forgetting_factor in FORGETTING_FACTORS:\n",
    "    objective_function_parameters[\"forgetting_factor\"] = forgetting_factor\n",
    "    \n",
    "    for trial in range(0, TRIALS):\n",
    "        seed = 0  # fix seed for reproducibility, this is the seed for the objective function, not the algorithms!!! all algorithms are still evaluated for different seeds\n",
    "\n",
    "        function = within_model_wrapper_ND(objective_function_parameters, seed)\n",
    "\n",
    "        # start optimization for each method\n",
    "        for method in methods:\n",
    "            # get trigger\n",
    "            options = {\"time_horizon\": objective_function_parameters[\"time_horizon\"],\n",
    "                       \"delta_T\": 0.1,\n",
    "                       \"r\": 1}\n",
    "\n",
    "            if \".\" in method:\n",
    "                eps_bounds = re.findall(r\"[-+]?(?:\\d*\\.*\\d+)\", method)\n",
    "                lower_bound = float(eps_bounds[0])\n",
    "                upper_bound = float(eps_bounds[1])\n",
    "                print(f\"ET-GP-UCB: \\t Lower bound: {lower_bound} \\t Upper bound: {upper_bound}\")\n",
    "\n",
    "                N = min(TIME_HORIZON, 12 * upper_bound ** (-0.25))\n",
    "                N_lower = int(round(N + 0.5))\n",
    "\n",
    "                N = min(TIME_HORIZON, 12 * lower_bound ** (-0.25))\n",
    "                N_upper = int(round(N + 0.5))\n",
    "\n",
    "                options[\"N_lower\"] = N_lower\n",
    "                options[\"N_upper\"] = N_upper\n",
    "\n",
    "            trigger = get_trigger(method, options, objective_function_parameters[\"forgetting_factor\"])\n",
    "            bounds = torch.tensor(objective_function_parameters[\"compact_set\"], dtype=torch.float).T\n",
    "\n",
    "            # use time-varying model for TV-GP-UCB\n",
    "            use_time_varying_model = method if method in [\"TV_GP_UCB\", \"UI_TVBO\"] else False\n",
    "            event_triggered_BO = EventTriggeredBO(objective_function=function,\n",
    "                                                  trigger=trigger,\n",
    "                                                  spatial_dimensions=objective_function_parameters[\"dimension\"],\n",
    "                                                  hyperparameter={\n",
    "                                                      'lengthscales': [objective_function_parameters[\"lengthscale\"]],\n",
    "                                                      'noise': objective_function_parameters[\"noise\"],\n",
    "                                                      'lengthscale_constraint': None,\n",
    "                                                      'lengthscale_hyperprior': None, },\n",
    "                                                  bounds=bounds,\n",
    "                                                  use_time_varying_model=use_time_varying_model,\n",
    "                                                  forgetting_factor=objective_function_parameters[\"forgetting_factor\"],\n",
    "                                                  learn_hyper_parameters=False,\n",
    "                                                  add_noise=True, \n",
    "                                                  verbose=VERBOSE)\n",
    "\n",
    "            queries, observations, triggered = event_triggered_BO.optimize(\n",
    "                time_horizon=objective_function_parameters[\"time_horizon\"], model_id=MODEL_ID, iteration=trial)\n",
    "            results = {\"queries\": queries,\n",
    "                       \"observations\": observations,\n",
    "                       \"triggered\": triggered}\n",
    "            name = f\"results/results_obj_2D_{forgetting_factor}_{method}_{MODEL_ID}_{trial}\".replace(\".\", \"_\")\n",
    "            data2pickle(results, path + name + \".pickle\")\n",
    "            print(f\"Results saved. \\t Trail: {trial} \\t Method: {method}\")"
   ],
   "outputs": [],
   "execution_count": null
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": "## Visualization of the results"
  },
  {
   "cell_type": "code",
   "metadata": {},
   "source": "visualize_results(\"synthetic_data_constant_eps_ND\", FORGETTING_FACTORS, TIME_HORIZON, TRIALS, DIM)",
   "outputs": [],
   "execution_count": null
  },
  {
   "cell_type": "code",
   "metadata": {
    "pycharm": {
     "name": "#%%"
    }
   },
   "source": [],
   "outputs": [],
   "execution_count": null
  }
 ],
 "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.9.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 1
}
