{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# UCI: Auto-MPG"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import xgboost\n",
    "import pandas as pd\n",
    "import numpy as np\n",
    "from sklearn.metrics import mean_absolute_error, r2_score, mean_squared_error\n",
    "from sklearn.model_selection import train_test_split, GridSearchCV\n",
    "from sklearn.neural_network import MLPRegressor\n",
    "from sklearn.ensemble import GradientBoostingRegressor, RandomForestRegressor\n",
    "from sklearn.gaussian_process import GaussianProcessRegressor\n",
    "from sklearn.linear_model import LinearRegression\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "PATH = '/Users/christofferriis/OneDrive - Danmarks Tekniske Universitet/PhD/NOSTROMO/nostromo/'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>mpg</th>\n",
       "      <th>cyl</th>\n",
       "      <th>dis</th>\n",
       "      <th>hp</th>\n",
       "      <th>weight</th>\n",
       "      <th>acc</th>\n",
       "      <th>year</th>\n",
       "      <th>origin</th>\n",
       "      <th>name</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>18.0</td>\n",
       "      <td>8</td>\n",
       "      <td>307.0</td>\n",
       "      <td>130.0</td>\n",
       "      <td>3504.0</td>\n",
       "      <td>12.0</td>\n",
       "      <td>70</td>\n",
       "      <td>1</td>\n",
       "      <td>chevrolet chevelle malibu</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>15.0</td>\n",
       "      <td>8</td>\n",
       "      <td>350.0</td>\n",
       "      <td>165.0</td>\n",
       "      <td>3693.0</td>\n",
       "      <td>11.5</td>\n",
       "      <td>70</td>\n",
       "      <td>1</td>\n",
       "      <td>buick skylark 320</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>18.0</td>\n",
       "      <td>8</td>\n",
       "      <td>318.0</td>\n",
       "      <td>150.0</td>\n",
       "      <td>3436.0</td>\n",
       "      <td>11.0</td>\n",
       "      <td>70</td>\n",
       "      <td>1</td>\n",
       "      <td>plymouth satellite</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>16.0</td>\n",
       "      <td>8</td>\n",
       "      <td>304.0</td>\n",
       "      <td>150.0</td>\n",
       "      <td>3433.0</td>\n",
       "      <td>12.0</td>\n",
       "      <td>70</td>\n",
       "      <td>1</td>\n",
       "      <td>amc rebel sst</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>17.0</td>\n",
       "      <td>8</td>\n",
       "      <td>302.0</td>\n",
       "      <td>140.0</td>\n",
       "      <td>3449.0</td>\n",
       "      <td>10.5</td>\n",
       "      <td>70</td>\n",
       "      <td>1</td>\n",
       "      <td>ford torino</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "    mpg  cyl    dis     hp  weight   acc  year  origin  \\\n",
       "0  18.0    8  307.0  130.0  3504.0  12.0    70       1   \n",
       "1  15.0    8  350.0  165.0  3693.0  11.5    70       1   \n",
       "2  18.0    8  318.0  150.0  3436.0  11.0    70       1   \n",
       "3  16.0    8  304.0  150.0  3433.0  12.0    70       1   \n",
       "4  17.0    8  302.0  140.0  3449.0  10.5    70       1   \n",
       "\n",
       "                        name  \n",
       "0  chevrolet chevelle malibu  \n",
       "1          buick skylark 320  \n",
       "2         plymouth satellite  \n",
       "3              amc rebel sst  \n",
       "4                ford torino  "
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df = pd.read_csv(PATH + 'UCI_Datasets/auto-mpg.csv', header=None)\n",
    "df.columns = ['mpg', 'cyl', 'dis', 'hp', 'weight', 'acc', 'year', 'origin', 'name']\n",
    "\n",
    "# Remove rows with no 'horsepower'\n",
    "df = df[df['hp'] != '?']\n",
    "\n",
    "# Convert horsepower to float\n",
    "df['hp'] = df.hp.astype(float)\n",
    "\n",
    "#df.reset_index(drop=True)\n",
    "df.head()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "One-hot-encoding"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>origin_1</th>\n",
       "      <th>origin_2</th>\n",
       "      <th>origin_3</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>...</th>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>393</th>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>394</th>\n",
       "      <td>0</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>395</th>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>396</th>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>397</th>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "<p>392 rows × 3 columns</p>\n",
       "</div>"
      ],
      "text/plain": [
       "     origin_1  origin_2  origin_3\n",
       "0           1         0         0\n",
       "1           1         0         0\n",
       "2           1         0         0\n",
       "3           1         0         0\n",
       "4           1         0         0\n",
       "..        ...       ...       ...\n",
       "393         1         0         0\n",
       "394         0         1         0\n",
       "395         1         0         0\n",
       "396         1         0         0\n",
       "397         1         0         0\n",
       "\n",
       "[392 rows x 3 columns]"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "dummy_df = pd.get_dummies(df['origin'])\n",
    "dummy_df.columns = ['origin_1', 'origin_2', 'origin_3']\n",
    "#dummy_df.reset_index(drop=True)\n",
    "dummy_df"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(392, 10)\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>mpg</th>\n",
       "      <th>cyl</th>\n",
       "      <th>dis</th>\n",
       "      <th>hp</th>\n",
       "      <th>weight</th>\n",
       "      <th>acc</th>\n",
       "      <th>year</th>\n",
       "      <th>origin_1</th>\n",
       "      <th>origin_2</th>\n",
       "      <th>origin_3</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>18.0</td>\n",
       "      <td>8</td>\n",
       "      <td>307.0</td>\n",
       "      <td>130.0</td>\n",
       "      <td>3504.0</td>\n",
       "      <td>12.0</td>\n",
       "      <td>70</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>15.0</td>\n",
       "      <td>8</td>\n",
       "      <td>350.0</td>\n",
       "      <td>165.0</td>\n",
       "      <td>3693.0</td>\n",
       "      <td>11.5</td>\n",
       "      <td>70</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>18.0</td>\n",
       "      <td>8</td>\n",
       "      <td>318.0</td>\n",
       "      <td>150.0</td>\n",
       "      <td>3436.0</td>\n",
       "      <td>11.0</td>\n",
       "      <td>70</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>16.0</td>\n",
       "      <td>8</td>\n",
       "      <td>304.0</td>\n",
       "      <td>150.0</td>\n",
       "      <td>3433.0</td>\n",
       "      <td>12.0</td>\n",
       "      <td>70</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>17.0</td>\n",
       "      <td>8</td>\n",
       "      <td>302.0</td>\n",
       "      <td>140.0</td>\n",
       "      <td>3449.0</td>\n",
       "      <td>10.5</td>\n",
       "      <td>70</td>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "    mpg  cyl    dis     hp  weight   acc  year  origin_1  origin_2  origin_3\n",
       "0  18.0    8  307.0  130.0  3504.0  12.0    70         1         0         0\n",
       "1  15.0    8  350.0  165.0  3693.0  11.5    70         1         0         0\n",
       "2  18.0    8  318.0  150.0  3436.0  11.0    70         1         0         0\n",
       "3  16.0    8  304.0  150.0  3433.0  12.0    70         1         0         0\n",
       "4  17.0    8  302.0  140.0  3449.0  10.5    70         1         0         0"
      ]
     },
     "execution_count": 5,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "new_df = pd.concat([df, dummy_df], axis=1)\n",
    "del new_df['origin'], new_df['name']\n",
    "print(new_df.shape)\n",
    "new_df.head()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Search_space"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(392, 8)"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "new_df.iloc[:, 1:-1].values.shape"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Make search space"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>mpg</th>\n",
       "      <th>cyl</th>\n",
       "      <th>dis</th>\n",
       "      <th>hp</th>\n",
       "      <th>weight</th>\n",
       "      <th>acc</th>\n",
       "      <th>year</th>\n",
       "      <th>origin</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>count</th>\n",
       "      <td>392.000000</td>\n",
       "      <td>392.000000</td>\n",
       "      <td>392.000000</td>\n",
       "      <td>392.000000</td>\n",
       "      <td>392.000000</td>\n",
       "      <td>392.000000</td>\n",
       "      <td>392.000000</td>\n",
       "      <td>392.000000</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>mean</th>\n",
       "      <td>23.445918</td>\n",
       "      <td>5.471939</td>\n",
       "      <td>194.411990</td>\n",
       "      <td>104.469388</td>\n",
       "      <td>2977.584184</td>\n",
       "      <td>15.541327</td>\n",
       "      <td>75.979592</td>\n",
       "      <td>1.576531</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>std</th>\n",
       "      <td>7.805007</td>\n",
       "      <td>1.705783</td>\n",
       "      <td>104.644004</td>\n",
       "      <td>38.491160</td>\n",
       "      <td>849.402560</td>\n",
       "      <td>2.758864</td>\n",
       "      <td>3.683737</td>\n",
       "      <td>0.805518</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>min</th>\n",
       "      <td>9.000000</td>\n",
       "      <td>3.000000</td>\n",
       "      <td>68.000000</td>\n",
       "      <td>46.000000</td>\n",
       "      <td>1613.000000</td>\n",
       "      <td>8.000000</td>\n",
       "      <td>70.000000</td>\n",
       "      <td>1.000000</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>25%</th>\n",
       "      <td>17.000000</td>\n",
       "      <td>4.000000</td>\n",
       "      <td>105.000000</td>\n",
       "      <td>75.000000</td>\n",
       "      <td>2225.250000</td>\n",
       "      <td>13.775000</td>\n",
       "      <td>73.000000</td>\n",
       "      <td>1.000000</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>50%</th>\n",
       "      <td>22.750000</td>\n",
       "      <td>4.000000</td>\n",
       "      <td>151.000000</td>\n",
       "      <td>93.500000</td>\n",
       "      <td>2803.500000</td>\n",
       "      <td>15.500000</td>\n",
       "      <td>76.000000</td>\n",
       "      <td>1.000000</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>75%</th>\n",
       "      <td>29.000000</td>\n",
       "      <td>8.000000</td>\n",
       "      <td>275.750000</td>\n",
       "      <td>126.000000</td>\n",
       "      <td>3614.750000</td>\n",
       "      <td>17.025000</td>\n",
       "      <td>79.000000</td>\n",
       "      <td>2.000000</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>max</th>\n",
       "      <td>46.600000</td>\n",
       "      <td>8.000000</td>\n",
       "      <td>455.000000</td>\n",
       "      <td>230.000000</td>\n",
       "      <td>5140.000000</td>\n",
       "      <td>24.800000</td>\n",
       "      <td>82.000000</td>\n",
       "      <td>3.000000</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "              mpg         cyl         dis          hp       weight  \\\n",
       "count  392.000000  392.000000  392.000000  392.000000   392.000000   \n",
       "mean    23.445918    5.471939  194.411990  104.469388  2977.584184   \n",
       "std      7.805007    1.705783  104.644004   38.491160   849.402560   \n",
       "min      9.000000    3.000000   68.000000   46.000000  1613.000000   \n",
       "25%     17.000000    4.000000  105.000000   75.000000  2225.250000   \n",
       "50%     22.750000    4.000000  151.000000   93.500000  2803.500000   \n",
       "75%     29.000000    8.000000  275.750000  126.000000  3614.750000   \n",
       "max     46.600000    8.000000  455.000000  230.000000  5140.000000   \n",
       "\n",
       "              acc        year      origin  \n",
       "count  392.000000  392.000000  392.000000  \n",
       "mean    15.541327   75.979592    1.576531  \n",
       "std      2.758864    3.683737    0.805518  \n",
       "min      8.000000   70.000000    1.000000  \n",
       "25%     13.775000   73.000000    1.000000  \n",
       "50%     15.500000   76.000000    1.000000  \n",
       "75%     17.025000   79.000000    2.000000  \n",
       "max     24.800000   82.000000    3.000000  "
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df.describe()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Large search space"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "mpg (10,) (not used for prediction)\n",
      "cyl (5,)\n",
      "dis (10,)\n",
      "hp (10,)\n",
      "weight (8,)\n",
      "acc (9,)\n",
      "year (13,)\n",
      "origin (3,)\n",
      "\n",
      "Combinations 1404000\n"
     ]
    }
   ],
   "source": [
    "mpg = np.arange(9, 46.6, 4)\n",
    "cyl = df['cyl'].unique()\n",
    "dis = np.arange(68, 455, 40)\n",
    "hp = np.arange(46, 230, 20)\n",
    "weight = np.arange(1613, 5140, 500)\n",
    "acc = np.arange(8, 24.8, 2)\n",
    "year = np.arange(70, 82.1, 1)\n",
    "origin = df['origin'].unique()\n",
    "\n",
    "print(\"mpg\", mpg.shape, \"(not used for prediction)\")\n",
    "print(\"cyl\", cyl.shape)\n",
    "print(\"dis\", dis.shape)\n",
    "print(\"hp\", hp.shape)\n",
    "print(\"weight\", weight.shape)\n",
    "print(\"acc\", acc.shape)\n",
    "print(\"year\", year.shape)\n",
    "print(\"origin\", origin.shape)\n",
    "\n",
    "print(\"\\nCombinations\", cyl.shape[0] * dis.shape[0] * hp.shape[0] * weight.shape[0] * acc.shape[0] * year.shape[0] * origin.shape[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 583,
   "metadata": {},
   "outputs": [],
   "source": [
    "# All combinations\n",
    "giant_search_space = np.meshgrid(cyl, dis, hp, weight, acc, year, origin, indexing='ij')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 584,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(5, 10, 10, 8, 9, 13, 3)\n",
      "(5, 10, 10, 8, 9, 13, 3)\n",
      "(5, 10, 10, 8, 9, 13, 3)\n",
      "(5, 10, 10, 8, 9, 13, 3)\n",
      "(5, 10, 10, 8, 9, 13, 3)\n",
      "(5, 10, 10, 8, 9, 13, 3)\n",
      "(5, 10, 10, 8, 9, 13, 3)\n"
     ]
    }
   ],
   "source": [
    "for i in range(7):\n",
    "    print(giant_search_space[i].shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 585,
   "metadata": {},
   "outputs": [],
   "source": [
    "giant_search_space2 = np.array(giant_search_space).T.reshape(-1,7)\n",
    "with open(\"state_dicts/auto_mpg-search_space.pkl\", 'wb') as fp:\n",
    "    pickle.dump(giant_search_space2, fp)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 586,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(1404000, 7)\n",
      "[1. 2. 3.]\n",
      "1404000\n"
     ]
    }
   ],
   "source": [
    "with open(\"state_dicts/auto_mpg-search_space.pkl\", 'rb') as fp:\n",
    "    ko = pickle.load(fp)\n",
    "\n",
    "print(ko.shape)\n",
    "print(np.unique(ko[:,6]))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Small search space"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "mpg (8,) (not used for prediction)\n",
      "cyl (5,)\n",
      "dis (8,)\n",
      "hp (7,)\n",
      "weight (8,)\n",
      "acc (5,)\n",
      "year (7,)\n",
      "origin (3,)\n",
      "\n",
      "Combinations 235200\n"
     ]
    }
   ],
   "source": [
    "mpg = np.arange(9, 46.6, 5)\n",
    "cyl = df['cyl'].unique()\n",
    "dis = np.arange(68, 455, 50)\n",
    "hp = np.arange(46, 230, 30)\n",
    "weight = np.arange(1613, 5140, 500)\n",
    "acc = np.arange(8, 24.8, 4)\n",
    "year = np.arange(70, 82.1, 2)\n",
    "origin = df['origin'].unique()\n",
    "\n",
    "print(\"mpg\", mpg.shape, \"(not used for prediction)\")\n",
    "print(\"cyl\", cyl.shape)\n",
    "print(\"dis\", dis.shape)\n",
    "print(\"hp\", hp.shape)\n",
    "print(\"weight\", weight.shape)\n",
    "print(\"acc\", acc.shape)\n",
    "print(\"year\", year.shape)\n",
    "print(\"origin\", origin.shape)\n",
    "\n",
    "print(\"\\nCombinations\", cyl.shape[0] * dis.shape[0] * hp.shape[0] * weight.shape[0] * acc.shape[0] * year.shape[0] * origin.shape[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "large_search_space = np.meshgrid(cyl, dis, hp, weight, acc, year, origin, indexing='ij')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Define data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "     cyl    dis     hp  weight   acc  year  origin\n",
      "0      8  307.0  130.0  3504.0  12.0    70       1\n",
      "1      8  350.0  165.0  3693.0  11.5    70       1\n",
      "2      8  318.0  150.0  3436.0  11.0    70       1\n",
      "3      8  304.0  150.0  3433.0  12.0    70       1\n",
      "4      8  302.0  140.0  3449.0  10.5    70       1\n",
      "..   ...    ...    ...     ...   ...   ...     ...\n",
      "393    4  140.0   86.0  2790.0  15.6    82       1\n",
      "394    4   97.0   52.0  2130.0  24.6    82       2\n",
      "395    4  135.0   84.0  2295.0  11.6    82       1\n",
      "396    4  120.0   79.0  2625.0  18.6    82       1\n",
      "397    4  119.0   82.0  2720.0  19.4    82       1\n",
      "\n",
      "[392 rows x 7 columns]\n",
      "0      18.0\n",
      "1      15.0\n",
      "2      18.0\n",
      "3      16.0\n",
      "4      17.0\n",
      "       ... \n",
      "393    27.0\n",
      "394    44.0\n",
      "395    32.0\n",
      "396    28.0\n",
      "397    31.0\n",
      "Name: mpg, Length: 392, dtype: float64\n"
     ]
    }
   ],
   "source": [
    "# Target: mpg, Features: all other variables except the name\n",
    "X, y = df.iloc[:,1:8],df.iloc[:,0]\n",
    "print(X)\n",
    "print(y)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "x_train, x_test, y_train, y_test=train_test_split(X, y, shuffle=True, random_state=42)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Try different models"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "mlp=MLPRegressor()\n",
    "gp=GaussianProcessRegressor(normalize_y=True)\n",
    "gb=GradientBoostingRegressor()\n",
    "rf=RandomForestRegressor()\n",
    "lr=LinearRegression()\n",
    "xgb = xgboost.XGBRegressor(colsample_bytree=0.4,\n",
    "                 gamma=0,                 \n",
    "                 learning_rate=0.1,\n",
    "                 max_depth=5,\n",
    "                 min_child_weight=1.5,\n",
    "                 n_estimators=200,) "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "def evaluate_models(models, x_test, y_test, x_train=[], y_train=[]):\n",
    "    if isinstance(x_train, pd.DataFrame):  \n",
    "        fit=True   #just for clarity\n",
    "    else: \n",
    "        fit=False\n",
    "        \n",
    "    for model, name in models:\n",
    "        if fit:\n",
    "            model.fit(x_train, y_train)\n",
    "            preds_train=model.predict(x_train)\n",
    "        preds_test=model.predict(x_test)\n",
    "        print(name+\":\")\n",
    "        if fit:\n",
    "            print(\"R^2:  training performance: %.1f testset performance: %.4f\"%(r2_score(y_train, preds_train),r2_score(y_test, preds_test)))\n",
    "            print(\"MAE:  training performance: %.1f testset performance: %.4f\"%(mean_absolute_error(y_train, preds_train),mean_absolute_error(y_test, preds_test)))\n",
    "            print(\"RMSE: training performance: %.1f testset performance: %.4f\"%(mean_squared_error(y_train, preds_train),mean_squared_error(y_test, preds_test)))\n",
    "        else:\n",
    "            print(\"R^2:  testset performance: %.4f\"%(r2_score(y_test, preds_test)))\n",
    "            print(\"MAE:  testset performance: %.4f\"%(mean_absolute_error(y_test, preds_test)))\n",
    "            print(\"RMSE: testset performance: %.4f\"%(mean_squared_error(y_test, preds_test)))\n",
    "        print(\"-\"*20+\"\\n\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "XGBoost:\n",
      "R^2:  training performance: 1.0 testset performance: 0.8536\n",
      "MAE:  training performance: 0.5 testset performance: 1.9725\n",
      "RMSE: training performance: 0.4 testset performance: 7.3856\n",
      "--------------------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/Users/christofferriis/envs/nostromo/lib/python3.9/site-packages/sklearn/neural_network/_multilayer_perceptron.py:614: ConvergenceWarning: Stochastic Optimizer: Maximum iterations (200) reached and the optimization hasn't converged yet.\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "MultiLayer Perceptron:\n",
      "R^2:  training performance: 0.7 testset performance: 0.5884\n",
      "MAE:  training performance: 3.4 testset performance: 3.5851\n",
      "RMSE: training performance: 20.8 testset performance: 20.7583\n",
      "--------------------\n",
      "\n",
      "Gaussian Processes:\n",
      "R^2:  training performance: 1.0 testset performance: -0.0173\n",
      "MAE:  training performance: 0.0 testset performance: 5.9344\n",
      "RMSE: training performance: 0.0 testset performance: 51.3063\n",
      "--------------------\n",
      "\n",
      "Gradient Boosting:\n",
      "R^2:  training performance: 1.0 testset performance: 0.8508\n",
      "MAE:  training performance: 1.0 testset performance: 1.8403\n",
      "RMSE: training performance: 1.7 testset performance: 7.5238\n",
      "--------------------\n",
      "\n",
      "Random Forest:\n",
      "R^2:  training performance: 1.0 testset performance: 0.8786\n",
      "MAE:  training performance: 0.8 testset performance: 1.7598\n",
      "RMSE: training performance: 1.2 testset performance: 6.1244\n",
      "--------------------\n",
      "\n",
      "Linear Regression:\n",
      "R^2:  training performance: 0.8 testset performance: 0.7989\n",
      "MAE:  training performance: 2.6 testset performance: 2.4086\n",
      "RMSE: training performance: 11.2 testset performance: 10.1426\n",
      "--------------------\n",
      "\n"
     ]
    }
   ],
   "source": [
    "evaluate_models([(xgb, 'XGBoost'), (mlp, \"MultiLayer Perceptron\"), (gp, \"Gaussian Processes\"), (gb, \"Gradient Boosting\"), (rf, \"Random Forest\"), (lr, \"Linear Regression\")], x_test, y_test, x_train, y_train)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Can we get some explanations and feature importance?"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "import shap"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [],
   "source": [
    "explainer = shap.Explainer(xgb)\n",
    "shap_values = explainer(x_train)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 576x309.6 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# summarize the effects of all the features\n",
    "shap.plots.beeswarm(shap_values)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "A tuned XGBoost"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "XGBRegressor(base_score=0.5, booster='gbtree', colsample_bylevel=1,\n",
       "             colsample_bynode=1, colsample_bytree=0.4, gamma=0, gpu_id=-1,\n",
       "             importance_type='gain', interaction_constraints='',\n",
       "             learning_rate=0.1, max_delta_step=0, max_depth=3,\n",
       "             min_child_weight=1.5, missing=nan, monotone_constraints='()',\n",
       "             n_estimators=200, n_jobs=8, num_parallel_tree=1, random_state=0,\n",
       "             reg_alpha=0.5, reg_lambda=0.1, scale_pos_weight=1, subsample=1,\n",
       "             tree_method='exact', validate_parameters=1, verbosity=None)"
      ]
     },
     "execution_count": 41,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "cv_params = {\n",
    "    'learning_rate': [0.001, 0.01, 0.05, 0.1, 0.2],\n",
    "    'max_depth': [3, 5, 10],\n",
    "    'reg_alpha': [0.1, 0.5, 1],\n",
    "    'reg_lambda': [0.1, 0.5, 1]\n",
    "}\n",
    "clf = GridSearchCV(estimator=xgb, param_grid=cv_params, n_jobs=-1)\n",
    "clf.fit(x_train, y_train)\n",
    "clf.best_estimator_"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "XGBoost:\n",
      "R^2:  training performance: 1.0 testset performance: 0.8883\n",
      "MAE:  training performance: 0.5 testset performance: 2.0770\n",
      "RMSE: training performance: 0.4 testset performance: 7.6870\n",
      "--------------------\n",
      "\n",
      "XGBoost tuned:\n",
      "R^2:  training performance: 1.0 testset performance: 0.8893\n",
      "MAE:  training performance: 0.9 testset performance: 2.1123\n",
      "RMSE: training performance: 1.5 testset performance: 7.6145\n",
      "--------------------\n",
      "\n"
     ]
    }
   ],
   "source": [
    "evaluate_models([(xgb, 'XGBoost'), (clf, \"XGBoost tuned\"),], x_test, y_test)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Make a dummy simulator from Auto-MPG"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 287,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "import gpytorch\n",
    "import torch\n",
    "from gpytorch.kernels import RBFKernel\n",
    "from utils.gp_utils import get_model, get_likelihood, get_optimizer, get_loss, fit_gpytorch_model\n",
    "from utils.transformations import transform\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 288,
   "metadata": {},
   "outputs": [],
   "source": [
    "class argument_handler:\n",
    "    def __init__(self):\n",
    "        # default configurations\n",
    "        self.seed = 2\n",
    "        \n",
    "        # Simulator\n",
    "        self.simulator = \"homo1\"\n",
    "\n",
    "        # Model related parameters\n",
    "        # Choose model= [homoscedastic/exact, heteroskedastic, ngd_model]\n",
    "        self.model_type = 'heteroskedastic'\n",
    "        self.set_prior = False\n",
    "        self.outputs = 1\n",
    "        self.inducing_points = 100\n",
    "        self.n_samples = 10\n",
    "\n",
    "        # Initial data set\n",
    "        self.initial_samples = 8\n",
    "        self.test_samples = 1020\n",
    "\n",
    "        # Data\n",
    "        self.path_train_data= \"/data_mercury/first_results.pkl\"\n",
    "        self.path_test_data= \"/data_mercury/first_results_test_inputs.pkl\"\n",
    "        # Transformation of data= [standardize, min_max_feature_scaling, minusone_one_feature_scaling, identity]\n",
    "        self.transformation_x = 'standardize'\n",
    "        self.transformation_y = 'standardize'\n",
    "\n",
    "        # How to query the next point? [variance, cross_corr, combi, random]\n",
    "        self.selection_criteria = 'random'\n",
    "        self.min_change_in_var = 0.1 # relative change (set to -1 when not using)\n",
    "\n",
    "        # Parameters for the active learning scheme\n",
    "        self.plot = True\n",
    "        self.active_learning_steps = 3\n",
    "        self.k_samples = 1        # Size of the batch of new points to query\n",
    "        self.beta_sampling = 0.1 # Distance between the points in the batch given by percentage of range\n",
    "        self.repeat_sampling = 8  # Number of times to sample for each input,\n",
    "                            # i.e. equal 4 will give for simulations for each data point\n",
    "\n",
    "        # Hyperparameters for optimizing hyperparameters of GP\n",
    "        self.milestones = [300, 600]        # Epochs at which we will lower the learning rate by a factor (non-variational optimizer)\n",
    "        self.initial_lr = .1                # Initial learning rate\n",
    "        self.n_epochs = 1000                # Number of epochs in optimizing the hyperparameters\n",
    "        self.n_runs = 1                     # Number of runs to try to find the global minimum\n",
    "        \n",
    "\n",
    "        # Output file\n",
    "        self.output_file = 'example.pkl'\n",
    "\n",
    "        \n",
    "args = argument_handler()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 289,
   "metadata": {},
   "outputs": [],
   "source": [
    "df = pd.read_csv(PATH + 'UCI_Datasets/auto-mpg.csv', header=None)\n",
    "df.columns = ['mpg', 'cyl', 'dis', 'hp', 'weight', 'acc', 'year', 'origin', 'name']\n",
    "df = df[df['hp'] != '?']  # Remove rows with no 'horsepower'\n",
    "df['hp'] = df.hp.astype(float)  # Convert horsepower to float\n",
    "X, y = df.iloc[:,1:8],df.iloc[:,0]\n",
    "\n",
    "# Split data\n",
    "x_train, x_test, y_train, y_test=train_test_split(X, y, shuffle=True, random_state=42)\n",
    "x_train = torch.Tensor(x_train.values)\n",
    "x_test = torch.Tensor(x_test.values)\n",
    "y_train = torch.Tensor(y_train.values)\n",
    "y_test = torch.Tensor(y_test.values)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 290,
   "metadata": {},
   "outputs": [],
   "source": [
    "train_x, mu_x, sigma_x = transform(x_train, method=args.transformation_x)\n",
    "train_y, mu_y, sigma_y = transform(y_train, method=args.transformation_y)\n",
    "\n",
    "model = get_model(args, train_x, train_y, RBFKernel(ard_num_dims=7), likelihood=get_likelihood(args))\n",
    "optimizer, scheduler = get_optimizer(args, model, num_data=train_y.size(0))\n",
    "mll = get_loss(args, model, num_data=train_y.size(0))\n",
    "nmll_loss, fit_losses = fit_gpytorch_model(args, (train_x, train_y), model, optimizer, scheduler, mll)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 291,
   "metadata": {},
   "outputs": [],
   "source": [
    "model.eval()\n",
    "with torch.no_grad(), gpytorch.settings.fast_pred_var():\n",
    "    # If train_x is equal to pred_x a warning about model.train() occur. This can be ignored.\n",
    "    v_min, _ = torch.min(train_x, dim=0)\n",
    "    v_max, _ = torch.max(train_x, dim=0)\n",
    "    pred_x = torch.from_numpy(np.linspace(v_min, v_max, 101)) ## THIS IS WRONG. We want a grid 101^7 or do we... is it better to save and load the model everytime?\n",
    "    pred_y = model.likelihood(model(pred_x))\n",
    "    mean = pred_y.mean\n",
    "    lower, upper = pred_y.confidence_region()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 292,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 600x150 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# clear_output()\n",
    "import gpytorch\n",
    "from utils.plotting import plot_single_flow, plot_one_variable\n",
    "\n",
    "p_dct = {}\n",
    "FEATURE = 0\n",
    "with torch.no_grad(), gpytorch.settings.fast_pred_var():\n",
    "    tmp_train_x = train_x[:,FEATURE]\n",
    "    tmp_mu_x = mu_x[FEATURE]\n",
    "    tmp_sigma_x = sigma_x[FEATURE]\n",
    "    tmp_pred_x = pred_x[:,FEATURE]\n",
    "    p_dct['train_x'] = transform(tmp_train_x, tmp_mu_x, tmp_sigma_x, method=args.transformation_x, inverse=True)\n",
    "    p_dct['train_y'] = transform(train_y, mu_y, sigma_y, method=args.transformation_y, inverse=True)\n",
    "    p_dct['test_x']  = transform(tmp_train_x, tmp_mu_x, tmp_sigma_x, method=args.transformation_x, inverse=True)\n",
    "    p_dct['test_y']  = transform(train_y, mu_y, sigma_y, method=args.transformation_y, inverse=True)\n",
    "    p_dct['pred_x']  = transform(tmp_pred_x, tmp_mu_x, tmp_sigma_x, method=args.transformation_x, inverse=True)\n",
    "    p_dct['mean']   = transform(mean, mu_y, sigma_y, method=args.transformation_y, inverse=True)\n",
    "    p_dct['lower'] = transform(lower, mu_y, sigma_y, method=args.transformation_y, inverse=True)\n",
    "    p_dct['upper'] = transform(upper, mu_y, sigma_y, method=args.transformation_y, inverse=True)\n",
    "    p_dct['variance'] = upper - lower #p_dct['upper'] - p_dct['lower']\n",
    "    #p_dct['variance'] = p_dct['variance'] / torch.sum(p_dct['variance'], dim=0)\n",
    "\n",
    "if args.outputs == 1:\n",
    "    #plot_single_flow(ax, p_dct, new_points=args.k_samples * args.repeat_sampling, test_data=True)\n",
    "    #ax.set_xlabel(\"Turn-around time [min]\")\n",
    "    #ax.set_ylabel(\"Flight delay (mean arrival) [min]\")\n",
    "    f, ax = plt.subplots(1, 2, figsize=(3 * 4, 3), dpi=50)\n",
    "    ax[0] = plot_single_flow(ax[0], p_dct, new_points=0,\n",
    "                             test_data=False)\n",
    "    #ax[1] = plot_mll(ax[1], nmll_losses_valid)\n",
    "    #ax[2] = plot_rmse(ax[2],rmse_losses_valid)\n",
    "    #ax[3] = plot_one_variable(ax[3], p_dct['variance'], \"Variance\")\n",
    "    ax[1] = plot_one_variable(ax[1], fit_losses, \"Fitting loss\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 293,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "mm = 1\n",
    "new_samples = pred_y.to_data_independent_dist().sample([mm])\n",
    "new_samples = transform(new_samples, mu_y, sigma_y, method=args.transformation_y, inverse=True)\n",
    "pred_x_t = transform(pred_x, mu_x, sigma_x, method=args.transformation_x, inverse=True)\n",
    "for j in range(mm):\n",
    "    plt.scatter(pred_x_t[:,3], new_samples[j,:], c='b', marker='.')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 294,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.collections.PolyCollection at 0x143d86430>"
      ]
     },
     "execution_count": 294,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "mu = pred_y.mean\n",
    "std = torch.sqrt(pred_y.covariance_matrix.diag())\n",
    "plt.plot(pred_x_t[:,0], mu)\n",
    "plt.fill_between(pred_x_t[:,0], mu-2*std, mu+2*std, alpha=.2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 295,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "R^2:  training performance: 0.9 testset performance: 0.8634\n",
      "MAE:  training performance: 1.1 testset performance: 1.8886\n",
      "RMSE: training performance: 3.9 testset performance: 6.8897\n"
     ]
    }
   ],
   "source": [
    "test_x, _, _ = transform(x_test, mu_x, sigma_x, method=args.transformation_x)\n",
    "\n",
    "model.eval()\n",
    "with torch.no_grad(), gpytorch.settings.fast_pred_var():\n",
    "    pred_test = model.likelihood(model(test_x))\n",
    "    mean_test = pred_test.mean\n",
    "    pred_train = model.likelihood(model(train_x))\n",
    "    mean_train = pred_train.mean\n",
    "    \n",
    "preds_test = transform(mean_test, mu_y, sigma_y, method=args.transformation_y, inverse=True)\n",
    "preds_train = transform(mean_train, mu_y, sigma_y, method=args.transformation_y, inverse=True)\n",
    "\n",
    "print(\"R^2:  training performance: %.1f testset performance: %.4f\"%(r2_score(y_train, preds_train),r2_score(y_test, preds_test)))\n",
    "print(\"MAE:  training performance: %.1f testset performance: %.4f\"%(mean_absolute_error(y_train, preds_train),mean_absolute_error(y_test, preds_test)))\n",
    "print(\"RMSE: training performance: %.1f testset performance: %.4f\"%(mean_squared_error(y_train, preds_train),mean_squared_error(y_test, preds_test)))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Train model on full data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 329,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Train on all the data\n",
    "x_train = torch.Tensor(X.values)\n",
    "y_train = torch.Tensor(y.values)\n",
    "\n",
    "# Transform data\n",
    "train_x, mu_x, sigma_x = transform(x_train, method=args.transformation_x)\n",
    "train_y, mu_y, sigma_y = transform(y_train, method=args.transformation_y)\n",
    "\n",
    "# Define and fit model\n",
    "model = get_model(args, train_x, train_y, RBFKernel(ard_num_dims=7), likelihood=get_likelihood(args))\n",
    "optimizer, scheduler = get_optimizer(args, model, num_data=train_y.size(0))\n",
    "mll = get_loss(args, model, num_data=train_y.size(0))\n",
    "nmll_loss, fit_losses = fit_gpytorch_model(args, (train_x, train_y), model, optimizer, scheduler, mll)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 330,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "R^2:  training performance: 0.9383\n",
      "MAE:  training performance: 1.1561\n",
      "RMSE: training performance: 3.7483\n"
     ]
    }
   ],
   "source": [
    "model.eval()\n",
    "with torch.no_grad(), gpytorch.settings.fast_pred_var():\n",
    "    pred_train = model.likelihood(model(train_x))\n",
    "    mean_train = pred_train.mean\n",
    "\n",
    "preds_train = transform(mean_train, mu_y, sigma_y, method=args.transformation_y, inverse=True)\n",
    "\n",
    "print(\"R^2:  training performance: %.4f\"%(r2_score(y_train, preds_train)))\n",
    "print(\"MAE:  training performance: %.4f\"%(mean_absolute_error(y_train, preds_train)))\n",
    "print(\"RMSE: training performance: %.4f\"%(mean_squared_error(y_train, preds_train)))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Save the model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 331,
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.save(model.state_dict(), 'state_dicts/auto_mpg-model_state.pth')\n",
    "\n",
    "import pickle\n",
    "auto_mpg_info = {\n",
    "    'train_x': train_x,\n",
    "    'mu_x': mu_x,\n",
    "    'sigma_x': sigma_x,\n",
    "    'mu_y': mu_y,\n",
    "    'sigma_y': sigma_y\n",
    "}\n",
    "\n",
    "\n",
    "with open('state_dicts/auto_mpg-data.pkl', 'wb') as fp:\n",
    "    pickle.dump(auto_mpg_info, fp)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 332,
   "metadata": {},
   "outputs": [],
   "source": [
    "with open('state_dicts/auto_mpg-data.pkl', 'rb') as fp:\n",
    "    ko = pickle.load(fp)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 333,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'train_x': tensor([[ 1.4821,  1.0759,  0.6633,  ..., -1.2836, -1.6232, -0.7157],\n",
       "         [ 1.4821,  1.4868,  1.5726,  ..., -1.4649, -1.6232, -0.7157],\n",
       "         [ 1.4821,  1.1810,  1.1829,  ..., -1.6461, -1.6232, -0.7157],\n",
       "         ...,\n",
       "         [-0.8629, -0.5678, -0.5318,  ..., -1.4286,  1.6343, -0.7157],\n",
       "         [-0.8629, -0.7111, -0.6617,  ...,  1.1087,  1.6343, -0.7157],\n",
       "         [-0.8629, -0.7207, -0.5838,  ...,  1.3986,  1.6343, -0.7157]]),\n",
       " 'mu_x': tensor([5.4719e+00, 1.9441e+02, 1.0447e+02, 2.9776e+03, 1.5541e+01, 7.5980e+01,\n",
       "         1.5765e+00]),\n",
       " 'sigma_x': tensor([1.7058e+00, 1.0464e+02, 3.8491e+01, 8.4940e+02, 2.7589e+00, 3.6837e+00,\n",
       "         8.0552e-01]),\n",
       " 'mu_y': tensor(23.4459),\n",
       " 'sigma_y': tensor(7.8050)}"
      ]
     },
     "execution_count": 333,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "ko"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Auto-MPG simulator"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 478,
   "metadata": {},
   "outputs": [],
   "source": [
    "def auto_mpg_simulator(x):\n",
    "    \"\"\"\n",
    "    \n",
    "    :param x: non-transformed data points\n",
    "    :returns: a new sample, the mean and the standard deviation for each input data point (in original space)\n",
    "    \"\"\"\n",
    "    \n",
    "    # Load state dict and info\n",
    "    state_dict = torch.load('state_dicts/auto_mpg-model_state.pth')\n",
    "    with open('state_dicts/auto_mpg-data.pkl', 'rb') as fp:\n",
    "        info = pickle.load(fp)\n",
    "    \n",
    "    # Make the model\n",
    "    model = ApproximateGPModel(inducing_points=info['train_x'], kernel=RBFKernel(ard_num_dims=7), likelihood=gpytorch.likelihoods.GaussianLikelihood())\n",
    "    model.load_state_dict(state_dict)\n",
    "\n",
    "    # Transform data to be suitable with the learned model\n",
    "    x, _, _ = transform(torch.from_numpy(x), info['mu_x'], info['sigma_x'], method='standardize')\n",
    "    # make sure that x have a batch dimension\n",
    "    x = x.view(1,-1) if len(x.shape)==1 else x\n",
    "    \n",
    "    # Sample from the model (aka the simulator)\n",
    "    model.eval()\n",
    "    with torch.no_grad(), gpytorch.settings.fast_pred_var():\n",
    "        pred = model.likelihood(model(x))\n",
    "    \n",
    "    # Sample a single value for each x\n",
    "    new_samples = pred.to_data_independent_dist().sample([100])\n",
    "    \n",
    "    # Transform predictions back\n",
    "    new_samples = transform(new_samples, info['mu_y'], info['sigma_y'], method='standardize', inverse=True)\n",
    "    mean = transform(pred.mean, info['mu_y'], info['sigma_y'], method='standardize', inverse=True)\n",
    "    std = transform(pred.stddev, 0, info['sigma_y'], method='standardize', inverse=True)\n",
    "    \n",
    "    return new_samples, mean, std"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 482,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([100, 3])\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<matplotlib.collections.PolyCollection at 0x148729b50>"
      ]
     },
     "execution_count": 482,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 1656x360 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "ns, mu, sigma = auto_mpg_simulator(x_train[:3,:].clone().detach().numpy())\n",
    "print(ns.shape)\n",
    "f = plt.figure(figsize=(23,5))\n",
    "\n",
    "for i in range(ns.shape[0]):\n",
    "    plt.plot(np.arange(ns.shape[1]), ns[i,:], 'b.', alpha=0.2)\n",
    "plt.plot(np.arange(ns.shape[1]), mu, 'ro')\n",
    "plt.fill_between(np.arange(ns.shape[1]), mu-2*sigma, mu+2*sigma, alpha=0.5)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 481,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[8.0000e+00, 3.0700e+02, 1.3000e+02, 3.5040e+03, 1.2000e+01, 7.0000e+01,\n",
       "         1.0000e+00],\n",
       "        [8.0000e+00, 3.5000e+02, 1.6500e+02, 3.6930e+03, 1.1500e+01, 7.0000e+01,\n",
       "         1.0000e+00],\n",
       "        [8.0000e+00, 3.1800e+02, 1.5000e+02, 3.4360e+03, 1.1000e+01, 7.0000e+01,\n",
       "         1.0000e+00]])"
      ]
     },
     "execution_count": 481,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "x_train[:3,:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 483,
   "metadata": {},
   "outputs": [],
   "source": [
    "args.simulator = 'autompg'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 492,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([17.6795, 17.6795, 15.1190, 16.6906])"
      ]
     },
     "execution_count": 492,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from utils.active_learning import oracle_simulator\n",
    "ix = [0, 0, 1, 2]\n",
    "\n",
    "oracle_simulator(args, x_train[ix, :], noise=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 494,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([17.6795, 17.6795, 15.1190, 16.6906])\n",
      "tensor([17.6795, 15.1190, 16.6906])\n"
     ]
    }
   ],
   "source": [
    "print(oracle_simulator(args, x_train[ix, :], noise=False))\n",
    "print(mu)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 496,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.7089, 0.7089, 0.5236, 0.9480])\n",
      "tensor([0.7089, 0.5236, 0.9480])\n"
     ]
    }
   ],
   "source": [
    "print(oracle_simulator(args, x_train[ix, :], get_sigma=True))\n",
    "print(sigma)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python NOSTROMO",
   "language": "python",
   "name": "nostromo"
  },
  "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.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
