{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "e74492fc",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "finish import\n"
     ]
    }
   ],
   "source": [
    "import os, sys\n",
    "dir2 = os.path.abspath('')\n",
    "dir1 = os.path.dirname(dir2)\n",
    "if not dir1 in sys.path:\n",
    "    sys.path.append(dir1)\n",
    "import json\n",
    "import numpy as np\n",
    "from models.model_utils import get_model\n",
    "from data.data_utils import get_dataloader\n",
    "import torch\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "import time\n",
    "\n",
    "from neuron_affinity import *\n",
    "from sklearn.decomposition import PCA\n",
    "from sklearn.preprocessing import StandardScaler\n",
    "from sklearn.pipeline import make_pipeline"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ee312e0d",
   "metadata": {},
   "outputs": [],
   "source": [
    "# ! pip install sklearn\n",
    "! nvidia-smi"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "08fe95cb",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Cuda True\n"
     ]
    }
   ],
   "source": [
    "print(f'Cuda {torch.cuda.is_available()}')\n",
    "root = None  # path to the experiment root\n",
    "# exp_name = 'Resnet18_CIFAR10_10k_100_COPIES'\n",
    "exp_name = 'Resnet18_CIFAR10_45k_30X_ALL_DIFF'\n",
    "exp_dir = root.format(exp_name)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "752823cc",
   "metadata": {},
   "outputs": [],
   "source": [
    "model_1_path = os.path.join(exp_dir, 'model_1')\n",
    "model_2_path = os.path.join(exp_dir, 'model_0')\n",
    "config_path = os.path.join(model_1_path, 'config')\n",
    "weight_1_path = os.path.join(model_1_path, 'weight.pt')\n",
    "weight_2_path = os.path.join(model_2_path, 'weight.pt')\n",
    "print('Loading config...')\n",
    "with open(config_path, 'r') as f:\n",
    "    config = json.load(f)\n",
    "print(config)\n",
    "model_1 = get_model(config).cuda()\n",
    "print('Model 1 created')\n",
    "state_dict = torch.load(weight_1_path, map_location=torch.device('cpu'))\n",
    "print('Reading weights from disk.')\n",
    "model_1.load_state_dict(state_dict)\n",
    "print('Model 1 loaded...')\n",
    "model_2 = get_model(config).cuda()\n",
    "state_dict = torch.load(weight_2_path, map_location=torch.device('cpu'))\n",
    "model_2.load_state_dict(state_dict)\n",
    "print('All models loaded!')\n",
    "# model_1.cuda()\n",
    "# model_2.cuda()\n",
    "# print('moving to cuda')\n",
    "print('getting data loader')\n",
    "all_dataloaders = get_dataloader(config)\n",
    "print('get data loader')\n",
    "_, test_loader, _ = all_dataloaders  # random subset\n",
    "stats = compute_statistics(\n",
    "    model_1=model_1,\n",
    "    model_2=model_2,\n",
    "    dataloader=test_loader,\n",
    "    layer_name='layer4',\n",
    "    neuron_index_1=0,\n",
    "    neuron_index_2=0\n",
    ")\n",
    "print(stats)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "ea856417",
   "metadata": {},
   "outputs": [],
   "source": [
    "model_1_repr = get_all_repr(model_1, test_loader, 'layer4')\n",
    "model_2_repr = get_all_repr(model_2, test_loader, 'layer4')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "669837aa",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(10000, 512, 4, 4)"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model_1_repr.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "13c8450d",
   "metadata": {},
   "outputs": [],
   "source": [
    "model_1_repr_flat = model_1_repr.reshape((len(model_1_repr), -1))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "a6128fee",
   "metadata": {},
   "outputs": [],
   "source": [
    "scaler = StandardScaler()\n",
    "pca = PCA(n_components=60)\n",
    "pca = make_pipeline(scaler, pca)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "979087a3",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Pipeline(steps=[('standardscaler', StandardScaler()),\n",
       "                ('pca', PCA(n_components=60))])"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "pca.fit(model_1_repr_flat)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "d8dcc422",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[0.10037851 0.07607154 0.06854229 0.0567801  0.05536329 0.04891356\n",
      " 0.04044061 0.03494425 0.02990319 0.0146265  0.0106576  0.01009009\n",
      " 0.00938504 0.00854084 0.00777959 0.00711056 0.00672178 0.00649285\n",
      " 0.00645779 0.00585233 0.00560552 0.00520636 0.00473567 0.00465718\n",
      " 0.00457579 0.0041459  0.00392966 0.00386449 0.00372582 0.00342288\n",
      " 0.00335044 0.00320106 0.00315528 0.00287117 0.0028062  0.00257455\n",
      " 0.00253643 0.00242645 0.00237556 0.00230532 0.00224815 0.00220835\n",
      " 0.00217405 0.0020829  0.00204217 0.00200221 0.00193156 0.0018943\n",
      " 0.00188332 0.00186654 0.00178919 0.0017426  0.00165871 0.00163514\n",
      " 0.00158466 0.00155216 0.00150684 0.00144907 0.00142928 0.00140823]\n",
      "0.7126134\n"
     ]
    }
   ],
   "source": [
    "print(pca[1].explained_variance_ratio_)\n",
    "print(np.sum(pca[1].explained_variance_ratio_))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "088c38af",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([2867.5786 , 2496.3533 , 2369.5957 , 2156.7163 , 2129.6384 ,\n",
       "       2001.7488 , 1820.1357 , 1691.9318 , 1565.1418 , 1094.6246 ,\n",
       "        934.3826 ,  909.1643 ,  876.825  ,  836.46   ,  798.3131 ,\n",
       "        763.21497,  742.0565 ,  729.311  ,  727.3393 ,  692.4036 ,\n",
       "        677.6458 ,  653.0737 ,  622.8532 ,  617.67017,  612.24915,\n",
       "        582.7796 ,  567.37756,  562.6532 ,  552.4665 ,  529.5301 ,\n",
       "        523.89734,  512.08514,  508.4101 ,  484.9806 ,  479.46246,\n",
       "        459.2466 ,  455.8336 ,  445.8413 ,  441.14194,  434.5709 ,\n",
       "        429.14865,  425.33328,  422.0171 ,  413.07556,  409.01627,\n",
       "        404.99484,  397.78543,  393.93036,  392.78644,  391.03287,\n",
       "        382.8452 ,  377.82822,  368.62125,  365.99243,  360.29944,\n",
       "        356.58548,  351.3402 ,  344.54   ,  342.17865,  339.6501 ],\n",
       "      dtype=float32)"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "pca[1].singular_values_"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "abca9429",
   "metadata": {},
   "outputs": [],
   "source": [
    "projected_model_1_repr = pca.transform(model_1_repr_flat)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "429ed852",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(10000, 60)"
      ]
     },
     "execution_count": 14,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "projected_model_1_repr.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "4218706c",
   "metadata": {},
   "outputs": [],
   "source": [
    "def project_repr(model_repr, n_component=50):\n",
    "    model_repr_flat = model_repr.reshape((len(model_repr), -1))\n",
    "    scaler = StandardScaler()\n",
    "    pca = PCA(n_components=n_component)\n",
    "    pipeline = make_pipeline(scaler, pca)\n",
    "    pipeline.fit(model_repr_flat)\n",
    "    return pipeline.transform(model_repr_flat)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "71f59fb7",
   "metadata": {},
   "outputs": [],
   "source": [
    "n_component = 50"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "411c954f",
   "metadata": {},
   "outputs": [],
   "source": [
    "projected_model_1_repr = project_repr(model_1_repr, n_component=n_component)\n",
    "projected_model_2_repr = project_repr(model_2_repr, n_component=n_component)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "e7e232cb",
   "metadata": {},
   "outputs": [],
   "source": [
    "def all_feature_mean(rep):\n",
    "    mean = []\n",
    "    for i in range(rep.shape[1]):\n",
    "        feature_rep = rep[:, i].reshape((rep.shape[0], -1))\n",
    "        mean.append(feature_rep.mean())\n",
    "    return np.array(mean)\n",
    "\n",
    "model_1_repr_mean = all_feature_mean(projected_model_1_repr)\n",
    "model_2_repr_mean = all_feature_mean(projected_model_2_repr)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "ec2f0717",
   "metadata": {},
   "outputs": [],
   "source": [
    "# model_1_repr_mean"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "995587b4",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "-3.982544e-07\n",
      "-8.0184935e-07\n"
     ]
    }
   ],
   "source": [
    "print(model_1_repr_mean[0])\n",
    "print(model_2_repr_mean[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "f9c7a0e9",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-22.83424     -7.5357285   -1.4188837   40.357063    21.326138\n",
      "  -1.5101359   -5.522785    19.10908     11.903843     4.0556064\n",
      "   1.5514964   -8.780796    -0.14379907   6.571035     7.572919\n",
      "  -7.5065413    5.4369826   -7.3627625    3.2183752   -8.83646\n",
      "   4.5968904    5.014798     2.5543118    0.8352678   -2.640864\n",
      "  -5.9397964    9.986574    -4.2488728    5.215234     3.2872396\n",
      "   4.435627    -2.953592     7.9459753   -1.3441305   -1.6950037\n",
      "   0.74115     -1.8856925    3.107887     4.1359367   -1.1591101\n",
      "   3.7779436   -0.161026    -1.4351733   -5.7558155    2.8780544\n",
      "   9.335415     0.18394457  11.754596    -0.44312105   6.3828387 ]\n",
      "[-14.2604885  -10.476031    -5.0365467   18.551376   -32.971928\n",
      "  38.287117    26.023464    33.80596    -13.686318     1.783886\n",
      "   5.817314    -0.6321578   14.103548    19.327595    23.854279\n",
      "   1.7995226    7.485539    19.003265     1.3297036   -2.7611957\n",
      "  -5.4939156   -4.1016026    3.7458868    2.9977603   14.821112\n",
      "  -3.5904171   -1.3067279    4.1792593    6.102993     4.063856\n",
      "   1.89277     -8.689761     8.693308     4.837392     3.6271558\n",
      "   7.6958046   -2.0958192    4.4915543   -5.633437   -10.221651\n",
      "   3.4185262    4.399722    -5.166271    -0.92185163   7.17426\n",
      "   7.653073     3.8968468   -0.8526167    1.8188728    3.3389144 ]\n"
     ]
    }
   ],
   "source": [
    "print(projected_model_1_repr[0])\n",
    "print(projected_model_2_repr[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "5b8589ea",
   "metadata": {},
   "outputs": [],
   "source": [
    "def all_feature_std(rep):\n",
    "    std = []\n",
    "    for i in range(rep.shape[1]):\n",
    "        feature_rep = rep[:, i].reshape((rep.shape[0], -1))\n",
    "        std.append(feature_rep.std())\n",
    "    return np.array(std)\n",
    "model_1_repr_std = all_feature_std(projected_model_1_repr)\n",
    "model_2_repr_std = all_feature_std(projected_model_2_repr)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "759ba12a",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "28.675817\n",
      "28.817894\n"
     ]
    }
   ],
   "source": [
    "print(model_1_repr_std[0])\n",
    "print(model_2_repr_std[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "dba07579",
   "metadata": {},
   "outputs": [],
   "source": [
    "cross_neuron_corr = [[None for _ in range(n_component)] for _ in range(n_component)]\n",
    "for i in range(n_component):\n",
    "    for j in range(n_component):\n",
    "        rep_1 = projected_model_1_repr[:, i]\n",
    "        rep_2 = projected_model_2_repr[:, j]\n",
    "        correlation = ((rep_1 - model_1_repr_mean[i])*(rep_2 - model_2_repr_mean[j])).mean()\n",
    "        correlation /= model_1_repr_std[i] * model_2_repr_std[j]\n",
    "        cross_neuron_corr[i][j] = correlation\n",
    "cross_neuron_corr = np.array(cross_neuron_corr)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "9c140638",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([[ 0.88073814, -0.25333494,  0.2373513 , ..., -0.01315741,\n",
       "        -0.00233607,  0.00690957],\n",
       "       [-0.25721747, -0.8697694 , -0.06519635, ..., -0.00587914,\n",
       "        -0.00966503, -0.00146298],\n",
       "       [-0.2581657 , -0.00815904,  0.8704353 , ..., -0.04772949,\n",
       "         0.02253933,  0.00145899],\n",
       "       ...,\n",
       "       [ 0.01556117, -0.00509797, -0.01888047, ...,  0.05851851,\n",
       "         0.07626711,  0.00536575],\n",
       "       [ 0.00636631,  0.00158764, -0.03205769, ...,  0.10447973,\n",
       "         0.21477483, -0.05193203],\n",
       "       [-0.0090469 , -0.00229963,  0.02590584, ...,  0.13607027,\n",
       "        -0.04480603,  0.08443604]], dtype=float32)"
      ]
     },
     "execution_count": 25,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "cross_neuron_corr"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "47e96662",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.88073814 -0.8697694\n"
     ]
    }
   ],
   "source": [
    "print(cross_neuron_corr.max(), cross_neuron_corr.min())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "2b04463e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "_ = plt.hist(cross_neuron_corr.reshape(-1), bins=30)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "3abf93af",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.image.AxesImage at 0x7fde423fed30>"
      ]
     },
     "execution_count": 28,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 1080x1080 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(figsize=(15, 15))\n",
    "plt.imshow(np.abs(cross_neuron_corr))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "a87a5b8e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "neuron_correlation_pca\n"
     ]
    }
   ],
   "source": [
    "sub_dir_name = 'neuron_correlation_pca'\n",
    "try:\n",
    "    os.mkdir(os.path.join(exp_dir, sub_dir_name))\n",
    "except FileExistsError:\n",
    "    print('Folder exists')\n",
    "print(sub_dir_name)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9b3b6c5a",
   "metadata": {},
   "outputs": [],
   "source": [
    "n_model = 20\n",
    "n_component = 50\n",
    "tic = time.time()\n",
    "for a in range(n_model):\n",
    "    for b in range(n_model):\n",
    "#         if os.path.exists(os.path.join(exp_dir, sub_dir_name, f'corr_{a}_{b}.npy')):\n",
    "#             print('already done')\n",
    "#             continue\n",
    "        model_1_path = os.path.join(exp_dir, f'model_{a}')\n",
    "        model_2_path = os.path.join(exp_dir, f'model_{b}')\n",
    "        config_path = os.path.join(model_1_path, 'config')\n",
    "        weight_1_path = os.path.join(model_1_path, 'weight.pt')\n",
    "        weight_2_path = os.path.join(model_2_path, 'weight.pt')\n",
    "        model_1 = get_model(config).cuda()\n",
    "        state_dict = torch.load(weight_1_path, map_location=torch.device('cpu'))\n",
    "        model_1.load_state_dict(state_dict)\n",
    "        model_2 = get_model(config).cuda()\n",
    "        state_dict = torch.load(weight_2_path, map_location=torch.device('cpu'))\n",
    "        model_2.load_state_dict(state_dict)\n",
    "        model_1_repr = get_all_repr(model_1, test_loader, 'layer4')\n",
    "        model_2_repr = get_all_repr(model_2, test_loader, 'layer4')\n",
    "        projected_model_1_repr = project_repr(model_1_repr, n_component=n_component)\n",
    "        projected_model_2_repr = project_repr(model_2_repr, n_component=n_component)\n",
    "        model_1_repr_mean = all_feature_mean(projected_model_1_repr)\n",
    "        model_2_repr_mean = all_feature_mean(projected_model_2_repr)\n",
    "        model_1_repr_std = all_feature_std(projected_model_1_repr)\n",
    "        model_2_repr_std = all_feature_std(projected_model_2_repr)\n",
    "        n_neuron = len(model_1_repr_mean)\n",
    "        cross_neuron_corr = [[None for _ in range(n_component)] for _ in range(n_component)]\n",
    "        for i in range(n_component):\n",
    "            for j in range(n_component):\n",
    "                rep_1 = projected_model_1_repr[:, i]\n",
    "                rep_2 = projected_model_2_repr[:, j]\n",
    "                correlation = ((rep_1 - model_1_repr_mean[i])*(rep_2 - model_2_repr_mean[j])).mean()\n",
    "                correlation /= model_1_repr_std[i] * model_2_repr_std[j]\n",
    "                cross_neuron_corr[i][j] = correlation\n",
    "        cross_neuron_corr = np.float32(np.array(cross_neuron_corr))\n",
    "        np.save(os.path.join(exp_dir, sub_dir_name , f'corr_{a}_{b}'), cross_neuron_corr)\n",
    "        print('='*60)\n",
    "        print(f'{a} {b} {(time.time()-tic)/(a*n_model + b + 1)} s/pair')\n",
    "#         if b == 2:\n",
    "#             print(cross_neuron_corr)\n",
    "#             assert False"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "38596a83",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "09d6c8c3",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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.9.12 (main, Mar 26 2022, 15:51:13) \n[Clang 12.0.0 (clang-1200.0.32.29)]"
  },
  "vscode": {
   "interpreter": {
    "hash": "aee8b7b246df8f9039afb4144a1f6fd8d2ca17a180786b69acc140d282b71a49"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
