{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "from sklearn import svm, linear_model\n",
    "from sklearn.model_selection import GridSearchCV\n",
    "import pandas as pd\n",
    "import ast\n",
    "import pickle\n",
    "import stats"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Requirement already satisfied: stats in /venv/lib/python3.6/site-packages (0.1.2a0)\n",
      "\u001B[33mWARNING: You are using pip version 20.2.2; however, version 21.2.4 is available.\n",
      "You should consider upgrading via the '/venv/bin/python3 -m pip install --upgrade pip' command.\u001B[0m\n",
      "Collecting descartes\n",
      "  Using cached descartes-1.1.0-py3-none-any.whl (5.8 kB)\n",
      "Requirement already satisfied: matplotlib in /venv/lib/python3.6/site-packages (from descartes) (3.3.1)\n",
      "Requirement already satisfied: cycler>=0.10 in /venv/lib/python3.6/site-packages (from matplotlib->descartes) (0.10.0)\n",
      "Requirement already satisfied: pillow>=6.2.0 in /venv/lib/python3.6/site-packages (from matplotlib->descartes) (7.2.0)\n",
      "Requirement already satisfied: pyparsing!=2.0.4,!=2.1.2,!=2.1.6,>=2.0.3 in /venv/lib/python3.6/site-packages (from matplotlib->descartes) (2.4.7)\n",
      "Requirement already satisfied: numpy>=1.15 in /venv/lib/python3.6/site-packages (from matplotlib->descartes) (1.19.1)\n",
      "Requirement already satisfied: python-dateutil>=2.1 in /venv/lib/python3.6/site-packages (from matplotlib->descartes) (2.8.1)\n",
      "Requirement already satisfied: kiwisolver>=1.0.1 in /venv/lib/python3.6/site-packages (from matplotlib->descartes) (1.2.0)\n",
      "Requirement already satisfied: certifi>=2020.06.20 in /venv/lib/python3.6/site-packages (from matplotlib->descartes) (2020.6.20)\n",
      "Requirement already satisfied: six in /venv/lib/python3.6/site-packages (from cycler>=0.10->matplotlib->descartes) (1.15.0)\n",
      "Installing collected packages: descartes\n",
      "Successfully installed descartes-1.1.0\n",
      "\u001B[33mWARNING: You are using pip version 20.2.2; however, version 21.2.4 is available.\n",
      "You should consider upgrading via the '/venv/bin/python3 -m pip install --upgrade pip' command.\u001B[0m\n",
      "Collecting alphashape\n",
      "  Using cached alphashape-1.3.1-py2.py3-none-any.whl (13 kB)\n",
      "Collecting networkx>=2.5\n",
      "  Using cached networkx-2.5.1-py3-none-any.whl (1.6 MB)\n",
      "Collecting click-log>=0.3.2\n",
      "  Using cached click_log-0.3.2-py2.py3-none-any.whl (4.6 kB)\n",
      "Requirement already satisfied: numpy>=1.8.0 in /venv/lib/python3.6/site-packages (from alphashape) (1.19.1)\n",
      "Collecting trimesh>=3.9.8\n",
      "  Downloading trimesh-3.9.32-py3-none-any.whl (637 kB)\n",
      "\u001B[K     |████████████████████████████████| 637 kB 10.7 MB/s eta 0:00:01\n",
      "\u001B[?25hCollecting Click>=6.0\n",
      "  Using cached click-8.0.1-py3-none-any.whl (97 kB)\n",
      "Collecting shapely>=1.4.0\n",
      "  Using cached Shapely-1.7.1-cp36-cp36m-manylinux1_x86_64.whl (1.0 MB)\n",
      "Requirement already satisfied: scipy>=1.0.0 in /venv/lib/python3.6/site-packages (from alphashape) (1.5.2)\n",
      "Collecting rtree>=0.9.7\n",
      "  Using cached Rtree-0.9.7-cp36-cp36m-manylinux2010_x86_64.whl (994 kB)\n",
      "Requirement already satisfied: decorator<5,>=4.3 in /venv/lib/python3.6/site-packages (from networkx>=2.5->alphashape) (4.4.2)\n",
      "Requirement already satisfied: setuptools in /venv/lib/python3.6/site-packages (from trimesh>=3.9.8->alphashape) (50.0.3)\n",
      "Requirement already satisfied: importlib-metadata; python_version < \"3.8\" in /venv/lib/python3.6/site-packages (from Click>=6.0->alphashape) (3.3.0)\n",
      "Requirement already satisfied: typing-extensions>=3.6.4; python_version < \"3.8\" in /venv/lib/python3.6/site-packages (from importlib-metadata; python_version < \"3.8\"->Click>=6.0->alphashape) (3.7.4.3)\n",
      "Requirement already satisfied: zipp>=0.5 in /venv/lib/python3.6/site-packages (from importlib-metadata; python_version < \"3.8\"->Click>=6.0->alphashape) (3.4.0)\n",
      "Installing collected packages: networkx, Click, click-log, trimesh, shapely, rtree, alphashape\n",
      "Successfully installed Click-8.0.1 alphashape-1.3.1 click-log-0.3.2 networkx-2.5.1 rtree-0.9.7 shapely-1.7.1 trimesh-3.9.32\n",
      "\u001B[33mWARNING: You are using pip version 20.2.2; however, version 21.2.4 is available.\n",
      "You should consider upgrading via the '/venv/bin/python3 -m pip install --upgrade pip' command.\u001B[0m\n"
     ]
    }
   ],
   "source": [
    "!pip install stats\n",
    "!pip install descartes\n",
    "!pip install alphashape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "class Predictor:\n",
    "\n",
    "    def train(self, examples, labels):\n",
    "\n",
    "        # Search for optimal regressor parameters\n",
    "        self.searcher.fit(examples, labels)\n",
    "\n",
    "        # Retrieve regressor trained with optimal parameters\n",
    "        self.regressor = self.searcher.best_estimator_\n",
    "\n",
    "\n",
    "    def predict(self, examples):\n",
    "\n",
    "        return self.regressor.predict(examples)\n",
    "\n",
    "    def predict_single(self, example):\n",
    "\n",
    "        return self.regressor.predict(example.reshape(1,-1))[0]\n",
    "\n",
    "\n",
    "class RidgePredictor(Predictor):\n",
    "\n",
    "    def __init__(self, alphas, max_iterations, verbose):\n",
    "\n",
    "        SEARCHER_VERBOSITY = 10\n",
    "\n",
    "        # Create regressor\n",
    "        self.regressor = linear_model.Ridge(max_iter = max_iterations)\n",
    "\n",
    "        # Create parameter searcher\n",
    "        search_parameters = {'alpha': alphas}\n",
    "        self.searcher = GridSearchCV(estimator = self.regressor, param_grid = search_parameters, n_jobs = -1,\n",
    "            scoring = 'neg_root_mean_squared_error', verbose = SEARCHER_VERBOSITY if (verbose) else 0)\n",
    "        \n",
    "class SNLatencyPredictor(RidgePredictor):\n",
    "\n",
    "    DEFAULT_ALPHAS = np.arange(0.5, 5.5, 0.5)\n",
    "    DEFAULT_MAX_ITERATIONS = 1000000\n",
    "\n",
    "    def __init__(self, alphas = DEFAULT_ALPHAS, max_iterations = DEFAULT_MAX_ITERATIONS, verbose = False):\n",
    "\n",
    "        super().__init__(alphas, max_iterations, verbose)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "class ParameterManager():\n",
    "    \n",
    "    def __init__(self, seed=None):\n",
    "        self.num_blocks = 12\n",
    "        self.num_stages = 5\n",
    "        self.num_w = 6\n",
    "        \n",
    "        self.d = [0, 1]\n",
    "        self.e = [0.2, 0.25]\n",
    "        self.w = [0, 1, 2]\n",
    "\n",
    "        if seed is not None:\n",
    "            self.set_seed(seed)\n",
    "\n",
    "    def set_seed(self, seed):\n",
    "        random.seed(seed)\n",
    "\n",
    "    def random_sample(self):\n",
    "        sample = {}\n",
    "        d_list = []\n",
    "        e_list = []\n",
    "        w_list = []\n",
    "        for i in range(self.num_stages):\n",
    "            d_list.append(random.choice(self.d))\n",
    "\n",
    "        for i in range(self.num_blocks):\n",
    "            e_list.append(random.choice(self.e))\n",
    "            \n",
    "        for i in range(self.num_w):\n",
    "            w_list.append(random.choice(self.w))\n",
    "\n",
    "        sample = {\n",
    "            'd': d_list,\n",
    "            'e': e_list,\n",
    "            'w': w_list,\n",
    "        }\n",
    "        return sample\n",
    "\n",
    "    def verify_sample(self, sample):\n",
    "        assert len(sample['d']) == self.num_stages\n",
    "        assert len(sample['e']) == self.num_blocks\n",
    "        assert len(sample['w']) == self.num_w\n",
    "\n",
    "    def _serialize_compact(self, sample):\n",
    "        sample_id = []\n",
    "        for key in ['k','e','d']:\n",
    "            if sample[key] == None:\n",
    "                s = 'None'\n",
    "            else:\n",
    "                s = ''.join([str(c) for c in sample[key]])\n",
    "            sample_id.append('{}:{}'.format(key, s))\n",
    "        \n",
    "        sample_id = '-'.join(sample_id)\n",
    "        return sample_id\n",
    "\n",
    "    def serialize_sample(self, sample, compact=False):\n",
    "        self.verify_sample(sample)\n",
    "        if compact:\n",
    "            return self._serialize_compact(sample)\n",
    "        else:\n",
    "            return json.dumps(sample)\n",
    "\n",
    "    # Helper for constructing the one-hot vectors.\n",
    "    def construct_maps(self, keys):\n",
    "        d = dict()\n",
    "        keys = list(set(keys))\n",
    "        for k in keys:\n",
    "            if k not in d:\n",
    "                d[k] = len(list(d.keys()))\n",
    "        return d\n",
    "\n",
    "    def bnas_spec2feats(self, d_list, e_list, w_list, mode='tensor'):\n",
    "\n",
    "        d_map = self.construct_maps(keys=(0, 1))\n",
    "        e_map = self.construct_maps(keys=(0.2, 0.25))\n",
    "        w_map = self.construct_maps(keys=(0, 1, 2))\n",
    "        \n",
    "        # convert to onehot\n",
    "        d_onehot = [0 for _ in range(10)]\n",
    "        e_onehot = [0 for _ in range(20)]\n",
    "        w_onehot = [0 for _ in range(18)]\n",
    "        \n",
    "        # For every 2 steps, only a single column can be '1'\n",
    "        for i in range(5):\n",
    "            start = i * 2\n",
    "            #if d_list[i] != 0:\n",
    "            d_onehot[start + d_map[d_list[i]]] = 1\n",
    "\n",
    "        # For every 2 steps, only a single column can be '1'\n",
    "        for i in range(10): \n",
    "            start = i * 2 \n",
    "            #if e_list[i] != 0:\n",
    "            e_onehot[start + e_map[e_list[i]]] = 1\n",
    "\n",
    "        # For every 3 steps, only a single column can be '1'\n",
    "        for i in range(6):\n",
    "            start = i * 3\n",
    "            #if w_list[i] != 0:\n",
    "            w_onehot[start + w_map[w_list[i]]] = 1\n",
    "\n",
    "        if mode == 'tensor':   \n",
    "            return np.array(d_onehot + e_onehot + w_onehot)\n",
    "        else:\n",
    "            return np.array(d_onehot + e_onehot + w_onehot)\n",
    "        \n",
    "        \n",
    "    def csv_to_onehot(self, filepath, col_list=['build', 'top1', 'top5', 'GFLOPs', 'parameters'], features='build', target='top1'):\n",
    "\n",
    "        df = pd.read_csv(filepath, names=col_list)\n",
    "\n",
    "        df = df[[features, target]]\n",
    "\n",
    "        one_hot_list = list()\n",
    "        for i in range(len(df)):\n",
    "            config = ast.literal_eval(df[features].iloc[i])\n",
    "            one_hot_list.append(self.bnas_spec2feats(config['d'], config['e'], config['w']))\n",
    "\n",
    "        df = df.join(pd.DataFrame.from_dict(one_hot_list))\n",
    "\n",
    "        df.drop(columns=[features], inplace = True)\n",
    "\n",
    "        # Scale to 0-1 instead of 0-100\n",
    "        #df['top1'] = df['top1']*0.01\n",
    "\n",
    "        return np.array(df.drop(columns=[target])), np.array(df[target])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "manager = ParameterManager()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "features, labels = manager.csv_to_onehot('../data/bnas_latency_clx8280_c112_b8.csv', col_list=['build', 'latency', 'latency_stdev'], features='build', target='latency')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 113,
   "metadata": {},
   "outputs": [],
   "source": [
    "pred = SNLatencyPredictor()\n",
    "pred.train(features[:500], labels[:500])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 114,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "114.6405568425861"
      ]
     },
     "execution_count": 114,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "pred.predict_single(manager.bnas_spec2feats([0,0,1,0,1], \n",
    "                                            [0.2, 0.25, 0.2, 0.25, 0.2, 0.25, 0.2, 0.25, 0.2, 0.25], \n",
    "                                            [0,1,2,0,1,2]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 115,
   "metadata": {},
   "outputs": [],
   "source": [
    "all_targets = labels[:]\n",
    "all_preds = pred.predict(features[:])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "SpearmanrResult(correlation=0.9619915339915339, pvalue=0.0)"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from scipy import stats\n",
    "stats.spearmanr(all_targets, all_preds)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 117,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "5.623662139666798"
      ]
     },
     "execution_count": 117,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from sklearn.metrics import mean_squared_error\n",
    "mean_squared_error(all_targets, all_preds, squared=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 118,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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DUep0bF8+XKam5upq6vj7rvvRkSqHZJS25zhtkEMbKQeFSKyELixYtM84H+AeuDDbJje44vGmDvGIga1/XnllVc48sgjOfTQQ7nhhhs0OSi1hZzhPGmo5UdF5Kdbc3JjzEvGmL2NMXsD+2HbNW4Jdn+/f58mBzVcL774IocddhiFQoGLLrqo2uEotU0bVoIAThti+3tHKxDsWhOvDtJjSqlhefbZZznmmGPwfZ/77ruPvfbaq9ohKbVN22QVk4gcHNx1ROTNQGVZfQGQHcVYTmPjcRUfFZEzgP8AnzbGdI3iudR2xvM83vve9xIOh1m8eDELFy6sdkhKbfPEmKEGSIOI+MFdw8bJwQCrgS8ZY3611UGIRIFWYDdjzFoRaQHag/N8DZhqjDlnkNedRzBh4KxZs/Z7+eWXtziGUqlENBrd4tePFY1r+J555hnC4TC77rprtUN5g4l4vUDjGqntMa54PP6YMWbRYPs2WYIwxjgAIvJk0E4wVo4HHjfGrA3Ou34acRG5Grh9iPiuAq4CWLRokYnFtq674ta+fqxoXEP797//zT333MPFF1/MokWLKBaLEyKuwWhcI6NxjcxYxDWsNogxTg5gx1Ssr14SkcqJ998JPDvG51fboAceeIBjjjmGX//61/T29lY7HKW2O8PtxfQ3ETlywLajROTOrQ1ARGqAtwE3V2z+jog8IyJPA0cAn9za86jty7333svxxx/PjBkzuP/++6mr0/UalBptwx0oty/wwIBtD7DxGIYtYozJAk0Dtn1wa4+rtl933XUXp5xyCvPnz+eee+6hpaWl2iEptV0abjdXH7sudaUwGzdcKzUuurq62G233fj73/+uyUGpMTTcBPEY8LEB2z4KPD664Sg1tLa2NgBOO+00HnroIZqbdUpupcbScBPE54GLRORhEblORB7Crk39mbELTakNbrzxRubOncv9998PQDi8NRMRK6WGY7i9mJ4G3gT8EegFbgLeZIx5agxjUwqA6667jve///3st99+7LvvvtUOR6kdxrB/hhlj1gDf7X8sIruJyBeMMR8fk8iUAq655ho+9KEPccQRR3DrrbdSU1NT7ZCU2mEMt4oJABGJicgZIvJP4Bls7yalxsQ///lPzj33XI455hhuv/12TQ5KjbPhLhj0JuB84HQgiU0sxxlj/jaGsakd3MEHH8yVV17JmWeeSTwer3Y4Su1wNlmCEJEPisiD2JHMhwGXANOBTuDJsQ5O7Zh+8pOf8MorryAinH/++ZoclKqSzVUx/QrYBTgxWJfhR8aYznGIS+2gLrvsMv77v/+bH//4x9UORakd3uYSxJeBPuBPInKLiLxdREbUbqHUcBhj+MpXvsLFF1/MBz/4Qb73ve9VOySldnib/LI3xlyGXQb0FOzU2zcBq7BLgk4b49jUDsIYwxe+8AUuvfRSzjnnHH75y1/qOAelJoDNlgaMdacx5l3AbOAnwBrgURH5/VgHqLZ/xWKRBx98kAsuuICrr76aUChU7ZCUUoxgHASAMWY18DUR+Tp2DYfzxiQqtUPwfZ9isUgikeDuu+8mkUggotN7KTVRbFF7QlCquMMYc8oox6N2EL7vc/7553PCCSdQKpVIJpOaHJSaYLTBWY07z/M4++yz+fnPf86hhx5KJDJwomCl1ESgLYFqXLmuyxlnnMENN9zApZdeype//OVqh6SUGoImCDWuPv7xj3PDDTfwrW99i89//vPVDkcptQmaINS4+tjHPsYee+zBBRdcUO1QlFKboW0Qaszl83l+/vOfY4xh11131eSg1DZCE4QaU7lcjne84x2cd955PProo9UORyk1AlrFpMZMX18fJ510Eg8++CDXXnstBxxwQLVDUkqNgCYINSZ6eno44YQTePjhh7n++ut53/veV+2QlFIjpAlCjYlHH32UJ554ghtvvJF3v/vd1Q5HKbUFNEGoUeV5HqFQiKOPPprXX3+dlpaWaoeklNpC2kitRk1bWxsHHHAAN910E4AmB6W2cVqCUKNi9erVHHXUUSxdupS6urpqh6OUGgWaINRWW7lyJUceeSStra3ceeedHHbYYdUOSSk1CjRBqK3S1dXFYYcdxrp167jrrrs45JBDqh2SUmqUTIgEISJLgQzgAa4xZpGINAI3AnOApcB7jTFd1YpRDa6+vp4PfOADnHTSSTrOQantzERqpD7CGLO3MWZR8Pgi4F5jzALg3uCxmiBeeeUVnn/+eUSESy+9VJODUtuhiZQgBjoZ+FVw/1fYdbHVBPD888/ztre9jfe97334vl/tcJRSY2SiJAgD/E1EHhOR/mVMW4IlTsGuga19JieAp59+msMPPxyAG264AceZKB8hpdRomxBtEMChxphVIjIZuFtEXqzcaYwxImIGvihIJucB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\n",
      "text/plain": [
       "<Figure size 432x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(figsize=(6,6))\n",
    "\n",
    "ax.scatter(all_targets, all_preds, marker='o', alpha=0.2, \n",
    "           label='DyNAS-T Predictor (1000 points)', color='#4E79A7')\n",
    "\n",
    "\n",
    "\n",
    "ax.plot([0, 200],[0, 200], color='black', linestyle='--')\n",
    "\n",
    "ax.grid(True, alpha=0.1)\n",
    "#ax.set_xlim(70,82)\n",
    "#ax.set_ylim(70,82)\n",
    "\n",
    "ax.set_xlabel('Predicted Top-1 (%)', fontsize=13)\n",
    "ax.set_ylabel('Actual Top-1 (%)', fontsize=13)\n",
    "ax.set_title('Linear Ridge Latency Predictor Performance \\nBootstrapNAS Test (CLX8280, b8, c=112)')\n",
    "#ax.legend(fancybox=True, fontsize=11, framealpha=1, borderpad=0.2, loc='upper left')\n",
    "\n",
    "plt.show();\n",
    "#plt.savefig('acc_pred_compare.png', dpi=150);"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "pred = SNLatencyPredictor()\n",
    "pred.train(features, labels)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "SpearmanrResult(correlation=0.9619915339915339, pvalue=0.0)"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "all_targets = labels[:]\n",
    "all_preds = pred.predict(features[:])\n",
    "stats.spearmanr(all_targets, all_preds)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 126,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "1000"
      ]
     },
     "execution_count": 126,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(features)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 121,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x7ff8b32acf28>]"
      ]
     },
     "execution_count": 121,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "rmse_list = list()\n",
    "spear_list = list()\n",
    "\n",
    "for i in [20, 50, 100, 200, 300, 400, 500]:\n",
    "    pred = SNLatencyPredictor()\n",
    "    pred.train(features[:i], labels[:i])\n",
    "    \n",
    "    all_targets = labels[500:]\n",
    "    all_preds = pred.predict(features[500:])\n",
    "    \n",
    "    r, _ = stats.spearmanr(all_targets, all_preds)\n",
    "    rmse = mean_squared_error(all_targets, all_preds, squared=False)\n",
    "    \n",
    "    spear_list.append(r)\n",
    "    rmse_list.append(rmse)\n",
    "    \n",
    "    \n",
    "fig, ax = plt.subplots(figsize=(6,6))\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "ax.plot(spear_list, color='black', linestyle='--')\n",
    "ax.plot(rmse_list, color='red', linestyle='--')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Save Predictor"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "##########################\n",
    "# SAVE-LOAD using pickle #\n",
    "##########################\n",
    "import pickle\n",
    "\n",
    "# save\n",
    "#with open('bnas_latency_pred_clx8280_c112.pkl','wb') as f:\n",
    "#    pickle.dump(pred, f)\n",
    "\n",
    "# load\n",
    "with open('bnas_latency_pred_clx8280_c112.pkl', 'rb') as f:\n",
    "    pred = pickle.load(f)\n",
    "\n",
    "#all_targets = labels[900:]\n",
    "#all_preds = pred_load.predict(features[900:])\n",
    "#stats.spearmanr(all_targets, all_preds)"
   ]
  }
 ],
 "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.6.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}