{
 "cells": [
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Mainfold Analysis with open t-SNE"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Prepare dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 62,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import pandas as pd \n",
    "import pickle\n",
    "import sys\n",
    "sys.path.insert(0, '../')\n",
    "sys.path.insert(0, '../../')\n",
    "def load_data(path=\"\"):\n",
    "    f = open(path, \"rb\")\n",
    "    data = pickle.load(f)\n",
    "    dataset = pd.DataFrame(data)\n",
    "    \n",
    "    return dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 63,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "17659852572.597836"
      ]
     },
     "execution_count": 63,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "nasbench101_graph_vec=load_data('../experiments/nasbench101/labelled_dataset.pkl')\n",
    "\n",
    "max_zc_score = max(nasbench101_graph_vec['zc_score'])\n",
    "max_zc_score"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 64,
   "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>adj_matrix</th>\n",
       "      <th>ops_features</th>\n",
       "      <th>zc_score</th>\n",
       "      <th>acc</th>\n",
       "      <th>flops</th>\n",
       "      <th>params</th>\n",
       "      <th>model_performance</th>\n",
       "      <th>log_zc_score</th>\n",
       "      <th>norm_zc_score</th>\n",
       "      <th>log_flops</th>\n",
       "      <th>norm_flops</th>\n",
       "      <th>norm_params</th>\n",
       "      <th>conditions</th>\n",
       "      <th>clusters</th>\n",
       "      <th>param</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....</td>\n",
       "      <td>[[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [0, 1...</td>\n",
       "      <td>1.086526e+09</td>\n",
       "      <td>0.100060</td>\n",
       "      <td>1.136208e+09</td>\n",
       "      <td>3667594</td>\n",
       "      <td>lower_bad(0-91)</td>\n",
       "      <td>15.369073</td>\n",
       "      <td>6.152523</td>\n",
       "      <td>20.850962</td>\n",
       "      <td>7.584185</td>\n",
       "      <td>7.338225</td>\n",
       "      <td>[7.584185498320549, 6.152522737801687, 7.33822...</td>\n",
       "      <td>0</td>\n",
       "      <td>3667630.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....</td>\n",
       "      <td>[[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...</td>\n",
       "      <td>1.142529e+09</td>\n",
       "      <td>0.913862</td>\n",
       "      <td>1.059465e+09</td>\n",
       "      <td>3553034</td>\n",
       "      <td>bad(91-92)</td>\n",
       "      <td>15.288905</td>\n",
       "      <td>6.469639</td>\n",
       "      <td>20.781030</td>\n",
       "      <td>7.071928</td>\n",
       "      <td>7.109010</td>\n",
       "      <td>[7.071928416816488, 6.469639330005084, 7.10900...</td>\n",
       "      <td>0</td>\n",
       "      <td>3553097.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....</td>\n",
       "      <td>[[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [0, 1...</td>\n",
       "      <td>4.681139e+08</td>\n",
       "      <td>0.854467</td>\n",
       "      <td>5.142057e+08</td>\n",
       "      <td>1664778</td>\n",
       "      <td>lower_bad(0-91)</td>\n",
       "      <td>14.525205</td>\n",
       "      <td>2.650724</td>\n",
       "      <td>20.058134</td>\n",
       "      <td>3.432322</td>\n",
       "      <td>3.330934</td>\n",
       "      <td>[3.4323221673597875, 2.650723959351599, 3.3309...</td>\n",
       "      <td>0</td>\n",
       "      <td>1664823.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....</td>\n",
       "      <td>[[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [0, 1...</td>\n",
       "      <td>5.178305e+09</td>\n",
       "      <td>0.905649</td>\n",
       "      <td>4.740295e+09</td>\n",
       "      <td>15037834</td>\n",
       "      <td>lower_bad(0-91)</td>\n",
       "      <td>16.795438</td>\n",
       "      <td>29.322469</td>\n",
       "      <td>22.279365</td>\n",
       "      <td>31.641458</td>\n",
       "      <td>30.088118</td>\n",
       "      <td>[31.641457864020282, 29.322468887313814, 30.08...</td>\n",
       "      <td>0</td>\n",
       "      <td>15037879.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....</td>\n",
       "      <td>[[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [0, 0...</td>\n",
       "      <td>9.629810e+08</td>\n",
       "      <td>0.872095</td>\n",
       "      <td>9.057178e+08</td>\n",
       "      <td>3031562</td>\n",
       "      <td>lower_bad(0-91)</td>\n",
       "      <td>15.145566</td>\n",
       "      <td>5.452939</td>\n",
       "      <td>20.624238</td>\n",
       "      <td>6.045664</td>\n",
       "      <td>6.065634</td>\n",
       "      <td>[6.045664443111509, 5.45293883822498, 6.065633...</td>\n",
       "      <td>0</td>\n",
       "      <td>3031625.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>...</th>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>7659</th>\n",
       "      <td>[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....</td>\n",
       "      <td>[[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...</td>\n",
       "      <td>1.002757e+10</td>\n",
       "      <td>0.933894</td>\n",
       "      <td>9.266014e+09</td>\n",
       "      <td>30515338</td>\n",
       "      <td>higher_medium(93-94)</td>\n",
       "      <td>17.504191</td>\n",
       "      <td>56.781753</td>\n",
       "      <td>22.949619</td>\n",
       "      <td>61.850627</td>\n",
       "      <td>61.055941</td>\n",
       "      <td>[61.850627187185104, 56.78175301192626, 61.055...</td>\n",
       "      <td>2</td>\n",
       "      <td>30515374.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>7660</th>\n",
       "      <td>[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....</td>\n",
       "      <td>[[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...</td>\n",
       "      <td>8.161503e+09</td>\n",
       "      <td>0.910958</td>\n",
       "      <td>6.643264e+09</td>\n",
       "      <td>22585482</td>\n",
       "      <td>bad(91-92)</td>\n",
       "      <td>17.205011</td>\n",
       "      <td>46.215013</td>\n",
       "      <td>22.616869</td>\n",
       "      <td>44.343771</td>\n",
       "      <td>45.189663</td>\n",
       "      <td>[44.343771165159005, 46.21501345907037, 45.189...</td>\n",
       "      <td>1</td>\n",
       "      <td>22585545.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>7661</th>\n",
       "      <td>[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....</td>\n",
       "      <td>[[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [0, 1...</td>\n",
       "      <td>6.623202e+09</td>\n",
       "      <td>0.897135</td>\n",
       "      <td>6.306343e+09</td>\n",
       "      <td>21384074</td>\n",
       "      <td>lower_bad(0-91)</td>\n",
       "      <td>17.147826</td>\n",
       "      <td>37.504291</td>\n",
       "      <td>22.564822</td>\n",
       "      <td>42.094827</td>\n",
       "      <td>42.785853</td>\n",
       "      <td>[42.094826967027174, 37.50429082203488, 42.785...</td>\n",
       "      <td>1</td>\n",
       "      <td>21384128.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>7662</th>\n",
       "      <td>[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....</td>\n",
       "      <td>[[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...</td>\n",
       "      <td>8.172993e+09</td>\n",
       "      <td>0.916566</td>\n",
       "      <td>6.609709e+09</td>\n",
       "      <td>22421642</td>\n",
       "      <td>bad(91-92)</td>\n",
       "      <td>17.199606</td>\n",
       "      <td>46.280074</td>\n",
       "      <td>22.611805</td>\n",
       "      <td>44.119795</td>\n",
       "      <td>44.861848</td>\n",
       "      <td>[44.119795395961674, 46.280074363995624, 44.86...</td>\n",
       "      <td>1</td>\n",
       "      <td>22421687.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>7663</th>\n",
       "      <td>[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....</td>\n",
       "      <td>[[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...</td>\n",
       "      <td>1.054518e+10</td>\n",
       "      <td>0.908253</td>\n",
       "      <td>9.299569e+09</td>\n",
       "      <td>31552906</td>\n",
       "      <td>lower_bad(0-91)</td>\n",
       "      <td>17.542746</td>\n",
       "      <td>59.712758</td>\n",
       "      <td>22.953234</td>\n",
       "      <td>62.074603</td>\n",
       "      <td>63.131936</td>\n",
       "      <td>[62.07460295638244, 59.71275779784698, 63.1319...</td>\n",
       "      <td>2</td>\n",
       "      <td>31552951.0</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "<p>7664 rows × 15 columns</p>\n",
       "</div>"
      ],
      "text/plain": [
       "                                             adj_matrix  \\\n",
       "0     [[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....   \n",
       "1     [[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....   \n",
       "2     [[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....   \n",
       "3     [[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....   \n",
       "4     [[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....   \n",
       "...                                                 ...   \n",
       "7659  [[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....   \n",
       "7660  [[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....   \n",
       "7661  [[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....   \n",
       "7662  [[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....   \n",
       "7663  [[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....   \n",
       "\n",
       "                                           ops_features      zc_score  \\\n",
       "0     [[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [0, 1...  1.086526e+09   \n",
       "1     [[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...  1.142529e+09   \n",
       "2     [[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [0, 1...  4.681139e+08   \n",
       "3     [[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [0, 1...  5.178305e+09   \n",
       "4     [[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [0, 0...  9.629810e+08   \n",
       "...                                                 ...           ...   \n",
       "7659  [[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...  1.002757e+10   \n",
       "7660  [[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...  8.161503e+09   \n",
       "7661  [[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [0, 1...  6.623202e+09   \n",
       "7662  [[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...  8.172993e+09   \n",
       "7663  [[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...  1.054518e+10   \n",
       "\n",
       "           acc         flops    params     model_performance  log_zc_score  \\\n",
       "0     0.100060  1.136208e+09   3667594       lower_bad(0-91)     15.369073   \n",
       "1     0.913862  1.059465e+09   3553034            bad(91-92)     15.288905   \n",
       "2     0.854467  5.142057e+08   1664778       lower_bad(0-91)     14.525205   \n",
       "3     0.905649  4.740295e+09  15037834       lower_bad(0-91)     16.795438   \n",
       "4     0.872095  9.057178e+08   3031562       lower_bad(0-91)     15.145566   \n",
       "...        ...           ...       ...                   ...           ...   \n",
       "7659  0.933894  9.266014e+09  30515338  higher_medium(93-94)     17.504191   \n",
       "7660  0.910958  6.643264e+09  22585482            bad(91-92)     17.205011   \n",
       "7661  0.897135  6.306343e+09  21384074       lower_bad(0-91)     17.147826   \n",
       "7662  0.916566  6.609709e+09  22421642            bad(91-92)     17.199606   \n",
       "7663  0.908253  9.299569e+09  31552906       lower_bad(0-91)     17.542746   \n",
       "\n",
       "      norm_zc_score  log_flops  norm_flops  norm_params  \\\n",
       "0          6.152523  20.850962    7.584185     7.338225   \n",
       "1          6.469639  20.781030    7.071928     7.109010   \n",
       "2          2.650724  20.058134    3.432322     3.330934   \n",
       "3         29.322469  22.279365   31.641458    30.088118   \n",
       "4          5.452939  20.624238    6.045664     6.065634   \n",
       "...             ...        ...         ...          ...   \n",
       "7659      56.781753  22.949619   61.850627    61.055941   \n",
       "7660      46.215013  22.616869   44.343771    45.189663   \n",
       "7661      37.504291  22.564822   42.094827    42.785853   \n",
       "7662      46.280074  22.611805   44.119795    44.861848   \n",
       "7663      59.712758  22.953234   62.074603    63.131936   \n",
       "\n",
       "                                             conditions  clusters       param  \n",
       "0     [7.584185498320549, 6.152522737801687, 7.33822...         0   3667630.0  \n",
       "1     [7.071928416816488, 6.469639330005084, 7.10900...         0   3553097.0  \n",
       "2     [3.4323221673597875, 2.650723959351599, 3.3309...         0   1664823.0  \n",
       "3     [31.641457864020282, 29.322468887313814, 30.08...         0  15037879.0  \n",
       "4     [6.045664443111509, 5.45293883822498, 6.065633...         0   3031625.0  \n",
       "...                                                 ...       ...         ...  \n",
       "7659  [61.850627187185104, 56.78175301192626, 61.055...         2  30515374.0  \n",
       "7660  [44.343771165159005, 46.21501345907037, 45.189...         1  22585545.0  \n",
       "7661  [42.094826967027174, 37.50429082203488, 42.785...         1  21384128.0  \n",
       "7662  [44.119795395961674, 46.280074363995624, 44.86...         1  22421687.0  \n",
       "7663  [62.07460295638244, 59.71275779784698, 63.1319...         2  31552951.0  \n",
       "\n",
       "[7664 rows x 15 columns]"
      ]
     },
     "execution_count": 64,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "nasbench101_graph_vec=load_data('../experiments/nasbench101/test_dataset.pkl')\n",
    "nasbench101_graph_vec\n"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Load VGAE\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 65,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/u1650783/.conda/envs/ss_gcn/lib/python3.7/site-packages/torch_geometric/deprecation.py:12: UserWarning: 'nn.glob.Set2Set' is deprecated, use 'nn.aggr.Set2Set' instead\n",
      "  warnings.warn(out)\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "VGAE(\n",
       "  (gc): ModuleList(\n",
       "    (0): GraphConvolution (6 -> 64)\n",
       "    (1): GraphConvolution (64 -> 64)\n",
       "    (2): GraphConvolution (64 -> 64)\n",
       "    (3): GraphConvolution (64 -> 64)\n",
       "  )\n",
       "  (bn): ModuleList(\n",
       "    (0): LayerNorm((64,), eps=1e-05, elementwise_affine=True)\n",
       "    (1): LayerNorm((64,), eps=1e-05, elementwise_affine=True)\n",
       "    (2): LayerNorm((64,), eps=1e-05, elementwise_affine=True)\n",
       "    (3): LayerNorm((64,), eps=1e-05, elementwise_affine=True)\n",
       "  )\n",
       "  (dropout): ModuleList(\n",
       "    (0): Dropout(p=0, inplace=False)\n",
       "    (1): Dropout(p=0, inplace=False)\n",
       "    (2): Dropout(p=0, inplace=False)\n",
       "    (3): Dropout(p=0, inplace=False)\n",
       "  )\n",
       "  (relu): ModuleList(\n",
       "    (0): ReLU()\n",
       "    (1): ReLU()\n",
       "    (2): ReLU()\n",
       "    (3): ReLU()\n",
       "  )\n",
       "  (pooling): Set2Set(64, 128)\n",
       "  (fc1): Linear(in_features=512, out_features=128, bias=True)\n",
       "  (fc2): Linear(in_features=512, out_features=128, bias=True)\n",
       "  (decoder): Decoder(\n",
       "    (adj_decoder): Sequential(\n",
       "      (0): Linear(in_features=128, out_features=128, bias=True)\n",
       "      (1): ReLU()\n",
       "      (2): Linear(in_features=128, out_features=64, bias=True)\n",
       "    )\n",
       "    (ops_decoder): Sequential(\n",
       "      (0): Linear(in_features=128, out_features=128, bias=True)\n",
       "      (1): ReLU()\n",
       "      (2): Linear(in_features=128, out_features=48, bias=True)\n",
       "    )\n",
       "  )\n",
       "  (condition_regressor): Sequential(\n",
       "    (0): Linear(in_features=128, out_features=128, bias=True)\n",
       "    (1): ReLU()\n",
       "    (2): Linear(in_features=128, out_features=3, bias=True)\n",
       "    (3): ReLU()\n",
       "  )\n",
       ")"
      ]
     },
     "execution_count": 65,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from network_designer.models.vgae import VGAE\n",
    "import torch\n",
    "m1 = VGAE(num_features=6, num_layers=4, num_hidden=64, num_latent=128, num_node=8)\n",
    "m1 = m1.cuda()\n",
    "m1.load_state_dict(torch.load(\"../experiments/nasbench101/m1_dist_1.0_no_latency.pt\"))\n",
    "m1.eval().cuda()"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Load NF"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 66,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1\n",
      "1\n",
      "Training T : 0.5\n",
      "Number of trainable parameters of Point CNF: 1461505\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "GraphFlow(\n",
       "  (graph_cnf): SequenttialFlow(\n",
       "    (chain): ModuleList(\n",
       "      (0): MovingBatchNorm1d(128, eps=0.0001, decay=0.1, bn_lag=0, affine=True)\n",
       "      (1): CNF(\n",
       "        (odefunc): ODEfunc(\n",
       "          (diffeq): ODEnet(\n",
       "            (layers): ModuleList(\n",
       "              (0): ConcatSquashLinear(\n",
       "                (_layer): Linear(in_features=128, out_features=512, bias=True)\n",
       "                (_hyper_bias): Linear(in_features=2, out_features=512, bias=False)\n",
       "                (_hyper_gate): Linear(in_features=2, out_features=512, bias=True)\n",
       "              )\n",
       "              (1): ConcatSquashLinear(\n",
       "                (_layer): Linear(in_features=512, out_features=512, bias=True)\n",
       "                (_hyper_bias): Linear(in_features=2, out_features=512, bias=False)\n",
       "                (_hyper_gate): Linear(in_features=2, out_features=512, bias=True)\n",
       "              )\n",
       "              (2): ConcatSquashLinear(\n",
       "                (_layer): Linear(in_features=512, out_features=512, bias=True)\n",
       "                (_hyper_bias): Linear(in_features=2, out_features=512, bias=False)\n",
       "                (_hyper_gate): Linear(in_features=2, out_features=512, bias=True)\n",
       "              )\n",
       "              (3): ConcatSquashLinear(\n",
       "                (_layer): Linear(in_features=512, out_features=512, bias=True)\n",
       "                (_hyper_bias): Linear(in_features=2, out_features=512, bias=False)\n",
       "                (_hyper_gate): Linear(in_features=2, out_features=512, bias=True)\n",
       "              )\n",
       "              (4): ConcatSquashLinear(\n",
       "                (_layer): Linear(in_features=512, out_features=512, bias=True)\n",
       "                (_hyper_bias): Linear(in_features=2, out_features=512, bias=False)\n",
       "                (_hyper_gate): Linear(in_features=2, out_features=512, bias=True)\n",
       "              )\n",
       "              (5): ConcatSquashLinear(\n",
       "                (_layer): Linear(in_features=512, out_features=512, bias=True)\n",
       "                (_hyper_bias): Linear(in_features=2, out_features=512, bias=False)\n",
       "                (_hyper_gate): Linear(in_features=2, out_features=512, bias=True)\n",
       "              )\n",
       "              (6): ConcatSquashLinear(\n",
       "                (_layer): Linear(in_features=512, out_features=128, bias=True)\n",
       "                (_hyper_bias): Linear(in_features=2, out_features=128, bias=False)\n",
       "                (_hyper_gate): Linear(in_features=2, out_features=128, bias=True)\n",
       "              )\n",
       "            )\n",
       "            (activation_fns): ModuleList(\n",
       "              (0): Tanh()\n",
       "              (1): Tanh()\n",
       "              (2): Tanh()\n",
       "              (3): Tanh()\n",
       "              (4): Tanh()\n",
       "              (5): Tanh()\n",
       "            )\n",
       "          )\n",
       "        )\n",
       "      )\n",
       "      (2): MovingBatchNorm1d(128, eps=0.0001, decay=0.1, bn_lag=0, affine=True)\n",
       "    )\n",
       "  )\n",
       ")"
      ]
     },
     "execution_count": 66,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from network_designer.models.cnf_modules.network import GraphFlow\n",
    "rev = torch.load('../experiments/nasbench101/cnf_zc_score.pt')\n",
    "rev_model = GraphFlow(num_latent=128, num_node=8, num_cond=1)\n",
    "rev_model.load_state_dict(rev['model'].state_dict(), strict=False) \n",
    "rev_model = rev_model.cuda().double()\n",
    "rev_model.eval()"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Rank nasbench-201 models with gt-acc find best model as existing best reference"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 67,
   "metadata": {},
   "outputs": [],
   "source": [
    "nasbench101_graph_vec = nasbench101_graph_vec.sort_values(by='acc', ascending=False, na_position='first')\n",
    "context = nasbench101_graph_vec[0:1]['conditions'].values[0]\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 68,
   "metadata": {},
   "outputs": [],
   "source": [
    "context = np.array(context)\n",
    "context = np.round(context)\n",
    "context = context[1]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 69,
   "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>adj_matrix</th>\n",
       "      <th>ops_features</th>\n",
       "      <th>zc_score</th>\n",
       "      <th>acc</th>\n",
       "      <th>flops</th>\n",
       "      <th>params</th>\n",
       "      <th>model_performance</th>\n",
       "      <th>log_zc_score</th>\n",
       "      <th>norm_zc_score</th>\n",
       "      <th>log_flops</th>\n",
       "      <th>norm_flops</th>\n",
       "      <th>norm_params</th>\n",
       "      <th>conditions</th>\n",
       "      <th>clusters</th>\n",
       "      <th>param</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>6575</th>\n",
       "      <td>[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....</td>\n",
       "      <td>[[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...</td>\n",
       "      <td>1.213013e+10</td>\n",
       "      <td>0.943109</td>\n",
       "      <td>1.011589e+10</td>\n",
       "      <td>33300362</td>\n",
       "      <td>good(94-100)</td>\n",
       "      <td>17.593783</td>\n",
       "      <td>68.687596</td>\n",
       "      <td>23.037373</td>\n",
       "      <td>67.523513</td>\n",
       "      <td>66.628295</td>\n",
       "      <td>[67.52351346638636, 68.68759557043573, 66.6282...</td>\n",
       "      <td>2</td>\n",
       "      <td>33300380.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3065</th>\n",
       "      <td>[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....</td>\n",
       "      <td>[[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...</td>\n",
       "      <td>1.475727e+10</td>\n",
       "      <td>0.942208</td>\n",
       "      <td>1.229279e+10</td>\n",
       "      <td>40848010</td>\n",
       "      <td>good(94-100)</td>\n",
       "      <td>17.796716</td>\n",
       "      <td>83.563928</td>\n",
       "      <td>23.232279</td>\n",
       "      <td>82.054379</td>\n",
       "      <td>81.729840</td>\n",
       "      <td>[82.05437894573771, 83.56392829491213, 81.7298...</td>\n",
       "      <td>3</td>\n",
       "      <td>40848055.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4515</th>\n",
       "      <td>[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....</td>\n",
       "      <td>[[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...</td>\n",
       "      <td>1.186842e+10</td>\n",
       "      <td>0.942007</td>\n",
       "      <td>9.843321e+09</td>\n",
       "      <td>32426634</td>\n",
       "      <td>good(94-100)</td>\n",
       "      <td>17.562556</td>\n",
       "      <td>67.205642</td>\n",
       "      <td>23.010059</td>\n",
       "      <td>65.704148</td>\n",
       "      <td>64.880115</td>\n",
       "      <td>[65.70414779433223, 67.20564201932224, 64.8801...</td>\n",
       "      <td>2</td>\n",
       "      <td>32426661.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5660</th>\n",
       "      <td>[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....</td>\n",
       "      <td>[[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [0, 1...</td>\n",
       "      <td>1.220971e+10</td>\n",
       "      <td>0.941907</td>\n",
       "      <td>9.876875e+09</td>\n",
       "      <td>32590474</td>\n",
       "      <td>good(94-100)</td>\n",
       "      <td>17.573648</td>\n",
       "      <td>69.138232</td>\n",
       "      <td>23.013462</td>\n",
       "      <td>65.928124</td>\n",
       "      <td>65.207931</td>\n",
       "      <td>[65.92812356352957, 69.13823246158326, 65.2079...</td>\n",
       "      <td>2</td>\n",
       "      <td>32590519.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2617</th>\n",
       "      <td>[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....</td>\n",
       "      <td>[[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [0, 0...</td>\n",
       "      <td>1.085832e+10</td>\n",
       "      <td>0.941907</td>\n",
       "      <td>9.844697e+09</td>\n",
       "      <td>32426634</td>\n",
       "      <td>good(94-100)</td>\n",
       "      <td>17.564867</td>\n",
       "      <td>61.485894</td>\n",
       "      <td>23.010199</td>\n",
       "      <td>65.713334</td>\n",
       "      <td>64.880115</td>\n",
       "      <td>[65.71333430049071, 61.48589391360672, 64.8801...</td>\n",
       "      <td>2</td>\n",
       "      <td>32426670.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>...</th>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1468</th>\n",
       "      <td>[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....</td>\n",
       "      <td>[[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...</td>\n",
       "      <td>1.559692e+09</td>\n",
       "      <td>0.100060</td>\n",
       "      <td>1.509698e+09</td>\n",
       "      <td>5095946</td>\n",
       "      <td>lower_bad(0-91)</td>\n",
       "      <td>15.716015</td>\n",
       "      <td>8.831852</td>\n",
       "      <td>21.135175</td>\n",
       "      <td>10.077228</td>\n",
       "      <td>10.196111</td>\n",
       "      <td>[10.077228288663546, 8.831851846270958, 10.196...</td>\n",
       "      <td>0</td>\n",
       "      <td>5096009.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>7122</th>\n",
       "      <td>[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....</td>\n",
       "      <td>[[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...</td>\n",
       "      <td>1.005748e+10</td>\n",
       "      <td>0.100060</td>\n",
       "      <td>9.267390e+09</td>\n",
       "      <td>30515338</td>\n",
       "      <td>lower_bad(0-91)</td>\n",
       "      <td>17.498810</td>\n",
       "      <td>56.951105</td>\n",
       "      <td>22.949768</td>\n",
       "      <td>61.859814</td>\n",
       "      <td>61.055941</td>\n",
       "      <td>[61.85981369334359, 56.95110483018176, 61.0559...</td>\n",
       "      <td>2</td>\n",
       "      <td>30515383.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4691</th>\n",
       "      <td>[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....</td>\n",
       "      <td>[[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...</td>\n",
       "      <td>6.612427e+09</td>\n",
       "      <td>0.100060</td>\n",
       "      <td>6.001601e+09</td>\n",
       "      <td>20346506</td>\n",
       "      <td>lower_bad(0-91)</td>\n",
       "      <td>17.086654</td>\n",
       "      <td>37.443273</td>\n",
       "      <td>22.515292</td>\n",
       "      <td>40.060672</td>\n",
       "      <td>40.709858</td>\n",
       "      <td>[40.06067203193418, 37.443272984967976, 40.709...</td>\n",
       "      <td>1</td>\n",
       "      <td>20346560.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6307</th>\n",
       "      <td>[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....</td>\n",
       "      <td>[[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...</td>\n",
       "      <td>6.829834e+09</td>\n",
       "      <td>0.099960</td>\n",
       "      <td>6.306343e+09</td>\n",
       "      <td>21384074</td>\n",
       "      <td>lower_bad(0-91)</td>\n",
       "      <td>17.145326</td>\n",
       "      <td>38.674356</td>\n",
       "      <td>22.564822</td>\n",
       "      <td>42.094827</td>\n",
       "      <td>42.785853</td>\n",
       "      <td>[42.094826967027174, 38.67435571641105, 42.785...</td>\n",
       "      <td>1</td>\n",
       "      <td>21384128.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5050</th>\n",
       "      <td>[[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....</td>\n",
       "      <td>[[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...</td>\n",
       "      <td>6.326076e+09</td>\n",
       "      <td>0.099760</td>\n",
       "      <td>5.729036e+09</td>\n",
       "      <td>19472778</td>\n",
       "      <td>lower_bad(0-91)</td>\n",
       "      <td>17.053092</td>\n",
       "      <td>35.821794</td>\n",
       "      <td>22.468813</td>\n",
       "      <td>38.241306</td>\n",
       "      <td>38.961678</td>\n",
       "      <td>[38.24130635988004, 35.82179395265913, 38.9616...</td>\n",
       "      <td>1</td>\n",
       "      <td>19472841.0</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "<p>7664 rows × 15 columns</p>\n",
       "</div>"
      ],
      "text/plain": [
       "                                             adj_matrix  \\\n",
       "6575  [[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....   \n",
       "3065  [[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....   \n",
       "4515  [[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....   \n",
       "5660  [[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....   \n",
       "2617  [[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....   \n",
       "...                                                 ...   \n",
       "1468  [[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....   \n",
       "7122  [[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....   \n",
       "4691  [[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....   \n",
       "6307  [[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....   \n",
       "5050  [[1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0], [0....   \n",
       "\n",
       "                                           ops_features      zc_score  \\\n",
       "6575  [[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...  1.213013e+10   \n",
       "3065  [[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...  1.475727e+10   \n",
       "4515  [[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...  1.186842e+10   \n",
       "5660  [[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [0, 1...  1.220971e+10   \n",
       "2617  [[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [0, 0...  1.085832e+10   \n",
       "...                                                 ...           ...   \n",
       "1468  [[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...  1.559692e+09   \n",
       "7122  [[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...  1.005748e+10   \n",
       "4691  [[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...  6.612427e+09   \n",
       "6307  [[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...  6.829834e+09   \n",
       "5050  [[0, 0, 0, 0, 0, 1], [0, 0, 0, 1, 0, 0], [1, 0...  6.326076e+09   \n",
       "\n",
       "           acc         flops    params model_performance  log_zc_score  \\\n",
       "6575  0.943109  1.011589e+10  33300362      good(94-100)     17.593783   \n",
       "3065  0.942208  1.229279e+10  40848010      good(94-100)     17.796716   \n",
       "4515  0.942007  9.843321e+09  32426634      good(94-100)     17.562556   \n",
       "5660  0.941907  9.876875e+09  32590474      good(94-100)     17.573648   \n",
       "2617  0.941907  9.844697e+09  32426634      good(94-100)     17.564867   \n",
       "...        ...           ...       ...               ...           ...   \n",
       "1468  0.100060  1.509698e+09   5095946   lower_bad(0-91)     15.716015   \n",
       "7122  0.100060  9.267390e+09  30515338   lower_bad(0-91)     17.498810   \n",
       "4691  0.100060  6.001601e+09  20346506   lower_bad(0-91)     17.086654   \n",
       "6307  0.099960  6.306343e+09  21384074   lower_bad(0-91)     17.145326   \n",
       "5050  0.099760  5.729036e+09  19472778   lower_bad(0-91)     17.053092   \n",
       "\n",
       "      norm_zc_score  log_flops  norm_flops  norm_params  \\\n",
       "6575      68.687596  23.037373   67.523513    66.628295   \n",
       "3065      83.563928  23.232279   82.054379    81.729840   \n",
       "4515      67.205642  23.010059   65.704148    64.880115   \n",
       "5660      69.138232  23.013462   65.928124    65.207931   \n",
       "2617      61.485894  23.010199   65.713334    64.880115   \n",
       "...             ...        ...         ...          ...   \n",
       "1468       8.831852  21.135175   10.077228    10.196111   \n",
       "7122      56.951105  22.949768   61.859814    61.055941   \n",
       "4691      37.443273  22.515292   40.060672    40.709858   \n",
       "6307      38.674356  22.564822   42.094827    42.785853   \n",
       "5050      35.821794  22.468813   38.241306    38.961678   \n",
       "\n",
       "                                             conditions  clusters       param  \n",
       "6575  [67.52351346638636, 68.68759557043573, 66.6282...         2  33300380.0  \n",
       "3065  [82.05437894573771, 83.56392829491213, 81.7298...         3  40848055.0  \n",
       "4515  [65.70414779433223, 67.20564201932224, 64.8801...         2  32426661.0  \n",
       "5660  [65.92812356352957, 69.13823246158326, 65.2079...         2  32590519.0  \n",
       "2617  [65.71333430049071, 61.48589391360672, 64.8801...         2  32426670.0  \n",
       "...                                                 ...       ...         ...  \n",
       "1468  [10.077228288663546, 8.831851846270958, 10.196...         0   5096009.0  \n",
       "7122  [61.85981369334359, 56.95110483018176, 61.0559...         2  30515383.0  \n",
       "4691  [40.06067203193418, 37.443272984967976, 40.709...         1  20346560.0  \n",
       "6307  [42.094826967027174, 38.67435571641105, 42.785...         1  21384128.0  \n",
       "5050  [38.24130635988004, 35.82179395265913, 38.9616...         1  19472841.0  \n",
       "\n",
       "[7664 rows x 15 columns]"
      ]
     },
     "execution_count": 69,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "nasbench101_graph_vec"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Now, we want to test if we use unseen best model as reference, the normalize flow log-likely-hood will obey the distance manner."
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Sample a graph from normalize flow"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 70,
   "metadata": {},
   "outputs": [],
   "source": [
    "def sample_gaussian(size=(1,32*9)):\n",
    "    y = torch.randn(*size).double()\n",
    "    return y.cuda()\n",
    "\n",
    "def post_process_adj(adj_recon):\n",
    "    adj_matrix = adj_recon.detach().cpu().numpy()\n",
    "    adj_matrix = np.triu(adj_matrix, 1)[0][1:,1:]\n",
    "    col, row = adj_matrix.shape\n",
    "    \n",
    "    #print((adj_matrix >0.5).astype(int))\n",
    "    for i in range(1, col):\n",
    "        #from last col:\n",
    "        adj_matrix[:, col-i] = (adj_matrix[:, col-i] > 0.5).astype(int)\n",
    "        #print(adj_matrix)\n",
    "        #if we have incoming connection from node j, j should has at-least one incoming connection\n",
    "        \n",
    "        #but if a intermediate node has no output we dont help them for parent\n",
    "        if i==1 or np.sum((adj_matrix[col-i,:] >0.5).astype(int)) > 0:\n",
    "            for j in range(1, col-i):\n",
    "                # so if node has prodecessors node j, node j should has at-least one prodecessor\n",
    "                if adj_matrix[j, col-i] >0 and np.sum((adj_matrix[:,j] >0.5).astype(int)) == 0:\n",
    "                    select_parent = np.argmax(adj_matrix[:,j])\n",
    "                    #print('add new parent {} to node {} '.format(select_parent, j))\n",
    "                    adj_matrix[select_parent, j] = 1\n",
    "                \n",
    "                \n",
    "    adj_matrix[:, 0] = (adj_matrix[:, 0] > 0.5).astype(int) \n",
    "    \n",
    "    #print(adj_matrix)\n",
    "    #process loss-end\n",
    "    col_sums = list(np.sum(adj_matrix, axis=0))\n",
    "    row_sums = list(np.sum(adj_matrix, axis=1))\n",
    "    \n",
    "    need_fill_output = []\n",
    "    for i ,(col, row) in enumerate(zip(col_sums[:-1], row_sums[:-1])):\n",
    "        #if a node has input but no output connect it to output node\n",
    "        #print(i, col, row)\n",
    "        if col>0 and row == 0:\n",
    "            need_fill_output.append(i)\n",
    "    for o in need_fill_output:\n",
    "        adj_matrix[o, -1]=1\n",
    "    #print(adj_matrix) \n",
    "    return adj_matrix"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 71,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n"
     ]
    }
   ],
   "source": [
    "from network_designer.design_space.core.search_space import SearchSpace\n",
    "from torch.distributions import Normal\n",
    "import numpy as np\n",
    "import networkx as nx\n",
    "from network_designer.design_space.nasbench101.network import Network\n",
    "from network_designer.design_space.nasbench101.nasbench1_spec import ModelSpec\n",
    "from network_designer.dataloader import define_dataloader\n",
    "from network_designer.design_space.core.net2vec import get_tcet\n",
    "import torch\n",
    "\n",
    "ss = SearchSpace(5, 3)\n",
    "\n",
    "from alethiometer import get_cifar_dataloaders\n",
    "\n",
    "train_loader, test_loader = get_cifar_dataloaders(64, 64, 'cifar10', 1)\n",
    "def sample_graph_from_cnf(rev_model,m1, cond, t=0.3):\n",
    "    print(cond)\n",
    "    noise = Normal(0, 1).sample(sample_shape=torch.Size([1, 128])) * t\n",
    "    z =  noise.cuda()\n",
    "    z = z.cuda()\n",
    "\n",
    "    samples = rev_model.sample(z.cuda().double(),torch.tensor(cond).cuda().double().unsqueeze(0))\n",
    "    adj, features = m1._decoder(samples)\n",
    "    #print(adj)\n",
    "    adj=torch.nn.Sigmoid()(adj)\n",
    "    features = torch.nn.Sigmoid()(features)\n",
    "    #print(features)\n",
    "    adj_matrix = post_process_adj(adj)\n",
    "    if ss.check_validity_of_adjacency_matrix(adj_matrix):\n",
    "        # print('adj matrix')\n",
    "        # print(adj_matrix)\n",
    "        #ops_mask = ((features>=0.5).int()>0).sum(dim=-1)\n",
    "        ops_feature = torch.nn.functional.one_hot(torch.argmax(features, dim=2) , num_classes=6).cpu().numpy()[0,1:,:-1]\n",
    "        # print('ops features')\n",
    "        # print(ops_feature)\n",
    "    return adj_matrix, ops_feature\n",
    "\n",
    "\n",
    "OPS=['conv3x3-bn-relu', 'conv1x1-bn-relu', 'maxpool3x3','input', 'output']\n",
    "dataloader, n_classes = define_dataloader(dataset='cifar10', dataset_path='../../data')\n",
    "\n",
    "def remove_zero_rows_cols(array):\n",
    "    n = array.shape[0]  # Get the size of the square array\n",
    "    indices = np.arange(n)\n",
    "    zero_indices = indices[np.all(array == 0, axis=0) & np.all(array == 0, axis=1)]\n",
    "\n",
    "    # Remove the zero rows and columns\n",
    "    trimmed_array = np.delete(array, zero_indices, axis=0)\n",
    "    trimmed_array = np.delete(trimmed_array, zero_indices, axis=1)\n",
    "\n",
    "    return trimmed_array\n",
    "\n",
    "def get_model(adj, ops, shave=True):\n",
    "    o = ops.astype(int)\n",
    "    a = adj.astype(int)\n",
    "    a = np.triu(a, 1)\n",
    "\n",
    "    labels = []\n",
    "    for idx, ops_f in enumerate(list(o)):\n",
    "        if sum(ops_f) > 0:\n",
    "            if ((sum(a[:, idx]) >0) or (idx in [0, 6])):\n",
    "\n",
    "                labels.append(OPS[np.argmax(ops_f)])\n",
    "    \n",
    "    a = remove_zero_rows_cols(a)\n",
    "    #print(a)\n",
    "    spec = ModelSpec(a, labels)\n",
    "    model = Network(spec, num_labels=10, in_channels=3,\n",
    "            stem_out_channels=128, num_stacks=3,\n",
    "            num_modules_per_stack=3,\n",
    "            momentum=0.9, eps=1e-5, tf_like=False)\n",
    "    \n",
    "    hash = spec.hash_spec(OPS)\n",
    "    \n",
    "    return model.cuda(), hash\n",
    "\n",
    "def draw_graph(adj_matrix, ops_feature, save_path, name):\n",
    "    G = nx.from_numpy_matrix(adj_matrix, create_using=nx.DiGraph)\n",
    "    a = nx.nx_agraph.to_agraph(G)\n",
    "    labels = {}\n",
    "    for idx, ops_f in enumerate(list(ops_feature)):\n",
    "        labels[idx]=OPS[np.argmax(ops_f)]\n",
    "\n",
    "    node_colors = []\n",
    "    node_shapes = []\n",
    "    for i,node in enumerate(G.nodes()):\n",
    "        if labels[i] in ['input', 'output']:\n",
    "            node_colors.append(\"#80d189\")\n",
    "            node_shapes.append('s')\n",
    "        else:\n",
    "            node_colors.append(\"#ccbfbe\")\n",
    "            node_shapes.append('o')\n",
    "\n",
    "\n",
    "    import matplotlib.pyplot as plt \n",
    "    plt.figure(figsize=(8,8))\n",
    "    pos = nx.drawing.nx_agraph.graphviz_layout(G,prog='dot')\n",
    "    nx.draw_networkx(G, pos = pos, arrows=True, labels=labels, with_labels=True, node_color = node_colors, node_shape='s')\n",
    "    plt.savefig(\"{}/{}.png\",format(save_path, name))\n",
    "    plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 72,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "69.0\n",
      "48\n",
      "7460956180.0\n",
      "8590997194.642784\n",
      "69.0\n",
      "63\n",
      "9876875284.0\n",
      "11240506265.288622\n",
      "69.0\n",
      "65\n",
      "9876875284.0\n",
      "11581716973.190653\n",
      "69.0\n",
      "69.0\n",
      "61\n",
      "9604311060.0\n",
      "10896353790.114239\n",
      "69.0\n",
      "13\n",
      "2285447188.0\n",
      "2418014553.496148\n",
      "69.0\n",
      "77\n",
      "12292794388.0\n",
      "13606176137.723541\n",
      "69.0\n",
      "62\n",
      "9876875284.0\n",
      "10954539151.5572\n",
      "69.0\n",
      "67\n",
      "9876875284.0\n",
      "11909424329.39443\n",
      "69.0\n",
      "17\n",
      "2889426964.0\n",
      "3138429640.819806\n",
      "69.0\n",
      "79\n",
      "12292794388.0\n",
      "14090104683.743567\n",
      "69.0\n",
      "46\n",
      "7460956180.0\n",
      "8176254731.930414\n",
      "69.0\n",
      "69.0\n",
      "13\n",
      "2149165076.0\n",
      "2364707692.4571342\n",
      "69.0\n",
      "63\n",
      "9604311060.0\n",
      "11126652206.769382\n",
      "69.0\n",
      "68\n",
      "9876875284.0\n",
      "12042229312.047745\n",
      "69.0\n",
      "81\n",
      "12292794388.0\n",
      "14474056733.949755\n",
      "69.0\n",
      "88\n",
      "12292794388.0\n",
      "15596183251.321285\n",
      "69.0\n",
      "65\n",
      "10149439508.0\n",
      "11622052883.883121\n",
      "69.0\n",
      "76\n",
      "11714111508.0\n",
      "13495133209.836542\n",
      "69.0\n",
      "63\n",
      "9876875284.0\n",
      "11297672192.597015\n",
      "69.0\n",
      "60\n",
      "9059182612.0\n",
      "10722421684.546791\n",
      "Sampled 22 times has 20 unique graph\n"
     ]
    }
   ],
   "source": [
    "from ptflops import get_model_complexity_info\n",
    "import copy\n",
    "sampled_adj = []\n",
    "sampled_ops = []\n",
    "sampled_flops_list = []\n",
    "sampled_zc_list = []\n",
    "flops_list = []\n",
    "input_size = (3, 32, 32)  # Modify this according to your model input size\n",
    "\n",
    "hash_list = []\n",
    "torch.set_grad_enabled(True)\n",
    "time_out = 0\n",
    "while(len(hash_list)<20 and time_out<200):\n",
    "\n",
    "    adj, ops = sample_graph_from_cnf(rev_model, m1, context)\n",
    "    model, hash = get_model(adj, ops,shave=False)\n",
    "    #print(model)\n",
    "    if hash not in hash_list:\n",
    "        hash_list.append(hash)\n",
    "        score = get_tcet(model, train_loader)\n",
    "\n",
    "        sampled_zc_list.append(score)\n",
    "        sampled_adj.append(adj)\n",
    "        sampled_ops.append(ops)\n",
    "        # Profile the model\n",
    "        macs, params = get_model_complexity_info(copy.deepcopy(model), input_size, as_strings=False, print_per_layer_stat=False)\n",
    "        \n",
    "        #params_list.append(float(params))\n",
    "        flops_list.append(float(macs) * 2)\n",
    "        print(int(score/max_zc_score*100))\n",
    "        print(float(macs) * 2)\n",
    "        print(score)\n",
    "    time_out+=1\n",
    "\n",
    "print('Sampled {} times has {} unique graph'.format(time_out, len(hash_list)))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 73,
   "metadata": {},
   "outputs": [],
   "source": [
    "import seaborn as  sns\n",
    "import matplotlib.pyplot as plt\n",
    "zc = sampled_zc_list \n",
    "CNF_label = ['CNF']*len(flops_list)\n",
    "type_list  = CNF_label  \n",
    "df = pd.DataFrame(\n",
    "    {\n",
    "    'zc': zc,\n",
    "    'type': type_list,\n",
    "    })\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 74,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Text(12643948533.37586, 8, 'reference model')"
      ]
     },
     "execution_count": 74,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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1157rY4cOVJPaX2zbNkyjRs3Ti1btpTNZtPHH3/8m+9ZunSp+vbtq+DgYLVr104vvfRS3Qc9TTXdTn/dD/ny73mMP+2HfNlOf90PWZnfF6B3331Xd911l+6//36tW7dOQ4YM0XnnnafMzMwTvn7ZsmU655xz9MUXXyg1NVXDhw/XuHHjtG7dunpOXjM13c5j8vPzNWnSJI0cObKekp4eX7bz0ksv1aJFi/Taa69p69atmjt3rjp37lyPqWuuptu5YsUKTZo0Sddff702bdqk9957Tz/88INuuOGGek5eM8XFxerZs6dmzpx5Sq/ftWuXxo4dqyFDhmjdunW67777dMcdd+iDDz6o46Snp6bb6a/7oZpu5zH+th/yZTv9cT9keYaf69+/v3HLLbdUW9a5c2fjz3/+8ymvo0uXLsaMGTNqO1qt8nU7L7vsMuOBBx4wHn74YaNnz551mLB21HQ7v/zySyMqKso4cuRIfcSrNTXdzqeeespo165dtWX//Oc/jdatW9dZxtomyfjoo49+9TV/+tOfjM6dO1dbdvPNNxsDBgyow2S161S280T8YT/0SzXZTn/bD/3SqWynv+6HrM6vjwCVl5crNTVVo0ePrrZ89OjRWrly5Smtw+v1qrCwUM2aNauLiLXC1+2cPXu2duzYoYcffriuI9YKX7Zz3rx56tevn5588km1atVKnTp10tSpU1VaWlofkX3iy3YOGjRI+/bt0xdffCHDMHTo0CG9//77Ov/88+sjcr1ZtWrVcT+XMWPGaM2aNaqoqDApVd3zh/2Qr/xtP+QLf9wPoQHcCfp05OTkyOPxHDd7fGxs7HGzzJ/M008/reLiYl166aV1EbFW+LKd27dv15///GctX75cTqd//DP7sp07d+7UihUrFBwcrI8++kg5OTm69dZblZub22DPv/uynYMGDdKcOXN02WWXqaysTG63WxdeeKGef/75+ohcb7Kysk74c3G73crJyVF8fLxJyeqWP+yHfOGP+yFf+ON+CI1gDJAk2Wy2at8bhnHcshOZO3eupk+frnfffVctWrSoq3i15lS30+Px6IorrtCMGTPUqVOn+opXa2ry7+n1emWz2TRnzhz1799fY8eO1TPPPKM33nijwf/1VZPt3Lx5s+644w499NBDSk1N1VdffaVdu3Y1yomDT/RzOdHyxsLf9kOnyt/3QzXhz/shK/PrSh4TEyOHw3HcX83Z2dnH/RX5v959911df/31eu+99zRq1Ki6jHnaarqdhYWFWrNmjdatW6fJkydLqvwFNQxDTqdTCxYs0IgRI+ole0348u8ZHx+vVq1aKSoqqmpZSkqKDMPQvn371LFjxzrN7AtftvPxxx/X4MGDdc8990iSevToobCwMA0ZMkR/+ctfGs2Rkbi4uBP+XJxOp6Kjo01KVXf8aT9UU/66H/KFP+6H4OdHgAIDA9W3b18tXLiw2vKFCxdq0KBBJ33f3Llzdc011+jtt9/2izEUNd3OyMhIbdiwQWlpaVVft9xyi5KTk5WWlqYzzzyzvqLXiC//noMHD9aBAwdUVFRUtWzbtm2y2+1q3bp1neb1lS/bWVJSIru9+q+rw+GQ9PMRksZg4MCBx/1cFixYoH79+ikgIMCkVHXD3/ZDNeWv+yFf+ON+CPL/q8DeeecdIyAgwHjttdeMzZs3G3fddZcRFhZm7N692zAMw/jzn/9sXHXVVVWvf/vttw2n02m88MILxsGDB6u+8vLyzNqEU1LT7fxf/nL1RU23s7Cw0GjdurVxySWXGJs2bTKWLl1qdOzY0bjhhhvM2oRTUtPtnD17tuF0Oo0XX3zR2LFjh7FixQqjX79+Rv/+/c3ahFNSWFhorFu3zli3bp0hyXjmmWeMdevWGXv27DEM4/jt3LlzpxEaGmrcfffdxubNm43XXnvNCAgIMN5//32zNuGU1HQ7/XU/VNPt/F/+sh+q6Xb6637I6vy+ABmGYbzwwgtGUlKSERgYaPTp08dYunRp1XNXX321MXTo0Krvhw4dakg67uvqq6+u/+A1VJPt/F/+suMxjJpvZ3p6ujFq1CgjJCTEaN26tTFlyhSjpKSknlPXXE2385///KfRpUsXIyQkxIiPjzcmTpxo7Nu3r55T18w333zzq79vJ9rOJUuWGL179zYCAwONNm3aGLNmzar/4DVU0+301/2QL/+ev+Qv+yFfttNf90NWZjOMRnT8HAAA4BT49RggAAAAX1CAAACA5VCAAACA5VCAAACA5VCAAACA5VCAAACA5VCAAACA5VCAAACNxrJlyzRu3Di1bNlSNptNH3/8cY3eX1ZWpmuuuUbdu3eX0+nURRdddMLXLV26VH379lVwcLDatWunl1566fTDo15RgAAAjUZxcbF69uypmTNn+vR+j8ejkJAQ3XHHHSedoHbXrl0aO3ashgwZonXr1um+++7THXfcoQ8++OB0oqOecSdoAECjZLPZ9NFHH1U7ilNeXq4HHnhAc+bMUV5enrp166YnnnhCw4YNO+7911xzjfLy8o47inTvvfdq3rx5Sk9Pr1p2yy236Mcff9SqVavqaGtQ2zgCBFjM7t27ZbPZjvs69h+Ab7/9VkOHDlVoaKiaNm2qMWPG6OjRo+aGBmrJtddeq2+//VbvvPOO1q9frwkTJujcc8/V9u3bT3kdq1at0ujRo6stGzNmjNasWaOKiorajow6QgECLCYhIUEHDx6s+lq3bp2io6N19tlnKy0tTSNHjlTXrl21atUqrVixQuPGjZPH4zE7NnDaduzYoblz5+q9997TkCFD1L59e02dOlVnnXWWZs+efcrrycrKUmxsbLVlsbGxcrvdysnJqe3YqCNOswMAqF8Oh0NxcXGSKgd8XnTRRRo4cKCmT5+uK6+8Uv369dOLL75Y9fquXbuaFRWoVWvXrpVhGOrUqVO15S6XS9HR0TVal81mq/b9sdEk/7scDRcFCLCw66+/XoWFhVq4cKHsdrvS0tI0YcIEs2MBdcLr9crhcCg1NVUOh6Pac+Hh4ae8nri4OGVlZVVblp2dLafTWeMiBfNQgACL+stf/qKvvvpK33//vSIiIiRJISEhJqcC6k7v3r3l8XiUnZ2tIUOG+LyegQMH6tNPP622bMGCBerXr58CAgJONybqCWOAAAv64IMP9Mgjj+i///2v2rdvX7W8R48eWrRokYnJgNNTVFSktLQ0paWlSaq8ZD0tLU2ZmZnq1KmTJk6cqEmTJunDDz/Url279MMPP+iJJ57QF198UbWOzZs3Ky0tTbm5ucrPz6+2Pqnyiq89e/ZoypQpSk9P1+uvv67XXntNU6dOreetxengMnjAYjZu3KgzzzxTU6ZM0W233Va1PDAwUDk5Oerevbuuv/563XLLLQoMDNQ333yjCRMmKCYmxsTUwKlZsmSJhg8fftzyq6++Wm+88YYqKir0l7/8RW+99Zb279+v6OhoDRw4UDNmzFD37t0lSW3atNGePXuOW8cv/3O5dOlS3X333dq0aZNatmype++9V7fcckvdbRhqHQUIsJg33nhD11577XHLhw4dqiVLlmjp0qW67777lJqaqpCQEJ155pl655131KRJk/oPCwB1hAIEAAAshzFAAADAcihAAADAcihAAADAcihAAADAcihAAADAcihAAADAcihAAADAcihAAADAcihAAADAcihAAADAcihAAADAcihAAADAcv4fbbx+JaPEOw4AAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 578.125x500 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig=sns.displot(df, x=\"zc\", hue=\"type\",  multiple='dodge', kde=True)\n",
    "plt.axvline(x=11643948533.37586, linestyle='--', color='r')\n",
    "plt.text(12643948533.37586,8, \"reference model\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 75,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "df = pd.DataFrame(\n",
    "{\n",
    "    'adj_matrix': sampled_adj,\n",
    "    'ops_features': sampled_ops\n",
    "})\n",
    "\n",
    "save_path = os.path.join('../experiments/nasbench101')\n",
    "os.makedirs(save_path, exist_ok=True)\n",
    "\n",
    "df.to_pickle(os.path.join(save_path, 'proposed.pkl'))  "
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "ss_gcn",
   "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.7.12"
  },
  "orig_nbformat": 4,
  "vscode": {
   "interpreter": {
    "hash": "8d6e9c7cf7c366526aa938ed3bdab251528ee1c2408af727c1d113b1b1a37c3a"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
