{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Solver configuration as a bandit problem\n",
    "\n",
    "In this tutorial we are interested in minimizing the expected branch-and-bound tree size of the SCIP solver by tuning the parameter [`branching/scorefac`](https://www.scipopt.org/doc/html/PARAMETERS.php), which takes values in the range $[0,1]$. This parameter, used in combination with the sum score function (`branching/scorefunc=s`), controls the weighting of downward and upward gain predictions in the computation of branching scores. It has a default value of 0.167.\n",
    "\n",
    "Required third-party libraries (and exact version used in the tutorial):\n",
    " - numpy (1.19.2)\n",
    " - matplotlib (3.3.2)\n",
    " - scikit-optimize (0.8.1)\n",
    "\n",
    "Ecole / Python version:\n",
    " - python (3.8.6)\n",
    " - Ecole (0.4.3)\n",
    "\n",
    "## 1. Setting up the Ecole environment\n",
    "\n",
    "We formulate this parameter tuning task as a continuum-armed bandit problem, which we instantiate using a [`Configuring`](https://doc.ecole.ai/master/reference/environments.html#configuring) environment. We request no observation (non-contextual bandit), and use the negative number of nodes as a reward (tree size minimization)."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import ecole as ec\n",
    "\n",
    "env = ec.environment.Configuring(\n",
    "\n",
    "    # set up a few SCIP parameters\n",
    "    scip_params={\n",
    "        \"branching/scorefunc\": 's',  # sum score function\n",
    "        \"branching/vanillafullstrong/priority\": 666666,  # use vanillafullstrong (highest priority)\n",
    "        \"presolving/maxrounds\": 0,  # deactivate presolving\n",
    "    },\n",
    "\n",
    "    # pure bandit, no observation\n",
    "    observation_function=None,\n",
    "\n",
    "    # minimize the total number of nodes\n",
    "    reward_function=-ec.reward.NNodes(),\n",
    "\n",
    "    # collect additional metrics for information purposes\n",
    "    information_function={\n",
    "        'nnodes': ec.reward.NNodes().cumsum(),\n",
    "        'lpiters': ec.reward.LpIterations().cumsum(),\n",
    "        'time': ec.reward.SolvingTime().cumsum(),\n",
    "    }\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We set up SCIP to use the sum score function for branching (`branching/scorefunc=s`), and the *vanillafullstrong* branching rule to mitigate the impact of branching heuristics (`branching/vanillafullstrong/priority=666666`). For the purpose of the tutorial we also deactivate presolving (`presolving/maxrounds=0`) in order to reduce computational time."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 2. Setting up the training distribution\n",
    "\n",
    "For the purpose of this tutorial we will consider randomly generated Combinatorial Auction problems, as the problem distribution for which we want to configure the solver. We hence set up a `CombinatorialAuctionGenerator` that will generate such instances on the fly."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "# infinite instance generator, new instances will be generated on-the-fly\n",
    "instances = ec.instance.CombinatorialAuctionGenerator(\n",
    "    n_items=100, n_bids=100, add_item_prob=0.7)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "For training we consider small-sized instances ($100\\times 100$), which are solved within seconds by SCIP but are difficult enough to produce tens of branch-and-bound nodes."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 3. Solving the control problem\n",
    "\n",
    "We can now readily solve the optimization problem using an off-the-shelf optimization library, such as `scikit-optimize`."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "iteration 10 / 100\n",
      "iteration 20 / 100\n",
      "iteration 30 / 100\n",
      "iteration 40 / 100\n",
      "iteration 50 / 100\n",
      "iteration 60 / 100\n",
      "iteration 70 / 100\n",
      "iteration 80 / 100\n",
      "iteration 90 / 100\n",
      "iteration 100 / 100\n"
     ]
    }
   ],
   "source": [
    "import skopt\n",
    "import numpy as np\n",
    "\n",
    "# change those values as desired\n",
    "n_iters = 100\n",
    "n_burnins = 10\n",
    "\n",
    "# make the training process deterministic\n",
    "seed = 42\n",
    "\n",
    "env.seed(seed)  # environment (SCIP)\n",
    "instances.seed(seed)  # instance generator\n",
    "rng = np.random.RandomState(seed)  # optimizer\n",
    "\n",
    "# set up the optimizer\n",
    "opt = skopt.Optimizer(\n",
    "    dimensions=[(0.0, 1.0)], base_estimator=\"GP\", n_initial_points=n_burnins, random_state=rng,\n",
    "    acq_func=\"PI\", acq_optimizer=\"sampling\", acq_optimizer_kwargs={'n_points': 10})\n",
    "\n",
    "assert n_iters > n_burnins\n",
    "\n",
    "# run the optimization\n",
    "for i in range(n_iters):\n",
    "\n",
    "    if (i+1) % 10 == 0:\n",
    "        print(f\"iteration {i+1} / {n_iters}\")\n",
    "\n",
    "    # pick up a new random instance\n",
    "    instance = next(instances)\n",
    "\n",
    "    # start a new episode\n",
    "    env.reset(instance)\n",
    "\n",
    "    # get the next action from the optimizer\n",
    "    x = opt.ask()\n",
    "    action = {\"branching/scorefac\": x[0]}\n",
    "\n",
    "    # apply the action and collect the reward\n",
    "    _, _, reward, _, _ = env.step(action)\n",
    "\n",
    "    # update the optimizer\n",
    "    opt.tell(x, -reward)  # minimize the negated reward (eq. maximize the reward)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We can now visualize the result of the optimization process."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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GD16dMWxGTNmcGcdLDT09ddf89FHH7Fu3bq41jt79mwaN27MscceC1ApMrVhJIFrgKVA00Q1kMg5mvXAUBFpJC7I04m4zvwDuNjLczHwSqIEMEWT/owbN45XX32Vffv2AW5Rs40bNzJs2DBfYfMbN25c8f35559n4sSJAGzZsoVzzjmHwYMHM3jwYD788MMqZU8++WS++eYb+vfvz/vvv1/JUtq6dSudOnUCYNq0aZx99tmccsopdO/enV/84hcVdbz++usMHDiQfv36ceKJJ7J27Voefvhh7r333op6J0+ezF133QW42GlDhw6lqKiIs846i+3btwPOSrvxxhsZMmQIRxxxBO+//37tTqxhACLSHjgNeCSR7STMolHVT0TkeWAeUAp8DkwFGgPPiciPcMro3ETJYIom/oyYNqLKsfN6n8eVg6+k+EAxY54aUyV9Yv+JTOw/ka3FWxn33LhKabMnzo7aXsuWLRkyZAivv/46Z5xxBjNmzOD8889HRPjtb39LixYtKCsr48QTT2ThwoWVoiJH45prruHnP/85w4YNY/369YwePZqlS5dWyvOPf/yD008/nfnz58esb/78+Xz++efk5ubSo0cPrrrqKvLy8rjssst477336Ny5M99++y0tWrTg8ssvp3Hjxlx//fUAvPPOOxX1XHTRRTzwwAOccMIJ3HLLLdx2223cd999AJSWlvLpp5/y2muvcdttt/H222/76qvRoMkSkeBx5KmqOjVo/z7gF0CThAqRyMpV9VYg9FFzH866STimaOoHgeGzgKJ57LHHABc2f+rUqZSWlrJp0yaWLFniW9G8/fbblcLs79y5k127dtGkSc3+byeeeCKFhYUA9OrVi3Xr1rF9+3aOP/54OnfuDIRffiCYHTt28N1333HCCScAcPHFF3PuuQefw84++2wAjjrqKNauXVsjOY0GR6mqDgqXICKnA9+o6mciMiKRQtTr4PmmaOJPNAukUXajqOmtGrWKacGE48wzz+S6665j3rx57N27l4EDB/oOmx8cmj84vby8nP/85z/k5+f7liMrK4tyb62d0LZyc3MrvmdmZlJaWupr+YHqEGgjUL9h1JLjgO+LyBggD2gqIn9TVd9OW36xEDRGytO4cWNGjBjBpEmTmDBhAuA/bP4hhxzC0qVLKS8v56WXXqo4fvLJJ/Pggw9W7PsZHuvUqROfffYZQMQ1coI55phjePfdd1mzZg0QfvmBYAoLC2nevHnF/MuTTz5ZYd0YRrxR1ZtUtb2qdsK9TP/vRCgZqOeKxiya+sOECRNYsGAB48ePB6KH1w9mypQpnH766YwaNYrDDjus4vj999/P3LlzKSoqolevXjz88MMxZbj++uv585//zLHHHsvWrVtj5m/dujVTp07l7LPPpl+/fpx//vkAjB07lpdeeqnCGSCY6dOnc8MNN1BUVMT8+fO55ZZbYrZjGKlOvV4mYPduaNIE7rgDbrghAYI1ACzMu1Eb7PpJLqmyTEDUORrP9W08MBz30uVeYBHwT+BfqlqecAlrgVk0hmEYySeiohGRx4F2wKvAH3AvVuYBRwCnAL8SkV+q6nt1IWhNCMzRWKwzwzCM5BHNorlbVReFOb4IeFFEcoAOiRErPog4ZWMWjWEYRvKIqGgiKJng9P3AyrhLFGdyckzRGIZhJJNoQ2dfABE9BVTV35txScYUjWEYRnKJNnR2uvf5U+/zSe/zB0BxwiSKM6ZoDMMwkkvE92hUdZ2qrgOOU9VfqOoX3vZLYHSkcqmGKZr0JzMzk/79+9OnTx/OPfdciov9P+dMmzaNn/3sZ3GRY9myZfTv358BAwawatWqSmljxozhu+++i0s76cTatWt5+umnky2GkeL4eWGzQESGBXZE5Fgg6X7ZfjFFk/7k5+czf/58Fi1aRE5OTpWXK8vKyupEjpdffpkzzjiDzz//nK5du1ZKe+2112jWrFlc26urftUGUzSGH/womknAn0RkrYisAR7yjqUF5nVWvxg+fDgrV65k9uzZjBw5kgsuuIC+fftSUlLCJZdcQt++fRkwYACzZs2qKPPVV19xyimn0KNHD2677baYbYQL1f/aa69x33338cgjjzBy5MgqZTp16sTWrVtZu3YtPXv25NJLL6VPnz784Ac/4O233+a4446je/fufPrppwBMnjyZCy+8kFGjRtG9e3f++te/Avju19FHH83ixYsr2h8xYgSfffYZe/bsYdKkSQwePJgBAwbwyituFY5p06Zx5plnMnbsWDp37syDDz7IPffcw4ABAxg6dGhFeJxVq1ZxyimncNRRRzF8+HCWLVsGwMSJE7n66qs59thj6dKlS0UInl/+8pe8//779O/fP66Lwxn1DFWNuAGZwM+9702Bwmj5E7U1atRIa0pRkeoZZ9S4eINnyZIlFd+vuUb1hBPiu11zTWwZCgoKVFX1wIED+v3vf18feughnTVrljZq1EhXr16tqqp33XWXTpw4UVVVly5dqocffrju3btXH3/8cT300EN169atWlxcrL1799Y5c+ZEba9v3746e/ZsVVW9+eab9RpPyFtvvVXvvPPOsGU6duyoW7Zs0TVr1mhmZqYuXLhQy8rKdODAgXrJJZdoeXm5vvzyy3qGdzHeeuutWlRUpMXFxbplyxZt3769/ve///Xdr3vuuUdvueUWVVXduHGjdu/eXVVVb7rpJn3yySdVVXX79u3avXt33b17tz7++OPatWtX3blzp37zzTfatGlT/fOf/6yqqtdee63ee++9qqo6atQoXbFihaqqfvzxxzpy5EhVVb344ot13LhxWlZWposXL9auXbuqquqsWbP0tNNOi3gug68fo+4B9mgS7tmhW1SLRlXLcMsxo6o7VXVHYtVe/LGhs/Rn79699O/fn0GDBtGhQwd+9KMfATBkyJCKEPwffPABF154IQA9e/akY8eOrFixAoCTTjqJli1bkp+fz9lnn80HH3wQsa1wofrfe6967yR37tyZvn37kpGRQe/evTnxxBMREfr27VspvP8ZZ5xBfn4+rVq1YuTIkRXWjp9+nXfeefz9738H3HIJgeUE3nzzTaZMmUL//v0ZMWIEJSUlrF+/HoCRI0fSpEkTWrduTWFhIWPHjgWokGv37t189NFHnHvuufTv35+f/OQnbNq0qULeM888k4yMDHr16sXmzZurdU6Mho2fZQI+FJEHgWeBioBjqjovYVLFEVM08cNbf6vOCczRhFJQcHCqUKPE7AsN1R/P0P3hCF4yICMjo2I/IyOjUnj/SHL56Ve7du1o2bIlCxcu5Nlnn+Uvf/lLRf4XXniBHj16VMr/ySefxJSrvLycZs2aRYxkHVw+2vk2jFD8zNEcC/QGbgfu9ra7EilUPMnJsRA0DYHjjz+ep556CoAVK1awfv36ipvtW2+9xbfffsvevXt5+eWXI0Z6hroN1f/KK69QUlLCtm3bmD17NoMHD66SJ1q/xo8fzx133MGOHTvo27cvAKNHj+aBBx6oUASff/65b3maNm1K586dKywlVWXBggVRy0Ra8sAwgompaFR1ZJhtVF0IFw/MomkYXHnllZSVldG3b1/OP/98pk2b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\n",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Best parameter value: branching/scorefac = 0.028056112224448895\n"
     ]
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "\n",
    "model = opt.models[-1]\n",
    "\n",
    "x = np.linspace(0, 1, 500)\n",
    "x_model = opt.space.transform(x.reshape(-1, 1).tolist())\n",
    "\n",
    "fig, ax1 = plt.subplots()\n",
    "\n",
    "# points sampled during optimization\n",
    "lns1 = ax1.plot(opt.Xi, opt.yi, \"r.\", markersize=8, label=\"Collected data\")\n",
    "\n",
    "# value function estimation\n",
    "y_mean, y_std = model.predict(x_model, return_std=True)\n",
    "lns2 = ax1.plot(x, y_mean, \"g--\", label=r\"Value function\")\n",
    "ax1.fill_between(x, y_mean - 1.6 * y_std, y_mean + 1.6 * y_std,\n",
    "                 alpha=0.2, fc=\"g\", ec=\"None\")\n",
    "\n",
    "# probability of improvement estimation\n",
    "x_pi = skopt.acquisition.gaussian_pi(x_model, model, y_opt=np.min(opt.yi))\n",
    "ax2 = ax1.twinx()\n",
    "lns3 = ax2.plot(x, x_pi, \"b\", label=\"Prob. of improvement\")\n",
    "\n",
    "ax1.set_title(f\"Model obtained after {n_iters} iterations\")\n",
    "ax1.set_ylabel(f\"number of nodes (neg. reward)\")\n",
    "ax1.set_xlabel(f\"$branching/scorefac$ parameter value (action)\")\n",
    "\n",
    "ax2.set_ylabel(f\"Probability value\")\n",
    "\n",
    "# Legend\n",
    "lns = lns1+lns2+lns3\n",
    "labs = [l.get_label() for l in lns]\n",
    "ax1.legend(lns, labs, loc='upper center')\n",
    "\n",
    "plt.show()\n",
    "\n",
    "# get best value based on a grid search on the value function estimator\n",
    "best_value = x[np.argmin(y_mean)]\n",
    "print(f\"Best parameter value: branching/scorefac = {best_value}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 4. Evaluation on harder instances"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "evaluating policy 'default'\n",
      "  instance 1: 237.0 nodes, 31245.0 lpiters, 90.493822 secs\n",
      "  instance 2: 129.0 nodes, 20839.0 lpiters, 51.530046 secs\n",
      "  instance 3: 85.0 nodes, 14522.0 lpiters, 30.396824000000002 secs\n",
      "  instance 4: 560.0 nodes, 61774.0 lpiters, 210.235129 secs\n",
      "  instance 5: 107.0 nodes, 16027.0 lpiters, 33.757798 secs\n",
      "  average performance: 223.6 nodes\n",
      "evaluating policy 'learned'\n",
      "  instance 1: 93.0 nodes, 16653.0 lpiters, 28.789144 secs\n",
      "  instance 2: 68.0 nodes, 13572.0 lpiters, 24.144271999999997 secs\n",
      "  instance 3: 72.0 nodes, 13019.0 lpiters, 20.179329 secs\n",
      "  instance 4: 135.0 nodes, 19150.0 lpiters, 50.97015 secs\n",
      "  instance 5: 58.0 nodes, 9649.0 lpiters, 15.742714000000001 secs\n",
      "  average performance: 85.2 nodes\n"
     ]
    }
   ],
   "source": [
    "# we set up more challenging instances\n",
    "test_instances = ec.instance.CombinatorialAuctionGenerator(\n",
    "    n_items=150, n_bids=750, add_item_prob=0.7)\n",
    "\n",
    "seed = 1337\n",
    "\n",
    "for policy in ('default', 'learned'):\n",
    "\n",
    "    print(f\"evaluating policy '{policy}'\")\n",
    "    results = []\n",
    "\n",
    "    for i in range(5):\n",
    "\n",
    "        # evaluate each policy in the exact same settings\n",
    "        env.seed(seed+i)  # environment (SCIP)\n",
    "        test_instances.seed(seed+i)  # instance generator\n",
    "\n",
    "        # pick up the next instance\n",
    "        instance = next(test_instances)\n",
    "\n",
    "        # set up the episode initial state\n",
    "        env.reset(instance)\n",
    "\n",
    "        # get the action from the policy\n",
    "        if policy == 'default':\n",
    "            action = {}  # will use the default value from SCIP\n",
    "        else:\n",
    "            action = {\"branching/scorefac\": best_value}\n",
    "\n",
    "        # apply the action and collect the reward\n",
    "        _, _, _, _, info = env.step(action)\n",
    "\n",
    "        print(f\"  instance {i+1}: {info['nnodes']} nodes, {info['lpiters']} lpiters, {info['time']} secs\")\n",
    "\n",
    "        results.append(info['nnodes'])\n",
    "\n",
    "    print(f\"  average performance: {np.mean(results)} nodes\")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.8.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
