{
 "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": 1,
   "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": 2,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "603914781.4454474"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "nasbench201_graph_vec=load_data('../experiments/nasbench201/labelled_dataset.pkl')\n",
    "\n",
    "max_zc_score = max(nasbench201_graph_vec['zc_score'])\n",
    "max_zc_score"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "nasbench201_graph_vec=load_data('../experiments/nasbench201/test_dataset.pkl')\n",
    "\n"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Load VGAE\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "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=576, out_features=128, bias=True)\n",
       "  (fc2): Linear(in_features=576, 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=81, 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=54, 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": 4,
     "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=9)\n",
    "m1 = m1.cuda()\n",
    "m1.load_state_dict(torch.load(\"../experiments/nasbench201/m1_dist_1.0_no_latency.pt\"))\n",
    "m1.eval().cuda()"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Load NF"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "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": 5,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from network_designer.models.cnf_modules.network import GraphFlow\n",
    "rev = torch.load('../experiments/nasbench201/cnf_zc_score.pt')\n",
    "rev_model = GraphFlow(num_latent=128, num_node=9, 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": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "nasbench201_graph_vec = nasbench201_graph_vec.sort_values(by='acc', ascending=False, na_position='first')\n",
    "context = nasbench201_graph_vec[0:1]['conditions'].values[0]\n",
    "context = np.array(context)\n",
    "context = np.round(context)\n",
    "context = context[1]"
   ]
  },
  {
   "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": 7,
   "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": 8,
   "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",
      "{0: (6.123233995736766e-17, 1.0), 1: (0.4338837391175582, 0.9009688679024191), 2: (0.7818314824680298, 0.6234898018587335), 3: (0.9749279121818236, 0.2225209339563144), 4: (0.9749279121818236, -0.2225209339563144), 5: (0.7818314824680298, -0.6234898018587335), 6: (0.4338837391175582, -0.9009688679024191), 7: (6.123233995736766e-17, -1.0)}\n"
     ]
    }
   ],
   "source": [
    "from network_designer.design_space.core.search_space import SearchSpace, calc_graph_hash\n",
    "from torch.distributions import Normal\n",
    "import numpy as np\n",
    "from network_designer.dataloader import define_dataloader\n",
    "from network_designer.design_space.core.net2vec import  get_tcet\n",
    "import torch\n",
    "from alethiometer import get_cifar_dataloaders\n",
    "from network_designer.design_space.nasbench201.model import Network\n",
    "from network_designer.design_space.nasbench201.operations import SearchSpaceNames\n",
    "from network_designer.dataloader import define_dataloader\n",
    "import torch\n",
    "\n",
    "train_loader, test_loader = get_cifar_dataloaders(64, 64, 'cifar10', 1)\n",
    "\n",
    "ss = SearchSpace(6, 3)\n",
    "\n",
    "\n",
    "def sample_graph_from_cnf(rev_model,m1, cond, t=0.4):\n",
    "    print(cond)\n",
    "    noise = Normal(0, 1).sample(sample_shape=torch.Size([1, 128]))*t\n",
    "    z = noise.cuda()\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",
    "search_space = SearchSpaceNames['nf_graph']\n",
    "OPS = ['conv_1x1', 'conv_3x3', 'max_pool_3x3','input', 'output']\n",
    "\n",
    "\n",
    "\n",
    "dataloader, n_classes = define_dataloader(dataset='cifar10', dataset_path='_dataset')\n",
    "def get_model(adj, ops, shave=True):\n",
    "    if shave:\n",
    "        o = ops.astype(int)\n",
    "        a = adj.astype(int)\n",
    "        a = np.triu(a, 1)\n",
    "        a = a[1:,1:]\n",
    "        o = o[1:]\n",
    "    else:\n",
    "        o = ops\n",
    "        a = adj\n",
    "        a = np.triu(a, 1)\n",
    "\n",
    "    return Network(C=16, N=5, num_classes=n_classes, search_space=search_space, adj_matrix=a, ops=o).cuda(), calc_graph_hash(a, o)\n",
    "\n",
    "import numpy as np\n",
    "# The labels for the nodes (let's convert the range to a list)\n",
    "\n",
    "labels = list(range(8))\n",
    "\n",
    "# Calculate the angles for each node\n",
    "angles = np.linspace(np.pi/2, -np.pi/2, len(labels)) \n",
    "\n",
    "fixed_positions = {}\n",
    "for label, theta in zip(labels, angles):\n",
    "    # Convert polar coordinates to Cartesian coordinates\n",
    "    x = np.cos(theta)\n",
    "    y = np.sin(theta)\n",
    "\n",
    "    fixed_positions[label] = (x, y)\n",
    "\n",
    "    \n",
    "print(fixed_positions)\n",
    "\n",
    "color_map = {\n",
    "    'conv_1x1': '#1f77b4',  # muted blue\n",
    "    'conv_3x3': '#ff7f0e',  # safety orange\n",
    "    'max_pool_3x3': '#2ca02c',  # cooked asparagus green\n",
    "    'input': '#d62728',  # brick red\n",
    "    'output': '#9467bd',  # muted purple\n",
    "}\n",
    "\n",
    "def draw_graph(adj_matrix, ops_feature, save_path, name, fixed_positions, zc=None, flop=None, cond=None):\n",
    "    import os\n",
    "    import networkx as nx\n",
    "    import matplotlib.pyplot as plt \n",
    "\n",
    "    G = nx.from_numpy_matrix(adj_matrix, create_using=nx.DiGraph)\n",
    "    labels = {idx: OPS[np.argmax(ops_f)] for idx, ops_f in enumerate(list(ops_feature))}\n",
    "\n",
    "    # Remove nodes without connections\n",
    "    isolated_nodes = [node for node in G.nodes() if G.degree(node) == 0]\n",
    "    G.remove_nodes_from(isolated_nodes)\n",
    "\n",
    "    # Update labels based on remaining nodes\n",
    "    labels = {node: labels[node] for node in G.nodes()}\n",
    "\n",
    "    node_colors = [color_map.get(labels[node], '#17becf') for node in G.nodes()]  # fallback color is blue-teal\n",
    "\n",
    "    pos = nx.drawing.nx_agraph.graphviz_layout(G, prog='dot')\n",
    "    plt.figure(figsize=(10, 10))\n",
    "    nx.draw_networkx(G, pos=pos, arrows=True, labels=labels, with_labels=True, node_color=node_colors, node_size=2400)\n",
    "    if cond:\n",
    "        plt.xlabel('FLOPS:{:.2f}, zc_score:{:.2f}, Edit_condition[{}, {}]'.format(zc, flop, cond[0][0], cond[0][1]), fontsize=15)\n",
    "    elif zc:\n",
    "        plt.xlabel('FLOPS:{:.2f}, zc_score:{:.2f}'.format(zc, flop), fontsize=18)\n",
    "        \n",
    "    os.makedirs(save_path, exist_ok=True)\n",
    "    plt.savefig(f\"{save_path}/{name}.png\")\n",
    "    plt.show()\n",
    "\n",
    "    return adj_matrix, list(labels.values())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "from ptflops import get_model_complexity_info\n",
    "import copy\n",
    "input_size = (3, 32, 32)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "401309434.07038426\n"
     ]
    }
   ],
   "source": [
    "input_size = (3, 32, 32)\n",
    "for index, row in nasbench201_graph_vec.iterrows():\n",
    "    adj = np.array(row['adj_matrix'])[1:, 1:]\n",
    "    ops = np.array(row['ops_features'])[1:]\n",
    "    model, _ = get_model(adj, ops,shave=False)\n",
    "    #print(model)\n",
    "\n",
    "    score = get_tcet(model, train_loader)\n",
    "\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",
    "    print(score)\n",
    "    break"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "71.0\n",
      "558411777.8229523\n",
      "92\n",
      "71.0\n",
      "456451708.11020315\n",
      "75\n",
      "71.0\n",
      "484354224.1042309\n",
      "80\n",
      "71.0\n",
      "314447918.1450783\n",
      "52\n",
      "71.0\n",
      "576156150.6787252\n",
      "95\n",
      "71.0\n",
      "453113639.94718796\n",
      "75\n",
      "71.0\n",
      "499559923.3231516\n",
      "82\n",
      "71.0\n",
      "505809491.8140882\n",
      "83\n",
      "71.0\n",
      "431403821.5522584\n",
      "71\n",
      "71.0\n",
      "485141895.9953601\n",
      "80\n",
      "71.0\n",
      "551387723.1652801\n",
      "91\n",
      "71.0\n",
      "515004693.4178982\n",
      "85\n",
      "71.0\n",
      "479582152.2696399\n",
      "79\n",
      "71.0\n",
      "458455574.89800143\n",
      "75\n",
      "71.0\n",
      "363394798.2468652\n",
      "60\n",
      "71.0\n",
      "457966106.4096347\n",
      "75\n",
      "71.0\n",
      "537463672.9906845\n",
      "88\n",
      "71.0\n",
      "499507396.0222184\n",
      "82\n",
      "71.0\n",
      "492449966.79659426\n",
      "81\n",
      "71.0\n",
      "443652556.67410606\n",
      "73\n",
      "Sampled 20 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",
    "    try:\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",
    "            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",
    "            \n",
    "            print(score)\n",
    "            print(int(score/max_zc_score*100))\n",
    "        time_out+=1\n",
    "    except:\n",
    "        pass\n",
    "\n",
    "print('Sampled {} times has {} unique graph'.format(time_out, len(hash_list)))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "import seaborn as  sns\n",
    "import matplotlib.pyplot as plt\n",
    "\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": 13,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Text(412398477.81339085, 8, 'reference model')"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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KJIUExOjW2+9Vbfccov8fr+OP/54lZeXa968eYqOjtb48eMP+1wtvd+P3XPPPRozZoyys7P129/+VnV1dfr000/1pz/9ST179tS4ceN02WWX6R//+IcGDRqk4uJiffXVV+rfv7/OOuuso3q/fui2227Tb37zGw0ePFhjxozRhx9+qHfffVdffvmlJOmUU05RTk5OQy3l5eW64447Gj3GuHHj9Mgjj2js2LG677771KlTJxUVFendd9/Vbbfd1mj6DIA1cBo80AYyMjL07bffyufz6fTTT1e/fv100003KS4uTnb7T/8a3n///br77rs1ZcoU9e7dW6effro+/PBDde3a9YjP19L7/dDo0aP11ltv6YMPPtDAgQN18skna+HChQ23T5s2TZdddpn++Mc/KicnR7/4xS+0cOFCZWZmNvk5muK8887T448/rkceeUR9+/bVM888o2nTpmn06NGSJLvdrvfee09er1fDhg3TVVddpQcffLDRY0RGRmru3LnKysrS+eefr969e+uKK65QdXU1I0KARdmMw02gAwAAtGOMAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEAAMshAAEA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f/3rX3I6nfL5fIfcJz09XTt37my0bffu3XI6nYf967U9aMn7dDhW+Ez9UFRUlPr37/+Tr9mKn6XD+bn36XCs9llC22MEqB0xDENer7fJ+y9durTdDsEfNGbMGC1fvrzRtssvv1y9evXSn//858OemZKbm6sPP/yw0bYvvvhCQ4cOlcvlatV6zdKS9+lwrPCZ+iGv16vVq1dr1KhRh73dip+lw/m59+lwrPZZggkMhKRJkyYZc+fONTZu3GgsW7bMuP322w273W588cUXhmEYxl/+8hfj0ksvbdj/0UcfNd577z1j3bp1xooVK4y//OUvhiTjnXfeMeslmObEE080brrppobrP36vNmzYYERGRhq33HKLsWrVKuP55583XC6X8fbbb5tQrXl+7n2y4mfqj3/8ozF79mxjw4YNxoIFC4xzzjnHiImJMTZt2mQYBp+lg5r7PlnxswTzMQIUonbt2qVLL71UO3bsUFxcnAYMGKDPPvtMp556qiRpx44dKioqati/pqZGt956q7Zt26aIiAj17dtXH3/8sc466yyzXkLQ+PF71bVrV33yySe65ZZb9OSTTyojI0P/+te/2v1pyz+Hz5S0detWXXTRRSouLlZKSopGjBihBQsWqHPnzpL4LB3U3PfJip8lmM9mGAeOyAMAALAIDoIGAACWQwACAACWQwACAACWQwACAACWQwACAACWQwACAACWQwACAACWQwACAFja3Llzde655yojI0M2m03vv/9+sx/j888/14gRIxQTE6OUlBT96le/0saNGwNfLAKGAAQAsLTKykodc8wxeuKJJ1p0/w0bNmjs2LE6+eSTlZ+fr88//1zFxcU6//zzA1wpAomVoAEAOMBms+m9997Teeed17CtpqZGd955p1577TWVlpaqX79+euihhzR69GhJ0ttvv62LLrpIXq9Xdnv9uMKHH36osWPHyuv1WqrxbShhBAiwoE2bNslmsx1yOfiF/u233+rEE09UZGSkEhISdPrpp6ukpMTcogGTXH755fr22281ffp0LVu2TBdccIHOOOMMrV+/XpI0dOhQORwOTZs2TT6fT2VlZXrllVd02mmnEX6CGAEIsKDMzEzt2LGj4bJ06VIlJSXphBNOUH5+vsaMGaO+fftq/vz5+uabb3TuuefK5/OZXTbQ5goLC/XGG2/orbfe0qhRo5Sdna1bb71Vxx9/vKZNmyZJ6tKli7744gvdfvvtcrvdio+P19atWzV9+nSTq8eR0A0esCCHw6H09HRJksfj0Xnnnafc3Fzde++9uuSSSzR06FA99dRTDfv37dvXrFIBUy1ZskSGYahnz56Ntnu9XiUlJUmSdu7cqauuukrjx4/XRRddpP379+vuu+/Wr3/9a82YMUM2m82M0vEzCECAxV155ZXav3+/ZsyYIbvdrvz8fF1wwQVmlwUEBb/fL4fDoby8PDkcjka3RUdHS5KefPJJxcbG6uGHH2647dVXX1VmZqYWLlyoESNGtGnNaBoCEGBhDzzwgD777DMtWrRIMTExkqSIiAiTqwKCx6BBg+Tz+bR7926NGjXqsPtUVVUdEo4OXvf7/a1eI1qGY4AAi3rnnXd033336f/+7/+UnZ3dsH3AgAGaOXOmiZUBbauiokL5+fnKz8+XJG3cuFH5+fkqKipSz549NW7cOF122WV69913tXHjRi1evFgPPfSQPvnkE0nS2WefrcWLF+u+++7T+vXrtWTJEl1++eXq3LmzBg0aZOIrw5FwGjxgQStWrNDw4cM1ceJETZgwoWF7WFiYiouL1b9/f1155ZX6/e9/r7CwMM2aNUsXXHCBkpOTTawaaB2zZ8/WSSeddMj28ePH68UXX1Rtba0eeOABvfzyy9q2bZuSkpKUm5uryZMnq3///pKk6dOn6+GHH9a6desUGRmp3NxcPfTQQ+rVq1dbvxw0EQEIsKAXX3xRl19++SHbTzzxRM2ePVtz5szR7bffrry8PEVERGj48OGaPn264uPj275YAGgFBCAAAGA5HAMEAAAshwAEAAAshwAEAAAshwAEAAAshwAEAAAshwAEAAAshwAEAAAshwAEAAAshwAEAAAshwAEAAAshwAEAAAshwAEAAAs5/8BqYaKFJz3BqgAAAAASUVORK5CYII=",
      "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=428485006.60936517, linestyle='--', color='r')\n",
    "plt.text(412398477.81339085,8, \"reference model\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "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/nasbench201')\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
}
