{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "6b6cbac6",
   "metadata": {},
   "outputs": [],
   "source": [
    "# notebook with basic loading tests for dataset classes"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "7515de4a",
   "metadata": {},
   "outputs": [],
   "source": [
    "# dataset_base\n",
    "# comparison: permutation of checkpoint -> vectorization -> slicing\n",
    "import logging\n",
    "logging.basicConfig(level=logging.WARNING)\n",
    "\n",
    "from pathlib import Path\n",
    "import json\n",
    "from shrp.datasets.dataset_base import ModelDatasetBase"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "5c87caab",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2023-01-31 18:03:53,510\tINFO worker.py:1538 -- Started a local Ray instance.\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "loading checkpoints from [PosixPath('/netscratch2/kschuerholt/code/versai/model_zoos/zoos/CIFAR10/resnet19/kaiming_uniform/tune_zoo_cifar10_resnet18_kaiming_uniform')]\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████| 14/14 [00:07<00:00,  1.84it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Data loaded. found 14 usable samples out of potential 14 samples.\n",
      "Load properties for samples from paths.\n",
      "### load data for dict_keys(['test_acc', 'training_iteration', 'ggap'])\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "0it [00:00, ?it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "training iteration 4 and epoch 5 don't match.\n",
      "### load data for dict_keys(['test_acc', 'training_iteration', 'ggap'])\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "14it [00:00, 37.23it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Properties loaded.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "config_key_list = []\n",
    "result_key_list = [\n",
    "    \"test_acc\",\n",
    "    \"training_iteration\",\n",
    "    \"ggap\",\n",
    "#     \"sparsity_ratio\",\n",
    "]\n",
    "property_keys = {\n",
    "    \"result_keys\": result_key_list,\n",
    "    \"config_keys\": config_key_list,\n",
    "}\n",
    "\n",
    "# path_root = Path('/netscratch2/dtaskiran/zoos/MNIST/tune_zoo_mnist_uniform/')\n",
    "path_root = Path('/netscratch2/kschuerholt/code/versai/model_zoos/zoos/CIFAR10/resnet19/kaiming_uniform/tune_zoo_cifar10_resnet18_kaiming_uniform')\n",
    "\n",
    "dataset_test = ModelDatasetBase(\n",
    "    root=path_root,\n",
    "    epoch_lst=[5,10],\n",
    "    mode=\"checkpoint\",\n",
    "    train_val_test=\"train\",  # determines whcih dataset split to use\n",
    "    ds_split=[0.7, 0.15,0.15],  #\n",
    "    max_samples=10,\n",
    "    weight_threshold=float(\"inf\"),\n",
    "    filter_function=None,  # gets sample path as argument and returns True if model needs to be filtered out\n",
    "    property_keys=property_keys,\n",
    "    num_threads=4,\n",
    "    shuffle_path=True,\n",
    "    verbosity=3,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "922f7ac5",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "14"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(dataset_test)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "2d94ba92",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████| 14/14 [00:11<00:00,  1.21it/s]\n"
     ]
    }
   ],
   "source": [
    "weights_per_channel = dataset_test.__get_weights_per_channel__()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "edcab048",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "14"
      ]
     },
     "execution_count": 5,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(weights_per_channel)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "71c88a19",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "18"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "idx = 4\n",
    "len(weights_per_channel[idx])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "3638a147",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "['layer_0-conv1.weight',\n",
       " 'layer_6-layer1.0.conv1.weight',\n",
       " 'layer_12-layer1.0.conv2.weight',\n",
       " 'layer_18-layer1.1.conv1.weight',\n",
       " 'layer_24-layer1.1.conv2.weight',\n",
       " 'layer_30-layer2.0.conv1.weight',\n",
       " 'layer_36-layer2.0.conv2.weight',\n",
       " 'layer_48-layer2.1.conv1.weight',\n",
       " 'layer_54-layer2.1.conv2.weight',\n",
       " 'layer_60-layer3.0.conv1.weight',\n",
       " 'layer_66-layer3.0.conv2.weight',\n",
       " 'layer_78-layer3.1.conv1.weight',\n",
       " 'layer_84-layer3.1.conv2.weight',\n",
       " 'layer_90-layer4.0.conv1.weight',\n",
       " 'layer_96-layer4.0.conv2.weight',\n",
       " 'layer_108-layer4.1.conv1.weight',\n",
       " 'layer_114-layer4.1.conv2.weight',\n",
       " 'layer_120-fc.weight']"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "\n",
    "dataset_test.channel_labels"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "2756b51f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([64, 27])\n",
      "torch.Size([64, 576])\n",
      "torch.Size([64, 576])\n",
      "torch.Size([64, 576])\n",
      "torch.Size([64, 576])\n",
      "torch.Size([128, 576])\n",
      "torch.Size([128, 1152])\n",
      "torch.Size([128, 1152])\n",
      "torch.Size([128, 1152])\n",
      "torch.Size([256, 1152])\n",
      "torch.Size([256, 2304])\n",
      "torch.Size([256, 2304])\n",
      "torch.Size([256, 2304])\n",
      "torch.Size([512, 2304])\n",
      "torch.Size([512, 4608])\n",
      "torch.Size([512, 4608])\n",
      "torch.Size([512, 4608])\n",
      "torch.Size([10, 513])\n",
      "10992330\n"
     ]
    }
   ],
   "source": [
    "count = 0\n",
    "for jdx in range(len(dataset_test.weights[idx])):\n",
    "    print(dataset_test.weights[idx][jdx].shape)\n",
    "    count += dataset_test.weights[idx][jdx].numel()\n",
    "print(count) # ought to be 2464 for normal MNIST-Seed"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "98bd5e01",
   "metadata": {},
   "source": [
    "# test new model dataset class with epoch resolution"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "c0821227",
   "metadata": {},
   "outputs": [],
   "source": [
    "# dataset_base\n",
    "from pathlib import Path\n",
    "import json\n",
    "from shrp.datasets.dataset_epochs import ModelDatasetBaseEpochs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ae9f933a",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2023-03-12 20:39:09,403\tINFO worker.py:1538 -- Started a local Ray instance.\n",
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 35700/35700 [04:12<00:00, 141.54it/s]\n",
      "0it [00:00, ?it/s]\n",
      "ERROR:root:Exception occurred: training iteration 0 and epoch 1 don't match.\n",
      "Traceback (most recent call last):\n",
      "  File \"/netscratch2/kschuerholt/code/shrp/src/shrp/datasets/dataset_epochs.py\", line 204, in __init__\n",
      "    self.read_properties(\n",
      "  File \"/netscratch2/kschuerholt/code/shrp/src/shrp/datasets/dataset_epochs.py\", line 293, in read_properties\n",
      "    assert (\n",
      "AssertionError: training iteration 0 and epoch 1 don't match.\n",
      "692it [02:44,  4.39it/s]"
     ]
    }
   ],
   "source": [
    "config_key_list = []\n",
    "result_key_list = [\n",
    "    \"test_acc\",\n",
    "    \"training_iteration\",\n",
    "    \"ggap\",\n",
    "#     \"sparsity_ratio\",\n",
    "]\n",
    "property_keys = {\n",
    "    \"result_keys\": result_key_list,\n",
    "    \"config_keys\": config_key_list,\n",
    "}\n",
    "\n",
    "# path_root = Path('/netscratch2/dtaskiran/zoos/MNIST/tune_zoo_mnist_uniform/')\n",
    "path_root = Path('/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/')\n",
    "# path_root = Path('/netscratch2/kschuerholt/code/versai/model_zoos/zoos/CIFAR10/resnet19/kaiming_uniform/tune_zoo_cifar10_resnet18_kaiming_uniform')\n",
    "\n",
    "dataset_ep_test = ModelDatasetBaseEpochs(\n",
    "    root=path_root,\n",
    "    epoch_lst=list(range(0,51)),\n",
    "    mode=\"checkpoint\",\n",
    "    train_val_test=\"train\",  # determines whcih dataset split to use\n",
    "    ds_split=[0.7, 0.15,0.15],  #\n",
    "#     max_samples=10,\n",
    "    weight_threshold=float(\"inf\"),\n",
    "    filter_function=None,  # gets sample path as argument and returns True if model needs to be filtered out\n",
    "    property_keys=property_keys,\n",
    "    num_threads=4,\n",
    "    shuffle_path=True,\n",
    "    verbosity=3,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "b805579c",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "700\n",
      "700\n",
      "700\n",
      "700\n",
      "700\n",
      "700\n",
      "700\n"
     ]
    }
   ],
   "source": [
    "print(len(dataset_ep_test.data))\n",
    "print(len(dataset_ep_test.paths))\n",
    "print(len(dataset_ep_test.epochs))\n",
    "print(len(dataset_ep_test.labels))\n",
    "\n",
    "for key in dataset_ep_test.properties.keys():\n",
    "    print(len(dataset_ep_test.properties[key]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "f85363bf",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "\n",
    "fig, ax = plt.subplots()\n",
    "\n",
    "for idx in range(25):\n",
    "    sns.lineplot(x=dataset_ep_test.properties['training_iteration'][idx],y=dataset_ep_test.properties['test_acc'][idx],ax=ax)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5cc8b638",
   "metadata": {},
   "outputs": [],
   "source": [
    "lst = [\n",
    "    [1, 2, None],\n",
    "    [None, 2, None],\n",
    "    [3,1,2],\n",
    "    [1,None,2],\n",
    "]\n",
    "lst2 = [1,2,1,None,None,1,None]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "085ce73e",
   "metadata": {},
   "outputs": [],
   "source": [
    "lst2clean = [ddx for ddx in lst2 if ddx]\n",
    "lst2clean"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c63cad83",
   "metadata": {},
   "outputs": [],
   "source": [
    "lst = [[ddx for ddx in lst[idx] if ddx] for idx in range(len(lst))]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5fdf5943",
   "metadata": {},
   "outputs": [],
   "source": [
    "lst"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "9ce39589",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "module_list.0.weight\n",
      "torch.Size([8, 1, 5, 5])\n",
      "module_list.0.bias\n",
      "torch.Size([8])\n",
      "module_list.3.weight\n",
      "torch.Size([6, 8, 5, 5])\n",
      "module_list.3.bias\n",
      "torch.Size([6])\n",
      "module_list.6.weight\n",
      "torch.Size([4, 6, 2, 2])\n",
      "module_list.6.bias\n",
      "torch.Size([4])\n",
      "module_list.9.weight\n",
      "torch.Size([20, 36])\n",
      "module_list.9.bias\n",
      "torch.Size([20])\n",
      "module_list.11.weight\n",
      "torch.Size([10, 20])\n",
      "module_list.11.bias\n",
      "torch.Size([10])\n"
     ]
    }
   ],
   "source": [
    "for key in dataset_ep_test.data[0][2].keys():\n",
    "    print(key)\n",
    "    print(dataset_ep_test.data[0][2][key].shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "5a5907f2",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([5, 16])"
      ]
     },
     "execution_count": 18,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import torch\n",
    "w = torch.randn([5,15])\n",
    "b = torch.randn([5])\n",
    "\n",
    "w2 = torch.cat([w,b.unsqueeze(dim=1)],dim=1)\n",
    "w2.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "db80bb1e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "75"
      ]
     },
     "execution_count": 21,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "w.shape\n",
    "w.numel()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "37d0c23e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "75"
      ]
     },
     "execution_count": 22,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "w3 = torch.randn([5,5,3])\n",
    "w3.numel()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "275cf991",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([5, 5, 3])"
      ]
     },
     "execution_count": 23,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "w4 = w.view(w3.shape)\n",
    "w4.shape"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "bdb77164",
   "metadata": {},
   "source": [
    "# Test SimCLRDataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "f087bda7",
   "metadata": {},
   "outputs": [],
   "source": [
    "# dataset_base\n",
    "\n",
    "import logging\n",
    "logging.basicConfig(level=logging.WARNING)\n",
    "\n",
    "from pathlib import Path\n",
    "import json\n",
    "from shrp.datasets.dataset_simclr import SimCLRDataset\n",
    "\n",
    "from shrp.git_re_basin.git_re_basin import (\n",
    "    PermutationSpec,\n",
    "    zoo_cnn_permutation_spec,\n",
    "    weight_matching,\n",
    "    apply_permutation,\n",
    ")\n",
    "\n",
    "# import logging\n",
    "# logging.basicConfig(level=logging.DEBUG)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "9119ae3c",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2023-03-15 00:07:43,744\tINFO worker.py:1538 -- Started a local Ray instance.\n",
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 10/10 [00:01<00:00,  5.15it/s]\n",
      "0it [00:00, ?it/s]\n",
      "ERROR:root:Exception occurred: training iteration 4 and epoch 5 don't match.\n",
      "Traceback (most recent call last):\n",
      "  File \"/netscratch2/kschuerholt/code/shrp/src/shrp/datasets/dataset_epochs.py\", line 204, in __init__\n",
      "    self.read_properties(\n",
      "  File \"/netscratch2/kschuerholt/code/shrp/src/shrp/datasets/dataset_epochs.py\", line 293, in read_properties\n",
      "    assert (\n",
      "AssertionError: training iteration 4 and epoch 5 don't match.\n",
      "10it [00:00, 220.84it/s]\n"
     ]
    }
   ],
   "source": [
    "config_key_list = []\n",
    "result_key_list = [\n",
    "    \"test_acc\",\n",
    "    \"training_iteration\",\n",
    "    \"ggap\",\n",
    "#     \"sparsity_ratio\",\n",
    "]\n",
    "property_keys = {\n",
    "    \"result_keys\": result_key_list,\n",
    "    \"config_keys\": config_key_list,\n",
    "}\n",
    "\n",
    "# path_root = Path('/netscratch2/dtaskiran/zoos/MNIST/tune_zoo_mnist_uniform/')\n",
    "path_root = Path('/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/')\n",
    "# path_root = Path('/netscratch2/kschuerholt/code/versai/model_zoos/zoos/CIFAR10/resnet19/kaiming_uniform/tune_zoo_cifar10_resnet18_kaiming_uniform')\n",
    "\n",
    "dataset_ep_test = SimCLRDataset(\n",
    "    root=path_root,\n",
    "    epoch_lst=[5],\n",
    "    mode=\"checkpoint\",\n",
    "#     mode=\"vector\",\n",
    "    permutations_number=0,\n",
    "    permutation_spec = zoo_cnn_permutation_spec(),\n",
    "    view_1_canonical = False,\n",
    "    view_2_canonical = False,\n",
    "    add_noise_view_1 = 0.0,  # [('input', 0.15), ('output', 0.013)]\n",
    "    add_noise_view_2 = 0.0,  # [('input', 0.15), ('output', 0.013)]\n",
    "    erase_augment=None,  # {\"p\": 0.5,\"scale\":(0.02,0.33),\"value\":0,\"mode\":\"block\"}\n",
    "    windowsize = 5,\n",
    "    train_val_test=\"train\",  # determines whcih dataset split to use\n",
    "    ds_split=[0.7, 0.15,0.15],  #\n",
    "    max_samples=15,\n",
    "    weight_threshold=float(\"inf\"),\n",
    "    filter_function=None,  # gets sample path as argument and returns True if model needs to be filtered out\n",
    "    property_keys=property_keys,\n",
    "    num_threads=4,\n",
    "    shuffle_path=True,\n",
    "    verbosity=3,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "5331cd52",
   "metadata": {},
   "outputs": [],
   "source": [
    "dataset_ep_test.set_module_window(windowsize = 3)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "cc544f49",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "<class 'collections.OrderedDict'>\n",
      "10\n"
     ]
    }
   ],
   "source": [
    "print(type(dataset_ep_test.data[0][-1]))\n",
    "print(len(dataset_ep_test.data[0][-1]))\n",
    "# print(len(dataset_ep_test.data[0][-1][0]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "caaa031e",
   "metadata": {},
   "outputs": [],
   "source": [
    "# idea:\n",
    "have index tensor 1: within components\n",
    "index tensor 2: numbers per component\n",
    "# permute both\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "63a5b112",
   "metadata": {},
   "outputs": [],
   "source": [
    "import copy\n",
    "import torch\n",
    "import ray\n",
    "if ray.is_initialized():\n",
    "    ray.shutdown()\n",
    "from shrp.datasets.progress_bar import ProgressBar\n",
    "\n",
    "def precompute_permutations(ref_checkpoint,permutation_number, perm_spec, num_threads=6):\n",
    "        logging.info(\"start precomputing permutations\")\n",
    "        model_curr = ref_checkpoint\n",
    "        # find permutation of model to itself as reference\n",
    "        reference_permutation = weight_matching(\n",
    "            ps=perm_spec, params_a=model_curr, params_b=model_curr\n",
    "        )\n",
    "\n",
    "        logging.info(\"get random permutation dicts\")\n",
    "        # compute random permutations\n",
    "        permutation_dicts = []\n",
    "        for ndx in range(permutation_number):\n",
    "            perm = copy.deepcopy(reference_permutation)\n",
    "            for key in perm.keys():\n",
    "                # get permuted indecs for current layer\n",
    "                perm[key] = torch.randperm(perm[key].shape[0]).float()\n",
    "            # append to list of permutation dicts\n",
    "            permutation_dicts.append(perm)\n",
    "\n",
    "\n",
    "        logging.info(\"get permutation indices\")\n",
    "        # get permutation data\n",
    "        ## get reference checkpoint\n",
    "        ref_checkpoint = copy.deepcopy(model_curr)\n",
    "        ## vectoirze\n",
    "        ref_vec_global = vectorize_checkpoint(ref_checkpoint)\n",
    "        ref_vec_kernel = copy.deepcopy(ref_vec_global)\n",
    "        ## get reference index vec\n",
    "        for idx, module in enumerate(ref_vec_global):\n",
    "            # get global index of permutation between kernels\n",
    "            index_global = torch.ones(module.numel())*idx\n",
    "            index_global = index_global.view(module.shape)\n",
    "            ref_vec_global[idx] = index_global\n",
    "            # got local index of permutation within kernels\n",
    "            index_kernel = torch.tensor(list(range(module.numel())))\n",
    "            index_kernel = index_kernel.view(module.shape)\n",
    "            ref_vec_kernel[idx] = index_kernel\n",
    "        ## map to checkpoint\n",
    "        ref_checkpoint_global = vector_to_checkpoint(\n",
    "            vector=ref_vec_global, reference_checkpoint=ref_checkpoint\n",
    "        )\n",
    "        ref_checkpoint_kernel = vector_to_checkpoint(\n",
    "            vector=ref_vec_kernel, reference_checkpoint=ref_checkpoint\n",
    "        )\n",
    "                                             \n",
    "\n",
    "        ## init multiprocessing environment ############\n",
    "        ray.init(num_cpus=num_threads)\n",
    "        pb = ProgressBar(total=permutation_number)\n",
    "        pb_actor = pb.actor\n",
    "        # get permutations\n",
    "        permutations_global = []\n",
    "        permutations_kernel = []\n",
    "        for perm_dict in permutation_dicts:\n",
    "            perm_curr_global = compute_single_perm.remote(\n",
    "                reference_checkpoint=ref_checkpoint_global,\n",
    "                permutation_dict=perm_dict,\n",
    "                perm_spec=perm_spec,\n",
    "                pba=pb_actor,\n",
    "            )\n",
    "            \n",
    "            perm_curr_kernel = compute_single_perm.remote(\n",
    "                reference_checkpoint=ref_checkpoint_kernel,\n",
    "                permutation_dict=perm_dict,\n",
    "                perm_spec=perm_spec,\n",
    "                pba=pb_actor,\n",
    "            )\n",
    "            \n",
    "            permutations_global.append(perm_curr_global)\n",
    "            permutations_kernel.append(perm_curr_kernel)\n",
    "\n",
    "        permutations_global = ray.get(permutations_global)\n",
    "        permutations_kernel = ray.get(permutations_kernel)\n",
    "                \n",
    "        permutations_global = [torch.tensor([perm[0].item() for perm in perm_g]).int() for perm_g in permutations_global]\n",
    "\n",
    "        permutations = [(perm_g, perm_k) for (perm_g, perm_k) in zip(permutations_global,permutations_kernel)]\n",
    "                \n",
    "        ray.shutdown()\n",
    "        \n",
    "        return permutation_dicts, permutations\n",
    "    \n",
    "@ray.remote(num_returns=1)\n",
    "def compute_single_perm(reference_checkpoint, permutation_dict, perm_spec, pba):\n",
    "    # copy reference checkpoint\n",
    "    index_check = copy.deepcopy(reference_checkpoint)\n",
    "    # apply permutation on checkpoint\n",
    "    index_check_perm = apply_permutation(\n",
    "        ps=perm_spec, perm=permutation_dict, params=index_check\n",
    "    )\n",
    "    # vectorize\n",
    "    index_perm = vectorize_checkpoint(index_check_perm)\n",
    "    # update counter\n",
    "    pba.update.remote(1)\n",
    "    # return list\n",
    "    return index_perm\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "1bd8e917",
   "metadata": {},
   "outputs": [],
   "source": [
    "def vectorize_checkpoint(checkpoint):\n",
    "    out = []\n",
    "    # use only weights and biases\n",
    "    for key in checkpoint.keys():\n",
    "        if \"weight\" in key:\n",
    "            w = checkpoint[key]\n",
    "            # flatten to out_channels x n\n",
    "            w = w.view(w.shape[0], -1)\n",
    "            # cat biases to channels if they exist in checkpoint\n",
    "            if key.replace(\"weight\", \"bias\") in checkpoint:\n",
    "                b = checkpoint[key.replace(\"weight\", \"bias\")]\n",
    "                w = torch.cat([w, b.unsqueeze(dim=1)], dim=1)\n",
    "            # split weights in slices along output channel dims\n",
    "            w = torch.split(w, w.shape[0])\n",
    "            # extend out with new tokens, zero's (and only entry) is a list\n",
    "            out.extend(w[0])\n",
    "\n",
    "    return out\n",
    "\n",
    "\n",
    "def vector_to_checkpoint(vector, reference_checkpoint):\n",
    "    # make copy to prevent memory management issues\n",
    "    checkpoint = copy.deepcopy(reference_checkpoint)\n",
    "    # use only weights and biases\n",
    "    idx_start = 0\n",
    "    for key in checkpoint.keys():\n",
    "        if \"weight\" in key:\n",
    "            # get correct slice of modules out of vec sequence\n",
    "            out_channels = checkpoint[key].shape[0]\n",
    "            idx_end = idx_start + out_channels\n",
    "            w = vector[idx_start:idx_end]\n",
    "            # update start\n",
    "            idx_start = idx_end\n",
    "            # get weight matrix from list of vectors\n",
    "            w = torch.stack(w, dim=0)\n",
    "            # extract bias\n",
    "            if key.replace(\"weight\", \"bias\") in checkpoint:\n",
    "                b = w[:, -1]\n",
    "                checkpoint[key.replace(\"weight\", \"bias\")] = b\n",
    "                w = w[:, :-1]\n",
    "            # reshape weight vector\n",
    "            w = w.view(checkpoint[key].shape)\n",
    "            logging.debug(\n",
    "                f\"vector_to_checkpoint: created weight {w.shape} for checkpoint weight {checkpoint[key].shape}\"\n",
    "            )\n",
    "            checkpoint[key] = w\n",
    "    return checkpoint"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "4cbb720e",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2023-03-14 23:54:58,595\tINFO worker.py:1538 -- Started a local Ray instance.\n"
     ]
    }
   ],
   "source": [
    "permdicts, perms = precompute_permutations(ref_checkpoint=dataset_ep_test.data[0][-1],permutation_number=3, perm_spec=zoo_cnn_permutation_spec())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "0f3abe09",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[{'P_bg0': tensor([5., 1., 2., 7., 6., 3., 4., 0.]),\n",
       "  'P_bg1': tensor([1., 3., 2., 4., 5., 0.]),\n",
       "  'P_bg2': tensor([2., 1., 3., 0.]),\n",
       "  'P_bg3': tensor([19.,  0.,  8., 12.,  9.,  6.,  7.,  5., 10., 13.,  1.,  3., 17., 11.,\n",
       "           4., 16., 18., 14.,  2., 15.])},\n",
       " {'P_bg0': tensor([2., 0., 4., 5., 1., 6., 7., 3.]),\n",
       "  'P_bg1': tensor([0., 2., 1., 4., 3., 5.]),\n",
       "  'P_bg2': tensor([2., 0., 3., 1.]),\n",
       "  'P_bg3': tensor([12., 13., 18., 15.,  3., 16.,  7.,  0., 19.,  6., 11.,  1., 17., 10.,\n",
       "           4.,  5.,  9., 14.,  2.,  8.])},\n",
       " {'P_bg0': tensor([6., 2., 7., 1., 4., 0., 5., 3.]),\n",
       "  'P_bg1': tensor([3., 1., 0., 2., 5., 4.]),\n",
       "  'P_bg2': tensor([0., 2., 1., 3.]),\n",
       "  'P_bg3': tensor([ 8., 16.,  3.,  2., 14., 13.,  7.,  4., 15., 17., 19., 11., 12.,  1.,\n",
       "           6., 18.,  5.,  0.,  9., 10.])}]"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "permdicts"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "6af939f8",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "3\n",
      "2\n"
     ]
    }
   ],
   "source": [
    "print(len(perms))\n",
    "print(len(perms[0]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "dbb477c1",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "<class 'tuple'>\n",
      "2\n",
      "tensor([ 5,  1,  2,  7,  6,  3,  4,  0,  9, 11, 10, 12, 13,  8, 16, 15, 17, 14,\n",
      "        37, 18, 26, 30, 27, 24, 25, 23, 28, 31, 19, 21, 35, 29, 22, 34, 36, 32,\n",
      "        20, 33, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47], dtype=torch.int32)\n"
     ]
    }
   ],
   "source": [
    "print(type(perms[0]))\n",
    "print(len(perms[0]))\n",
    "print(perms[0][0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "f9fe74e3",
   "metadata": {},
   "outputs": [],
   "source": [
    "def permute_model_vector(vec,idx_start,window,perm):\n",
    "    # create index vector of tokens\n",
    "    index = list(range(len(vec)))\n",
    "    \n",
    "    # apply global permutation on index\n",
    "    # using (slices of) the permuted index to access tokens equals permuting all tokens and slicing after\n",
    "    perm_glob = perm[0]\n",
    "    index = [index[idx] for idx in perm_glob]\n",
    "    \n",
    "    # slice index\n",
    "    idx_end = idx_start + window\n",
    "    index = index[idx_start:idx_end]\n",
    "    \n",
    "    # slice token sequence\n",
    "    vec = [vec[idx] for idx in index] \n",
    "        \n",
    "    # slice permutations\n",
    "    perm_loc = perm[1]\n",
    "    perm_loc = [perm_loc[idx] for idx in index]\n",
    "\n",
    "    # apply token permutation\n",
    "    vec = [vecdx[permdx] for (vecdx,permdx) in zip(vec,perm_loc)]\n",
    "    \n",
    "    # return tokens\n",
    "    return vec"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "32033871",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "<class 'collections.OrderedDict'>\n",
      "<class 'list'>\n",
      "48\n"
     ]
    }
   ],
   "source": [
    "check1 = dataset_ep_test.data[0][-1]\n",
    "print(type(check1))\n",
    "vec1 = vectorize_checkpoint(check1)\n",
    "print(type(vec1))\n",
    "print(len(vec1))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "96d1dae9",
   "metadata": {},
   "source": [
    "# Test 1: computational load - compare permutations on checkpoint with permutation on vector"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "a40adcdc",
   "metadata": {},
   "outputs": [],
   "source": [
    "import random"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "6573c702",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1.57 µs ± 30.8 ns per loop (mean ± std. dev. of 7 runs, 1,000,000 loops each)\n"
     ]
    }
   ],
   "source": [
    "%%timeit\n",
    "# baseline - how long does it take to draw a random integer\n",
    "idx = random.randint(0,len(dataset_ep_test.data))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "ee5ce77e",
   "metadata": {},
   "outputs": [],
   "source": [
    "# prepare a list of vecs\n",
    "vecs = []\n",
    "for idx in range(len(dataset_ep_test.data)):\n",
    "    checkdx = dataset_ep_test.data[idx][-1]\n",
    "    vecdx = vectorize_checkpoint(checkdx)\n",
    "    vecs.append(vecdx)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "992864ab",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "154 µs ± 312 ns per loop (mean ± std. dev. of 7 runs, 10,000 loops each)\n"
     ]
    }
   ],
   "source": [
    "%%timeit\n",
    "# compute random sliced permutations\n",
    "idx = random.randint(0,len(vecs)-1)\n",
    "perm1 = perms[0]\n",
    "vec1 = vecs[idx]\n",
    "vec2 = permute_model_vector(vec=vec1,idx_start=2,window=13,perm=perm1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 143,
   "id": "93ddebe9",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "7"
      ]
     },
     "execution_count": 143,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# comparison: permute checkpoints -> vectorize -> slice"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "30d1258a",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "616 µs ± 1.21 µs per loop (mean ± std. dev. of 7 runs, 1,000 loops each)\n"
     ]
    }
   ],
   "source": [
    "%%timeit\n",
    "\n",
    "idx = random.randint(0,len(dataset_ep_test.data)-1)\n",
    "\n",
    "check1 = dataset_ep_test.data[idx][-1]\n",
    "\n",
    "permd = permdicts[0]\n",
    "\n",
    "check_perm = apply_permutation(ps=zoo_cnn_permutation_spec(), perm=permd, params=check1)\n",
    "\n",
    "vec3 = vectorize_checkpoint(check_perm )\n",
    "\n",
    "vec4 = vec3[2:15]\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8c4a5eca",
   "metadata": {},
   "source": [
    "permutation on vectorized form appears <25% of the runtime of permutation on the checkpoint"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4e91ff96",
   "metadata": {},
   "source": [
    "# test 2: check slice of permutation ar the same to sliced permutations"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 127,
   "id": "daf50ccd",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n",
      "True\n"
     ]
    }
   ],
   "source": [
    "# check that slice of permuted model is the same as permute/slice at the same time\n",
    "\n",
    "# since all true -> that works :)\n",
    "perm1 = perms[0]\n",
    "sstart = 7\n",
    "slen = 34\n",
    "vec2 = permute_model_vector(vec=vec1,idx_start=0,window=len(vec1),perm=perm1)\n",
    "vec3 = permute_model_vector(vec=vec1,idx_start=sstart,window=slen,perm=perm1)\n",
    "\n",
    "vec23 = vec2[sstart:sstart+slen]\n",
    "for m1,m2 in zip(vec23,vec3):\n",
    "    print(torch.allclose(m1,m2))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "070a58a1",
   "metadata": {},
   "source": [
    "# test 3: test that permuted models have the same mapping"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2414a0b5",
   "metadata": {},
   "outputs": [],
   "source": [
    "# permute model with full window\n",
    "# cast back to checkpoint\n",
    "# load both checkpoint\n",
    "# verify closeness of predictions\n",
    "\n",
    "# repeat for 10 models and 15 iterations\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "45ff0f69",
   "metadata": {},
   "outputs": [],
   "source": [
    "# load config\n",
    "from shrp.models.def_net import NNmodule\n",
    "config_path = dataset_ep_test.paths[0][0].joinpath('params.json')\n",
    "config = json.load(config_path.open('r'))\n",
    "config['training::batchsize'] = 1000\n",
    "model = NNmodule(config)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "196d9923",
   "metadata": {},
   "outputs": [],
   "source": [
    "# load dataset\n",
    "imgdata = torch.load(config['dataset::dump'].replace('netscratch','netscratch2'))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "172e1c1c",
   "metadata": {},
   "outputs": [],
   "source": [
    "from shrp.datasets.def_FastTensorDataLoader import FastTensorDataLoader\n",
    "n_samples = 1000\n",
    "\n",
    "testset_1, testset_2 = torch.utils.data.random_split(\n",
    "            imgdata['testset'], [n_samples, len(imgdata['testset'])-n_samples], generator=torch.Generator().manual_seed(42))\n",
    "\n",
    "testloader_raw = torch.utils.data.DataLoader(\n",
    "    dataset=testset_1, batch_size=len(testset_1), shuffle=True\n",
    "        )\n",
    "assert testloader_raw.__len__() == 1, \"temp testloader has more than one batch\"\n",
    "for test_data, test_labels in testloader_raw:\n",
    "    pass\n",
    "\n",
    "testset = torch.utils.data.TensorDataset(test_data, test_labels)\n",
    "testloader = FastTensorDataLoader(testset)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "a846fb4d",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2023-03-15 00:06:50,151\tINFO worker.py:1538 -- Started a local Ray instance.\n"
     ]
    }
   ],
   "source": [
    "permdicts, perms = precompute_permutations(ref_checkpoint=dataset_ep_test.data[0][-1],permutation_number=15, perm_spec=zoo_cnn_permutation_spec())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "id": "9b70f1e8",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "load first checkpoint\n",
      "compute predictions of first checkpoint\n",
      "load permuted checkpoint\n",
      "compute predictions of permuted checkpoint\n",
      "check equality\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "True"
      ]
     },
     "execution_count": 36,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "check1 = dataset_ep_test.data[0][-1]\n",
    "permd = permdicts[0]\n",
    "check_perm = apply_permutation(ps=zoo_cnn_permutation_spec(), perm=permd, params=check1)\n",
    "\n",
    "print('load first checkpoint')\n",
    "model.model.load_state_dict(check1)\n",
    "print('compute predictions of first checkpoint')\n",
    "with torch.no_grad():\n",
    "    pred1 = model(test_data)\n",
    "    \n",
    "print('load permuted checkpoint')\n",
    "model.model.load_state_dict(check_perm)\n",
    "print('compute predictions of permuted checkpoint')\n",
    "with torch.no_grad():\n",
    "    pred2 = model(test_data)\n",
    "    \n",
    "print('check equality')\n",
    "torch.allclose(pred1,pred2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 45,
   "id": "e9204c7a",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 30%|███████████████████████████████████████████████▍                                                                                                              | 3/10 [00:52<02:02, 17.52s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "True\n",
      "True\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 10/10 [03:01<00:00, 18.12s/it]\n"
     ]
    }
   ],
   "source": [
    "import tqdm\n",
    "\n",
    "n_models = 10\n",
    "n_perms = 15\n",
    "\n",
    "res = []\n",
    "\n",
    "for idx in tqdm.tqdm(range(n_models)):\n",
    "    for jdx in range(n_perms):\n",
    "\n",
    "        check1 = dataset_ep_test.data[idx][-1]\n",
    "        permd = permdicts[jdx]\n",
    "        check_perm = apply_permutation(ps=zoo_cnn_permutation_spec(), perm=permd, params=check1)\n",
    "\n",
    "#         print('load first checkpoint')\n",
    "        model.model.load_state_dict(check1)\n",
    "#         print('compute predictions of first checkpoint')\n",
    "        with torch.no_grad():\n",
    "            pred1 = model(test_data)\n",
    "\n",
    "#         print('load permuted checkpoint')\n",
    "        model.model.load_state_dict(check_perm)\n",
    "#         print('compute predictions of permuted checkpoint')\n",
    "        with torch.no_grad():\n",
    "            pred2 = model(test_data)\n",
    "\n",
    "#         print('check equality')\n",
    "        restmp = torch.allclose(pred1,pred2)\n",
    "        if not restmp:\n",
    "            print(torch.allclose(pred1,pred2,atol=1e-05, rtol=1e-03))\n",
    "        res.append(restmp)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "id": "e864ceb4",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.9866666666666667\n"
     ]
    }
   ],
   "source": [
    "print(sum(res)/len(res))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b33d89ee",
   "metadata": {},
   "outputs": [],
   "source": [
    "# test the same on vector level"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "id": "6f633df7",
   "metadata": {},
   "outputs": [],
   "source": [
    "# prepare a list of vecs\n",
    "vecs = []\n",
    "for idx in range(len(dataset_ep_test.data)):\n",
    "    checkdx = dataset_ep_test.data[idx][-1]\n",
    "    vecdx = vectorize_checkpoint(checkdx)\n",
    "    vecs.append(vecdx)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "id": "b9f07845",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "load first checkpoint\n",
      "compute predictions of first checkpoint\n",
      "load permuted checkpoint\n",
      "compute predictions of permuted checkpoint\n",
      "check equality\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "True"
      ]
     },
     "execution_count": 47,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "vec1 = vecs[0]\n",
    "perm1 = perms[0]\n",
    "\n",
    "vecperm = permute_model_vector(vec=vec1,idx_start=0,window=len(vec1),perm=perm1)\n",
    "\n",
    "check1 = vector_to_checkpoint(vector=vec1, reference_checkpoint=check1)\n",
    "check_perm = vector_to_checkpoint(vector=vecperm, reference_checkpoint=check1)\n",
    "\n",
    "print('load first checkpoint')\n",
    "model.model.load_state_dict(check1)\n",
    "print('compute predictions of first checkpoint')\n",
    "with torch.no_grad():\n",
    "    pred1 = model(test_data)\n",
    "    \n",
    "print('load permuted checkpoint')\n",
    "model.model.load_state_dict(check_perm)\n",
    "print('compute predictions of permuted checkpoint')\n",
    "with torch.no_grad():\n",
    "    pred2 = model(test_data)\n",
    "    \n",
    "print('check equality')\n",
    "torch.allclose(pred1,pred2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "id": "44257999",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 30%|███████████████████████████████████████████████▍                                                                                                              | 3/10 [00:55<02:10, 18.58s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "True\n",
      "True\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 10/10 [03:04<00:00, 18.49s/it]\n"
     ]
    }
   ],
   "source": [
    "import tqdm\n",
    "\n",
    "n_models = 10\n",
    "n_perms = 15\n",
    "\n",
    "res = []\n",
    "\n",
    "for idx in tqdm.tqdm(range(n_models)):\n",
    "    for jdx in range(n_perms):\n",
    "\n",
    "        vec1 = vecs[idx]\n",
    "        perm1 = perms[jdx]\n",
    "\n",
    "        vecperm = permute_model_vector(vec=vec1,idx_start=0,window=len(vec1),perm=perm1)\n",
    "\n",
    "        check1 = vector_to_checkpoint(vector=vec1, reference_checkpoint=check1)\n",
    "        check_perm = vector_to_checkpoint(vector=vecperm, reference_checkpoint=check1)\n",
    "\n",
    "#         print('load first checkpoint')\n",
    "        model.model.load_state_dict(check1)\n",
    "#         print('compute predictions of first checkpoint')\n",
    "        with torch.no_grad():\n",
    "            pred1 = model(test_data)\n",
    "\n",
    "#         print('load permuted checkpoint')\n",
    "        model.model.load_state_dict(check_perm)\n",
    "#         print('compute predictions of permuted checkpoint')\n",
    "        with torch.no_grad():\n",
    "            pred2 = model(test_data)\n",
    "\n",
    "#         print('check equality')\n",
    "        restmp = torch.allclose(pred1,pred2)\n",
    "        if not restmp:\n",
    "            print(torch.allclose(pred1,pred2,atol=1e-05, rtol=1e-03))\n",
    "        res.append(restmp)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 52,
   "id": "7136fdd8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[ 1.,  2.,  3.],\n",
       "        [21., 22., 23.],\n",
       "        [31., 32., 33.]])"
      ]
     },
     "execution_count": 52,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "lst = [\n",
    "    torch.tensor([1,2,3]),\n",
    "    torch.tensor([21,22,23]),\n",
    "    torch.tensor([31,32,33]),\n",
    "]\n",
    "\n",
    "tst = torch.stack(lst,dim=0).float()\n",
    "tst"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 53,
   "id": "d9024b1b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(18.6667)"
      ]
     },
     "execution_count": 53,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.mean(tst)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 54,
   "id": "26ddadd0",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor(13.2571)"
      ]
     },
     "execution_count": 54,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.std(tst)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "25850cca",
   "metadata": {},
   "outputs": [],
   "source": [
    "# test dataset with standardization, permutation, map to canonical\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "726c6b56",
   "metadata": {},
   "outputs": [],
   "source": [
    "# dataset_base\n",
    "\n",
    "import logging\n",
    "# logging.basicConfig(level=logging.INFO)\n",
    "logging.basicConfig(level=logging.DEBUG)\n",
    "\n",
    "from pathlib import Path\n",
    "import json\n",
    "from shrp.datasets.dataset_simclr import SimCLRDataset\n",
    "\n",
    "from shrp.git_re_basin.git_re_basin import (\n",
    "    PermutationSpec,\n",
    "    zoo_cnn_permutation_spec,\n",
    "    weight_matching,\n",
    "    apply_permutation,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "72b9b91a",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2023-03-18 00:13:11,412\tINFO worker.py:1538 -- Started a local Ray instance.\n",
      "INFO:root:loading checkpoints from [PosixPath('/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform')]\n",
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 30/30 [00:03<00:00,  9.11it/s]\n",
      "INFO:root:Data loaded. found 10 usable samples out of potential 30 samples.\n",
      "INFO:root:Load properties for samples from paths.\n",
      "INFO:root:### load data for dict_keys(['test_acc', 'training_iteration', 'ggap'])\n",
      "10it [00:00, 13.75it/s]\n",
      "INFO:root:Properties loaded.\n",
      "INFO:root:prepare canonical form\n",
      "2023-03-18 00:13:22,034\tINFO worker.py:1538 -- Started a local Ray instance.\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "preparing computing canon form...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO:root:init dataset length\n",
      "INFO:root:Get Positions\n",
      "INFO:root:vectorize data\n",
      "INFO:root:Get layer mapping\n",
      "DEBUG:root:layer mapping: {'0': {'start_idx': 0, 'end_idx': 7}, '1': {'start_idx': 8, 'end_idx': 13}, '2': {'start_idx': 14, 'end_idx': 17}, '3': {'start_idx': 18, 'end_idx': 37}, '4': {'start_idx': 38, 'end_idx': 47}}\n",
      "INFO:root:Get layer-wise mean and std\n",
      "INFO:root:Apply standardization\n",
      "INFO:root:Discover tokensize\n"
     ]
    }
   ],
   "source": [
    "config_key_list = []\n",
    "result_key_list = [\n",
    "    \"test_acc\",\n",
    "    \"training_iteration\",\n",
    "    \"ggap\",\n",
    "#     \"sparsity_ratio\",\n",
    "]\n",
    "property_keys = {\n",
    "    \"result_keys\": result_key_list,\n",
    "    \"config_keys\": config_key_list,\n",
    "}\n",
    "\n",
    "# path_root = Path('/netscratch2/dtaskiran/zoos/MNIST/tune_zoo_mnist_uniform/')\n",
    "path_root = Path('/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/')\n",
    "# path_root = Path('/netscratch2/kschuerholt/code/versai/model_zoos/zoos/CIFAR10/resnet19/kaiming_uniform/tune_zoo_cifar10_resnet18_kaiming_uniform')\n",
    "\n",
    "dataset_ep_test = SimCLRDataset(\n",
    "    root=path_root,\n",
    "    epoch_lst=[1,5,25],\n",
    "#     mode=\"checkpoint\",\n",
    "    mode=\"vector\",\n",
    "    permutations_number=0,\n",
    "    permutation_spec = zoo_cnn_permutation_spec(),\n",
    "#     view_1_canonical = False,\n",
    "    view_1_canonical = False,\n",
    "    view_2_canonical = True,\n",
    "#     view_2_canonical = False,\n",
    "    add_noise_view_1 = 0.1,  # [('input', 0.15), ('output', 0.013)]\n",
    "    add_noise_view_2 = 0.0,  # [('input', 0.15), ('output', 0.013)]\n",
    "    noise_multiplicative=True,\n",
    "    erase_augment=None,  # {\"p\": 0.5,\"scale\":(0.02,0.33),\"value\":0,\"mode\":\"block\"}\n",
    "    windowsize = 12,\n",
    "    standardize=True,\n",
    "    train_val_test=\"train\",  # determines whcih dataset split to use\n",
    "    ds_split=[0.7, 0.15,0.15],  #\n",
    "    max_samples=15,\n",
    "    weight_threshold=float(\"inf\"),\n",
    "    filter_function=None,  # gets sample path as argument and returns True if model needs to be filtered out\n",
    "    property_keys=property_keys,\n",
    "    num_threads=4,\n",
    "    shuffle_path=True,\n",
    "    verbosity=3,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "d46bc894",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Checks\n",
    "# -[x] All options run without error\n",
    "# -[x] NAN / INFS\n",
    "# -[x] run permutation checks from above again\n",
    "# -[x] check lenghts of windows\n",
    "# -[x] check standardization worsk\n",
    "# -[x] check noising works\n",
    "# -[x] check augmentations are applied correctly\n",
    "# -[x] check relation of all datapoints is preserved\n",
    "# -[x] check positions\n",
    "\n",
    "# =[]comment all code \n",
    "# =[]code review while commenting\n",
    "# =[]load models, compute test_acc, compare to recorded one"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "a4791954",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "30"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "dataset_ep_test.__len__()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "d41cd090",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "10"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(dataset_ep_test.data)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "74696f9e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "3"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(dataset_ep_test.data[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "fbd1df22",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "id": "a947cbe7",
   "metadata": {},
   "source": [
    "### test noise "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "id": "54bdb2fb",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([ 0.6178,  1.0873,  0.0407,  0.0093,  0.5361, -0.5169,  0.6908, -0.2309,\n",
      "        -0.3285,  0.1974,  0.4596,  0.4653, -0.0077,  0.2002,  0.4147, -0.5666,\n",
      "         1.7816,  1.4433,  0.5817,  1.0785,  0.2645,  0.6473, -0.2172,  0.7955,\n",
      "        -0.1908])\n",
      "tensor([ 0.8240,  0.9750,  0.0380,  0.0066,  0.4346, -0.5247,  0.6876, -0.2180,\n",
      "        -0.2840,  0.1956,  0.4612,  0.4618, -0.0086,  0.2470,  0.3720, -0.5622,\n",
      "         1.3629,  1.6588,  0.5280,  1.1187,  0.2334,  0.5823, -0.2271,  0.7855,\n",
      "        -0.1828])\n",
      "tensor([ 0.7212,  1.0149,  0.0400,  0.0066,  0.4646, -0.4545,  0.6207, -0.2456,\n",
      "        -0.3473,  0.1681,  0.4642,  0.3805, -0.0081,  0.2214,  0.4619, -0.5562,\n",
      "         1.5033,  1.5660,  0.6301,  1.1723,  0.2358,  0.6135, -0.2521,  0.8279,\n",
      "        -0.1709])\n",
      "tensor([ 0.7916,  1.1427,  0.0344,  0.0080,  0.4719, -0.4345,  0.7106, -0.2521,\n",
      "        -0.2859,  0.1705,  0.4819,  0.3783, -0.0058,  0.1908,  0.3508, -0.3702,\n",
      "         1.7695,  1.4537,  0.6129,  1.0355,  0.2838,  0.6718, -0.2450,  0.8243,\n",
      "        -0.1934])\n",
      "tensor([ 0.7389,  1.1845,  0.0446,  0.0081,  0.4063, -0.4910,  0.7072, -0.2420,\n",
      "        -0.2235,  0.2006,  0.4358,  0.4096, -0.0072,  0.2246,  0.4255, -0.4792,\n",
      "         1.5815,  1.7055,  0.5801,  1.1538,  0.2635,  0.5545, -0.2333,  0.7678,\n",
      "        -0.1939])\n"
     ]
    }
   ],
   "source": [
    "dataset_ep_test.set_module_window(len(dataset_ep_test.data[0][-1]))\n",
    "index = 21\n",
    "for _ in range(5):\n",
    "    tw,_,_,_,pw = dataset_ep_test.__getitem__(index)\n",
    "    print(tw[15])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "db8b8bc0",
   "metadata": {},
   "source": [
    "seems to work as expected "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "id": "f5c98392",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00548_548_seed=549_2021-07-27_15-51-15#_#epoch_1\n",
      "0.18335125998770743 - 1\n",
      "/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00935_935_seed=936_2021-07-28_08-01-46#_#epoch_5\n",
      "0.23044714197910265 - 5\n",
      "/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00257_257_seed=258_2021-07-27_03-43-04#_#epoch_1\n",
      "0.19817916410571604 - 1\n",
      "/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00786_786_seed=787_2021-07-28_01-50-08#_#epoch_5\n",
      "0.21838506453595574 - 5\n",
      "/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00321_321_seed=322_2021-07-27_06-12-40#_#epoch_5\n",
      "0.21842347879532883 - 5\n",
      "/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00935_935_seed=936_2021-07-28_08-01-46#_#epoch_25\n",
      "0.7004456054087277 - 25\n",
      "/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00935_935_seed=936_2021-07-28_08-01-46#_#epoch_5\n",
      "0.23044714197910265 - 5\n",
      "/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00599_599_seed=600_2021-07-27_18-04-45#_#epoch_1\n",
      "0.1958358942839582 - 1\n",
      "/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00827_827_seed=828_2021-07-28_03-34-52#_#epoch_1\n",
      "0.19564382298709282 - 1\n",
      "/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00689_689_seed=690_2021-07-27_21-54-40#_#epoch_5\n",
      "0.1958743085433313 - 5\n",
      "/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00548_548_seed=549_2021-07-27_15-51-15#_#epoch_1\n",
      "0.18335125998770743 - 1\n",
      "/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00548_548_seed=549_2021-07-27_15-51-15#_#epoch_1\n",
      "0.18335125998770743 - 1\n",
      "/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00689_689_seed=690_2021-07-27_21-54-40#_#epoch_1\n",
      "0.1977181929932391 - 1\n",
      "/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00321_321_seed=322_2021-07-27_06-12-40#_#epoch_1\n",
      "0.19652735095267362 - 1\n",
      "/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00599_599_seed=600_2021-07-27_18-04-45#_#epoch_5\n",
      "0.21231561155500922 - 5\n"
     ]
    }
   ],
   "source": [
    "### check relation - checkpoint - properties\n",
    "for _ in range(15):\n",
    "    index = random.randint(0,len(dataset_ep_test)-1)\n",
    "    _,l,_,_,pw = dataset_ep_test.__getitem__(index)\n",
    "    mdx,edx = dataset_ep_test._index[index]\n",
    "    props = dataset_ep_test.properties\n",
    "    print(l)\n",
    "    print(f\"{props['test_acc'][mdx][edx]} - {props['training_iteration'][mdx][edx]}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c0cda91a",
   "metadata": {},
   "source": [
    "### check positions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "e02930a0",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[(0, 0),\n",
       " (0, 1),\n",
       " (0, 2),\n",
       " (0, 3),\n",
       " (0, 4),\n",
       " (0, 5),\n",
       " (0, 6),\n",
       " (0, 7),\n",
       " (1, 0),\n",
       " (1, 1),\n",
       " (1, 2),\n",
       " (1, 3),\n",
       " (1, 4),\n",
       " (1, 5),\n",
       " (2, 0),\n",
       " (2, 1),\n",
       " (2, 2),\n",
       " (2, 3),\n",
       " (3, 0),\n",
       " (3, 1),\n",
       " (3, 2),\n",
       " (3, 3),\n",
       " (3, 4),\n",
       " (3, 5),\n",
       " (3, 6),\n",
       " (3, 7),\n",
       " (3, 8),\n",
       " (3, 9),\n",
       " (3, 10),\n",
       " (3, 11),\n",
       " (3, 12),\n",
       " (3, 13),\n",
       " (3, 14),\n",
       " (3, 15),\n",
       " (3, 16),\n",
       " (3, 17),\n",
       " (3, 18),\n",
       " (3, 19),\n",
       " (4, 0),\n",
       " (4, 1),\n",
       " (4, 2),\n",
       " (4, 3),\n",
       " (4, 4),\n",
       " (4, 5),\n",
       " (4, 6),\n",
       " (4, 7),\n",
       " (4, 8),\n",
       " (4, 9)]"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "dataset_ep_test.positions"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "62c0c3e3",
   "metadata": {},
   "source": [
    "### check standardization\n",
    "seems to work well"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "172b8261",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'0': {'start_idx': 0,\n",
       "  'end_idx': 7,\n",
       "  'mean': tensor(0.3796),\n",
       "  'std': tensor(0.7530)},\n",
       " '1': {'start_idx': 8,\n",
       "  'end_idx': 13,\n",
       "  'mean': tensor(0.3528),\n",
       "  'std': tensor(0.7909)},\n",
       " '2': {'start_idx': 14,\n",
       "  'end_idx': 17,\n",
       "  'mean': tensor(0.3028),\n",
       "  'std': tensor(0.5307)},\n",
       " '3': {'start_idx': 18,\n",
       "  'end_idx': 37,\n",
       "  'mean': tensor(0.3382),\n",
       "  'std': tensor(0.5479)},\n",
       " '4': {'start_idx': 38,\n",
       "  'end_idx': 47,\n",
       "  'mean': tensor(0.4690),\n",
       "  'std': tensor(0.5501)}}"
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "dataset_ep_test.layers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "id": "9021ec16",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "10\n",
      "30\n",
      "mean: 1.6513841405441099e-09| std: 0.999972939491272\n",
      "nans: False | infs: False\n"
     ]
    }
   ],
   "source": [
    "# stack all components\n",
    "\n",
    "data = dataset_ep_test.data\n",
    "print(len(data))\n",
    "data = [ddx for idx in range(len(data)) for ddx in data[idx]]\n",
    "print(len(data))\n",
    "data = [torch.cat(ddx,dim=0)for ddx in data]\n",
    "data = torch.stack(data,dim=0)\n",
    "data.shape\n",
    "\n",
    "print(f'mean: {torch.mean(data)}| std: {torch.std(data)}')\n",
    "print(f'nans: {torch.isnan(data).any()} | infs: {torch.isinf(data).any()}')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4a7c55be",
   "metadata": {},
   "source": [
    "### check permutations (again)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "9ea077d5",
   "metadata": {},
   "outputs": [],
   "source": [
    "# dataset_base\n",
    "\n",
    "import logging\n",
    "logging.basicConfig(level=logging.INFO)\n",
    "\n",
    "from pathlib import Path\n",
    "import json\n",
    "from shrp.datasets.dataset_simclr import SimCLRDataset\n",
    "\n",
    "from shrp.git_re_basin.git_re_basin import (\n",
    "    PermutationSpec,\n",
    "    zoo_cnn_permutation_spec,\n",
    "    weight_matching,\n",
    "    apply_permutation,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "id": "58aceb76",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2023-03-16 22:11:47,569\tINFO worker.py:1538 -- Started a local Ray instance.\n",
      "INFO:root:loading checkpoints from [PosixPath('/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform')]\n",
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 105/105 [00:02<00:00, 43.87it/s]\n",
      "INFO:root:Data loaded. found 35 usable samples out of potential 105 samples.\n",
      "INFO:root:Load properties for samples from paths.\n",
      "INFO:root:### load data for dict_keys(['test_acc', 'training_iteration', 'ggap'])\n",
      "35it [00:00, 80.65it/s]\n",
      "INFO:root:Properties loaded.\n",
      "INFO:root:prepare canonical form\n",
      "2023-03-16 22:11:56,780\tINFO worker.py:1538 -- Started a local Ray instance.\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "preparing computing canon form...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO:root:init dataset length\n",
      "INFO:root:Get Positions\n",
      "INFO:root:vectorize data\n",
      "INFO:root:init permutations\n",
      "INFO:root:start precomputing permutations\n",
      "INFO:root:0/P_bg3: 0.0\n",
      "INFO:root:0/P_bg0: 0.0\n",
      "INFO:root:0/P_bg2: 0.0\n",
      "INFO:root:0/P_bg1: 0.0\n",
      "INFO:root:get random permutation dicts\n",
      "INFO:root:get permutation indices\n",
      "2023-03-16 22:12:06,684\tINFO worker.py:1538 -- Started a local Ray instance.\n"
     ]
    }
   ],
   "source": [
    "config_key_list = []\n",
    "result_key_list = [\n",
    "    \"test_acc\",\n",
    "    \"training_iteration\",\n",
    "    \"ggap\",\n",
    "#     \"sparsity_ratio\",\n",
    "]\n",
    "property_keys = {\n",
    "    \"result_keys\": result_key_list,\n",
    "    \"config_keys\": config_key_list,\n",
    "}\n",
    "\n",
    "# path_root = Path('/netscratch2/dtaskiran/zoos/MNIST/tune_zoo_mnist_uniform/')\n",
    "path_root = Path('/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/')\n",
    "# path_root = Path('/netscratch2/kschuerholt/code/versai/model_zoos/zoos/CIFAR10/resnet19/kaiming_uniform/tune_zoo_cifar10_resnet18_kaiming_uniform')\n",
    "\n",
    "dataset_ep_test = SimCLRDataset(\n",
    "    root=path_root,\n",
    "    epoch_lst=[1,5,25],\n",
    "#     mode=\"checkpoint\",\n",
    "    mode=\"vector\",\n",
    "    permutations_number=20,\n",
    "    permutation_spec = zoo_cnn_permutation_spec(),\n",
    "    view_1_canonical = False,\n",
    "    view_2_canonical = True,\n",
    "#     view_2_canonical = False,\n",
    "    add_noise_view_1 = 0.0,  # [('input', 0.15), ('output', 0.013)]\n",
    "    add_noise_view_2 = 0.0,  # [('input', 0.15), ('output', 0.013)]\n",
    "    erase_augment=None,  # {\"p\": 0.5,\"scale\":(0.02,0.33),\"value\":0,\"mode\":\"block\"}\n",
    "#     windowsize = 12,\n",
    "    standardize=False,\n",
    "    train_val_test=\"train\",  # determines whcih dataset split to use\n",
    "    ds_split=[0.7, 0.15,0.15],  #\n",
    "    max_samples=50,\n",
    "    weight_threshold=float(\"inf\"),\n",
    "    filter_function=None,  # gets sample path as argument and returns True if model needs to be filtered out\n",
    "    property_keys=property_keys,\n",
    "    num_threads=4,\n",
    "    shuffle_path=True,\n",
    "    verbosity=3,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 50,
   "id": "70e3c0ad",
   "metadata": {},
   "outputs": [],
   "source": [
    "dataset_ep_test.set_module_window(len(dataset_ep_test.data[0][-1]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "c66336eb",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO:root:cuda unavailable:: train model on cpu\n",
      "INFO:root:=> creating model CNN\n",
      "INFO:root:initialze model\n"
     ]
    }
   ],
   "source": [
    "# load config\n",
    "import torch\n",
    "from shrp.models.def_net import NNmodule\n",
    "config_path = dataset_ep_test.paths[0][0].joinpath('params.json')\n",
    "config = json.load(config_path.open('r'))\n",
    "config['training::batchsize'] = 1000\n",
    "model = NNmodule(config)\n",
    "# load dataset\n",
    "imgdata = torch.load(config['dataset::dump'].replace('netscratch','netscratch2'))\n",
    "from shrp.datasets.def_FastTensorDataLoader import FastTensorDataLoader\n",
    "n_samples = 1000\n",
    "\n",
    "testset_1, testset_2 = torch.utils.data.random_split(\n",
    "            imgdata['testset'], [n_samples, len(imgdata['testset'])-n_samples], generator=torch.Generator().manual_seed(42))\n",
    "\n",
    "testloader_raw = torch.utils.data.DataLoader(\n",
    "    dataset=testset_1, batch_size=len(testset_1), shuffle=True\n",
    "        )\n",
    "assert testloader_raw.__len__() == 1, \"temp testloader has more than one batch\"\n",
    "for test_data, test_labels in testloader_raw:\n",
    "    pass\n",
    "\n",
    "testset = torch.utils.data.TensorDataset(test_data, test_labels)\n",
    "testloader = FastTensorDataLoader(testset)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "03cb6cc0",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|                                                                                                                                                                      | 0/10 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "True\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 20%|███████████████████████████████▌                                                                                                                              | 2/10 [01:35<06:18, 47.32s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "True\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 30%|███████████████████████████████████████████████▍                                                                                                              | 3/10 [02:22<05:32, 47.53s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "True\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|███████████████████████████████████████████████████████████████▏                                                                                              | 4/10 [03:11<04:47, 47.99s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "True\n",
      "True\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 50%|███████████████████████████████████████████████████████████████████████████████                                                                               | 5/10 [04:03<04:07, 49.44s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "True\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 80%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████▍                               | 8/10 [06:24<01:35, 47.89s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "True\n",
      "True\n",
      "True\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 90%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████▏               | 9/10 [07:17<00:49, 49.68s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "True\n",
      "True\n",
      "True\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 10/10 [08:10<00:00, 49.08s/it]\n"
     ]
    }
   ],
   "source": [
    "import tqdm\n",
    "import random\n",
    "from shrp.datasets.dataset_simclr import vector_to_checkpoint\n",
    "\n",
    "n_models = 10\n",
    "n_perms = 15\n",
    "\n",
    "res = []\n",
    "\n",
    "check = dataset_ep_test.reference_checkpoint\n",
    "for idx in tqdm.tqdm(range(n_models)):\n",
    "        \n",
    "    for jdx in range(n_perms):\n",
    "        \n",
    "        edx = random.randint(0,2)\n",
    "        jdx = idx*3+edx\n",
    "        vec1,lab1,_,_,_ = dataset_ep_test.__getitem__(jdx)\n",
    "        vec2,lab2,_,_,_ = dataset_ep_test.__getitem__(jdx)\n",
    "\n",
    "        check1 = vector_to_checkpoint(vector=vec1, reference_checkpoint=check)\n",
    "        check2 = vector_to_checkpoint(vector=vec2, reference_checkpoint=check)\n",
    "\n",
    "#         print('load first checkpoint')\n",
    "        model.model.load_state_dict(check1)\n",
    "#         print('compute predictions of first checkpoint')\n",
    "        with torch.no_grad():\n",
    "            pred1 = model(test_data)\n",
    "\n",
    "#         print('load permuted checkpoint')\n",
    "        model.model.load_state_dict(check2)\n",
    "#         print('compute predictions of permuted checkpoint')\n",
    "        with torch.no_grad():\n",
    "            pred2 = model(test_data)\n",
    "\n",
    "#         print('check equality')\n",
    "        restmp = torch.allclose(pred1,pred2)\n",
    "        if not restmp:\n",
    "            print(torch.allclose(pred1,pred2,atol=1e-05, rtol=1e-03))\n",
    "        res.append(restmp)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3819829f",
   "metadata": {},
   "source": [
    "No errors, all close enough with relaxed tolerances. Permutations still work "
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f20f47d2",
   "metadata": {},
   "source": [
    "### compare recorded to computed accuracy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 55,
   "id": "85a9ecbd",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|                                                                                                                                                                      | 0/10 [00:00<?, ?it/s]INFO:root:validate at epoch 0\n",
      "INFO:root:test ::: loss: 2.180163366317749; accuracy: 0.213\n",
      "INFO:root:validate at epoch 0\n",
      "INFO:root:test ::: loss: 2.180163356781006; accuracy: 0.213\n",
      " 10%|███████████████▊                                                                                                                                              | 1/10 [01:25<12:50, 85.60s/it]INFO:root:validate at epoch 0\n",
      "INFO:root:test ::: loss: 1.1837517328262328; accuracy: 0.614\n",
      "INFO:root:validate at epoch 0\n",
      "INFO:root:test ::: loss: 1.1837517595291138; accuracy: 0.614\n",
      " 20%|███████████████████████████████▌                                                                                                                              | 2/10 [02:29<09:44, 73.01s/it]INFO:root:validate at epoch 0\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "acc_1: 0.614 | acc_2: 0.614 | acc_rec: 0.6262676705593117\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO:root:test ::: loss: 2.2183909492492675; accuracy: 0.197\n",
      "INFO:root:validate at epoch 0\n",
      "INFO:root:test ::: loss: 2.2183907890319823; accuracy: 0.197\n",
      " 30%|███████████████████████████████████████████████▍                                                                                                              | 3/10 [03:47<08:46, 75.24s/it]INFO:root:validate at epoch 0\n",
      "INFO:root:test ::: loss: 2.0843050088882444; accuracy: 0.276\n",
      "INFO:root:validate at epoch 0\n",
      "INFO:root:test ::: loss: 2.0843049564361573; accuracy: 0.276\n",
      " 40%|███████████████████████████████████████████████████████████████▏                                                                                              | 4/10 [05:23<08:20, 83.48s/it]INFO:root:validate at epoch 0\n",
      "INFO:root:test ::: loss: 2.2148335647583006; accuracy: 0.198\n",
      "INFO:root:validate at epoch 0\n",
      "INFO:root:test ::: loss: 2.2148336029052733; accuracy: 0.198\n",
      " 50%|███████████████████████████████████████████████████████████████████████████████                                                                               | 5/10 [06:21<06:11, 74.33s/it]INFO:root:validate at epoch 0\n",
      "INFO:root:test ::: loss: 2.1776208724975588; accuracy: 0.223\n",
      "INFO:root:validate at epoch 0\n",
      "INFO:root:test ::: loss: 2.177620843887329; accuracy: 0.223\n",
      " 60%|██████████████████████████████████████████████████████████████████████████████████████████████▊                                                               | 6/10 [07:26<04:43, 70.98s/it]INFO:root:validate at epoch 0\n",
      "INFO:root:test ::: loss: 1.1041694135665894; accuracy: 0.653\n",
      "INFO:root:validate at epoch 0\n",
      "INFO:root:test ::: loss: 1.1041693668365478; accuracy: 0.653\n",
      " 70%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████▌                                               | 7/10 [09:09<04:04, 81.48s/it]INFO:root:validate at epoch 0\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "acc_1: 0.653 | acc_2: 0.653 | acc_rec: 0.6694068838352797\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO:root:test ::: loss: 2.191834867477417; accuracy: 0.219\n",
      "INFO:root:validate at epoch 0\n",
      "INFO:root:test ::: loss: 2.1918348865509034; accuracy: 0.219\n",
      " 80%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████▍                               | 8/10 [10:24<02:38, 79.29s/it]INFO:root:validate at epoch 0\n",
      "INFO:root:test ::: loss: 1.1670266799926758; accuracy: 0.649\n",
      "INFO:root:validate at epoch 0\n",
      "INFO:root:test ::: loss: 1.167026635169983; accuracy: 0.649\n",
      " 90%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████▏               | 9/10 [11:24<01:13, 73.54s/it]INFO:root:validate at epoch 0\n",
      "INFO:root:test ::: loss: 2.109017961502075; accuracy: 0.292\n",
      "INFO:root:validate at epoch 0\n",
      "INFO:root:test ::: loss: 2.1090178623199463; accuracy: 0.292\n",
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 10/10 [12:52<00:00, 77.21s/it]\n"
     ]
    }
   ],
   "source": [
    "import tqdm\n",
    "import random\n",
    "from shrp.datasets.dataset_simclr import vector_to_checkpoint\n",
    "\n",
    "repetitions = 10\n",
    "\n",
    "res = []\n",
    "\n",
    "check = dataset_ep_test.reference_checkpoint\n",
    "for idx in tqdm.tqdm(range(repetitions)):\n",
    "        \n",
    "    kdx = random.randint(0,len(dataset_ep_test)-1)\n",
    "    mdx,edx = dataset_ep_test._index[kdx]\n",
    "    vec1,lab1,vec2,_,_ = dataset_ep_test.__getitem__(kdx)\n",
    "\n",
    "    check1 = vector_to_checkpoint(vector=vec1, reference_checkpoint=check)\n",
    "    check2 = vector_to_checkpoint(vector=vec2, reference_checkpoint=check)\n",
    "\n",
    "#         print('load first checkpoint')\n",
    "    model.model.load_state_dict(check1)\n",
    "#         print('compute predictions of first checkpoint')\n",
    "    with torch.no_grad():\n",
    "        loss_1,acc_1 = model.test_epoch(testloader,epoch=0)\n",
    "\n",
    "#         print('load permuted checkpoint')\n",
    "    model.model.load_state_dict(check2)\n",
    "#         print('compute predictions of permuted checkpoint')\n",
    "    with torch.no_grad():\n",
    "        loss_2,acc_2 = model.test_epoch(testloader,epoch=0)\n",
    "\n",
    "#         print('check equality')\n",
    "\n",
    "    acc_recorded = dataset_ep_test.properties['test_acc'][mdx][edx]\n",
    "    restmp_pair = torch.allclose(torch.tensor(acc_1),torch.tensor(acc_2),atol=1e-02, rtol=1e-03)\n",
    "    restmp_base = torch.allclose(torch.tensor(acc_1),torch.tensor(acc_recorded),atol=1e-02, rtol=1e-03)\n",
    "    if not (restmp_pair and restmp_base):\n",
    "        print(f'acc_1: {acc_1} | acc_2: {acc_2} | acc_rec: {acc_recorded}')\n",
    "    res.append(restmp)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 59,
   "id": "aafefb6c",
   "metadata": {},
   "outputs": [],
   "source": [
    "t1 = torch.randn(3)\n",
    "t2 = torch.randn(5)\n",
    "t3 = torch.randn(4)\n",
    "\n",
    "t = [t1,t2,t3]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 60,
   "id": "8d7268b9",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[tensor([0.3595, 0.7960, 0.8118]),\n",
       " tensor([0.8026, 0.0560, 0.8345, 1.0049, 1.4833]),\n",
       " tensor([-0.0634,  2.2114, -0.4224, -0.2023])]"
      ]
     },
     "execution_count": 60,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "t"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 62,
   "id": "b9de9b77",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[0., 0., 0., 0., 0.],\n",
       "        [0., 0., 0., 0., 0.],\n",
       "        [0., 0., 0., 0., 0.]])"
      ]
     },
     "execution_count": 62,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "out = torch.zeros(len(t),5)\n",
    "out"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 81,
   "id": "ccf1bd11",
   "metadata": {},
   "outputs": [],
   "source": [
    "from typing import List\n",
    "\n",
    "\n",
    "def tokenize(t:List[torch.tensor],tokensize:int)->torch.tensor:\n",
    "    \"\"\"\n",
    "    transforms list of tokens of differen lenght to tensor\n",
    "    input t: List[torch.tensor]: list of 1d input tokens of different lenghts\n",
    "    input tokensize: int output dimension of each token\n",
    "    return out: torch.tensor\n",
    "    \"\"\"\n",
    "    # init output with zeros\n",
    "    tokens = torch.zeros(len(t),tokensize)\n",
    "    mask = torch.zeros(len(t),tokensize)\n",
    "    # iterate over inputs\n",
    "    for idx,tdx in enumerate(t):\n",
    "        # get end of token, either the length of the input or them maximum length\n",
    "        tdx_end = min(tdx.shape[0],tokensize)\n",
    "        # put at position idx\n",
    "        tokens[idx,:tdx_end] = tdx[:tdx_end]\n",
    "        mask[idx,:tdx_end] = torch.ones(tdx_end)\n",
    "        \n",
    "    # return\n",
    "    return tokens,mask\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 82,
   "id": "75a0d4df",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([[ 0.3595,  0.7960,  0.8118,  0.0000,  0.0000,  0.0000],\n",
      "        [ 0.8026,  0.0560,  0.8345,  1.0049,  1.4833,  0.0000],\n",
      "        [-0.0634,  2.2114, -0.4224, -0.2023,  0.0000,  0.0000]])\n",
      "tensor([[1., 1., 1., 0., 0., 0.],\n",
      "        [1., 1., 1., 1., 1., 0.],\n",
      "        [1., 1., 1., 1., 0., 0.]])\n"
     ]
    }
   ],
   "source": [
    "tt,mt = tokenize(t,tokensize=6)\n",
    "print(tt)\n",
    "print(mt)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 71,
   "id": "3d1600fc",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([[ 0.3595,  0.7960,  0.8118,  0.0000,  0.0000],\n",
       "        [ 0.8026,  0.0560,  0.8345,  1.0049,  1.4833],\n",
       "        [-0.0634,  2.2114, -0.4224, -0.2023,  0.0000]])"
      ]
     },
     "execution_count": 71,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": []
  },
  {
   "cell_type": "markdown",
   "id": "e42fc0e8",
   "metadata": {},
   "source": [
    "# test position embeddings"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "1dc4ee24",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "testloader = torch.utils.data.DataLoader(dataset_ep_test,batch_size=5)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "8d6d2f31",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([5, 12, 201])\n",
      "torch.Size([5, 12, 201])\n",
      "('/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00321_321_seed=322_2021-07-27_06-12-40#_#epoch_1', '/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00321_321_seed=322_2021-07-27_06-12-40#_#epoch_5', '/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00321_321_seed=322_2021-07-27_06-12-40#_#epoch_25', '/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00689_689_seed=690_2021-07-27_21-54-40#_#epoch_1', '/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00689_689_seed=690_2021-07-27_21-54-40#_#epoch_5')\n",
      "torch.Size([5, 12, 201])\n",
      "torch.Size([5, 12, 201])\n",
      "('/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00321_321_seed=322_2021-07-27_06-12-40#_#epoch_1#_#canon', '/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00321_321_seed=322_2021-07-27_06-12-40#_#epoch_5#_#canon', '/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00321_321_seed=322_2021-07-27_06-12-40#_#epoch_25#_#canon', '/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00689_689_seed=690_2021-07-27_21-54-40#_#epoch_1#_#canon', '/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00689_689_seed=690_2021-07-27_21-54-40#_#epoch_5#_#canon')\n",
      "torch.Size([5, 12, 2])\n"
     ]
    }
   ],
   "source": [
    "w1,m1,l1,w2,m2,l2,p = next(iter(testloader))\n",
    "print(w1.shape)\n",
    "print(m1.shape)\n",
    "print(l1)\n",
    "print(w2.shape)\n",
    "print(m2.shape)\n",
    "print(l2)\n",
    "print(p.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "c9cf2d1f",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch.nn as nn\n",
    "class PositionEmbs(nn.Module):\n",
    "    \"\"\"Adds learned positional embeddings to the inputs.\n",
    "    Attributes:\n",
    "        posemb_init: positional embedding initializer.\n",
    "        max_positions: maximum number of positions to embed.\n",
    "        embedding_dim: dimension of the input embeddings.\n",
    "    \"\"\"\n",
    "\n",
    "    def __init__(self, max_positions=[48,256], embedding_dim=128):\n",
    "        super().__init__()\n",
    "        self.max_positions = max_positions\n",
    "        self.embedding_dim = embedding_dim\n",
    "        self.pe1 = nn.Embedding(max_positions[0], embedding_dim//2)\n",
    "        self.pe2 = nn.Embedding(max_positions[1], embedding_dim//2)\n",
    "\n",
    "    def forward(self, inputs, pos):\n",
    "        \"\"\"Applies the AddPositionEmbs module.\n",
    "        Args:\n",
    "            inputs: Inputs to the layer, shape `(batch_size, seq_len, emb_dim)`.\n",
    "            pos: Position of the first token in each sequence, shape `(batch_size,seq_len,2)`.\n",
    "        Returns:\n",
    "            Output tensor with shape `(batch_size, seq_len, emb_dim + 2)`.\n",
    "        \"\"\"\n",
    "        assert (\n",
    "            inputs.ndim == 3\n",
    "        ), f\"Number of dimensions should be 3, but it is {inputs.ndim}\"\n",
    "        assert (\n",
    "            pos.shape[2] == 2\n",
    "        ), \"Position tensors should have two dimensions (layer, module in layer)\"\n",
    "        assert (\n",
    "            pos.shape[0] == inputs.shape[0]\n",
    "        ), \"Position tensors should have the same batch size as inputs\"\n",
    "        assert (\n",
    "            pos.shape[1] == inputs.shape[1]\n",
    "        ), \"Position tensors should have the same seq length as inputs\"\n",
    "\n",
    "        # TODO: fix embeding code\n",
    "        pos_emb1 = self.pe1(pos[:,:,0])\n",
    "        pos_emb2 = self.pe2(pos[:,:,1])\n",
    "        pos_emb = torch.cat([pos_emb1,pos_emb2],dim=2)\n",
    "        \n",
    "        out = torch.cat([inputs,pos_emb],dim=2)\n",
    "        return out"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "e011ead5",
   "metadata": {},
   "outputs": [],
   "source": [
    "PosEmbs = PositionEmbs()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "0901f54f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([5, 12, 2])\n",
      "torch.Size([5, 12, 329])\n"
     ]
    }
   ],
   "source": [
    "print(p.shape)\n",
    "pse = PosEmbs(w1,p)\n",
    "print(pse.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b6eb4ccf",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "03310363",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "id": "34e7beca",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([[1, 2],\n",
      "        [3, 4]])\n",
      "tensor([[5, 6],\n",
      "        [7, 7]])\n",
      "tensor([[ 5, 12],\n",
      "        [21, 28]])\n"
     ]
    }
   ],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "id": "0b475669",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "yes\n"
     ]
    }
   ],
   "source": [
    "for idx in range(5):\n",
    "    if idx==3:\n",
    "        break\n",
    "print('yes')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "id": "7c1d058d",
   "metadata": {},
   "outputs": [],
   "source": [
    "data = [[None,1,2,4],[3,None,1],[None,None]]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "id": "615e99f6",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[[1, 2, 4], [3, 1]]"
      ]
     },
     "execution_count": 44,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "data = [[ddx for ddx in data[idx] if ddx] for idx in range(len(data))]\n",
    "data = [ddx for ddx in data if len(ddx)>0]\n",
    "data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 45,
   "id": "cf042d31",
   "metadata": {},
   "outputs": [],
   "source": [
    "path = Path('/netscratch2/kschuerholt/code/shrp/experiments/02_representation_learning/01_test/tune/ae_resnet_test_subset/dataset.pt')\n",
    "dataset = torch.load(path)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 49,
   "id": "46ad1451",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "15\n",
      "3\n",
      "9610\n",
      "27\n"
     ]
    }
   ],
   "source": [
    "testset = dataset['testset']\n",
    "print(len(testset.data))\n",
    "print(len(testset.data[0]))\n",
    "print(len(testset.data[0][0]))\n",
    "print(len(testset.data[0][0][0]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 51,
   "id": "bb068d76",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "4608"
      ]
     },
     "execution_count": 51,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "testset.tokensize"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1383b60e",
   "metadata": {},
   "source": [
    "# benchmark dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "dca42577",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO:torch.distributed.nn.jit.instantiator:Created a temporary directory at /tmp/tmpljm_5ej0\n",
      "INFO:torch.distributed.nn.jit.instantiator:Writing /tmp/tmpljm_5ej0/_remote_module_non_scriptable.py\n"
     ]
    }
   ],
   "source": [
    "#\n",
    "import logging\n",
    "\n",
    "logging.basicConfig(level=logging.INFO)\n",
    "\n",
    "from pathlib import Path\n",
    "import json\n",
    "from shrp.datasets.dataset_simclr import SimCLRDataset\n",
    "\n",
    "from shrp.git_re_basin.git_re_basin import (\n",
    "    PermutationSpec,\n",
    "    zoo_cnn_permutation_spec,\n",
    "    weight_matching,\n",
    "    apply_permutation,\n",
    ")\n",
    "\n",
    "from shrp.models.def_AE_module import AEModule\n",
    "\n",
    "# from lightning.fabric import Fabric\n",
    "# from lightning.fabric.strategies import SingleDeviceStrategy\n",
    "\n",
    "import torch\n",
    "\n",
    "# from torch.profiler import profile, record_function, ProfilerActivity\n",
    "\n",
    "import random"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "991cbf4e",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a230afdb",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0186d788",
   "metadata": {},
   "outputs": [],
   "source": [
    "# without permutations"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "37edcec2",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO:root:reference checkpoint found at /netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/NN_tune_trainable_97ebe_00321_321_seed=322_2021-07-27_06-12-40\n",
      "2023-04-12 03:54:18,053\tINFO worker.py:1553 -- Started a local Ray instance.\n",
      "INFO:root:loading checkpoints from [PosixPath('/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform')]\n",
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 2800/2800 [00:08<00:00, 347.48it/s]\n",
      "INFO:root:Data loaded. found 700 usable samples out of potential 2800 samples.\n",
      "INFO:root:Load properties for samples from paths.\n",
      "INFO:root:### load data for dict_keys(['test_acc', 'training_iteration', 'ggap'])\n",
      "700it [00:12, 57.73it/s]\n",
      "INFO:root:Properties loaded.\n",
      "INFO:root:both view 1 and view 2 are set to canonical. number of permutations is set to 0\n",
      "INFO:root:prepare canonical form\n",
      "2023-04-12 03:54:54,356\tINFO worker.py:1553 -- Started a local Ray instance.\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "preparing computing canon form...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 700/700 [00:02<00:00, 251.84it/s]\n",
      "INFO:root:init dataset length\n",
      "INFO:root:Get Positions\n",
      "INFO:root:vectorize data\n",
      "INFO:root:Get layer mapping\n",
      "INFO:root:Get layer-wise mean and std\n",
      "INFO:root:Apply standardization\n",
      "INFO:root:Discover tokensize\n"
     ]
    }
   ],
   "source": [
    "config_key_list = []\n",
    "result_key_list = [\n",
    "    \"test_acc\",\n",
    "    \"training_iteration\",\n",
    "    \"ggap\",\n",
    "    #     \"sparsity_ratio\",\n",
    "]\n",
    "property_keys = {\n",
    "    \"result_keys\": result_key_list,\n",
    "    \"config_keys\": config_key_list,\n",
    "}\n",
    "\n",
    "path_root = Path(\"/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/\")\n",
    "\n",
    "dataset_train = SimCLRDataset(\n",
    "    root=path_root,\n",
    "    epoch_lst=[1, 5, 20, 25],\n",
    "#     epoch_lst=list(range(1, 26)),\n",
    "    #     mode=\"checkpoint\",\n",
    "    mode=\"vector\",\n",
    "    permutations_number=0,\n",
    "    permutation_spec=zoo_cnn_permutation_spec(),\n",
    "    #     view_1_canonical = False,\n",
    "    view_1_canonical=True,\n",
    "    view_2_canonical=True,\n",
    "    #     view_2_canonical = False,\n",
    "    add_noise_view_1=0.1,  # [('input', 0.15), ('output', 0.013)]\n",
    "    add_noise_view_2=0.0,  # [('input', 0.15), ('output', 0.013)]\n",
    "    noise_multiplicative=True,\n",
    "    erase_augment=None,  # {\"p\": 0.5,\"scale\":(0.02,0.33),\"value\":0,\"mode\":\"block\"}\n",
    "    windowsize=12,\n",
    "    standardize=True,\n",
    "    train_val_test=\"train\",  # determines whcih dataset split to use\n",
    "    ds_split=[0.7, 0.15, 0.15],  #\n",
    "    max_samples=1000,\n",
    "    # weight_threshold=float(\"inf\"),\n",
    "    weight_threshold=15,\n",
    "    filter_function=None,  # gets sample path as argument and returns True if model needs to be filtered out\n",
    "    property_keys=property_keys,\n",
    "    num_threads=6,\n",
    "    shuffle_path=True,\n",
    "    verbosity=3,\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "88570ee9",
   "metadata": {},
   "source": [
    "### single dataset access"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "ab21a4dc",
   "metadata": {},
   "outputs": [],
   "source": [
    "max_int = len(dataset_train)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "481a632c",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "331 ms ± 7.15 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)\n"
     ]
    }
   ],
   "source": [
    "%%timeit\n",
    "for idx in range(64):\n",
    "    kdx = random.randint(0, max_int)\n",
    "    (x_i, m_i, x_j, m_j, p) = dataset_train.__getitem__(kdx)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a970162e",
   "metadata": {},
   "source": [
    "fetching one batch of 64 samples takes ~400ms (~0.4s), sequentially. "
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8564c9e2",
   "metadata": {},
   "source": [
    "### dataloader"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "20f5cf97",
   "metadata": {},
   "outputs": [],
   "source": [
    "dloader = torch.utils.data.DataLoader(\n",
    "    dataset_train,\n",
    "    batch_size=64,\n",
    "    shuffle=True,\n",
    "    drop_last=True,\n",
    "    num_workers=4,\n",
    "    pin_memory=True,\n",
    "    prefetch_factor=2,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "9661590f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "5.06 s ± 233 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)\n"
     ]
    }
   ],
   "source": [
    "%%timeit\n",
    "\n",
    "for idx, (x_i, m_i, x_j, m_j, p) in enumerate(dloader):\n",
    "    continue"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "48e65fee",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "43"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(dloader)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "77ad08cf",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.1755813953488372"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "7.55/len(dloader)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "0d876851",
   "metadata": {},
   "outputs": [],
   "source": [
    "dloader = torch.utils.data.DataLoader(\n",
    "    dataset_train,\n",
    "    batch_size=64,\n",
    "    shuffle=True,\n",
    "    drop_last=True,\n",
    "    num_workers=8,\n",
    "    pin_memory=True,\n",
    "    prefetch_factor=2,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "464f978f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "9.72 s ± 246 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)\n"
     ]
    }
   ],
   "source": [
    "%%timeit\n",
    "\n",
    "for idx, (x_i, m_i, l_i, x_j, m_j, _, p) in enumerate(dloader):\n",
    "    continue"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "f6c1bfb8",
   "metadata": {},
   "outputs": [],
   "source": [
    "dloader = torch.utils.data.DataLoader(\n",
    "    dataset_train,\n",
    "    batch_size=64,\n",
    "    shuffle=True,\n",
    "    drop_last=True,\n",
    "    num_workers=4,\n",
    "    pin_memory=True,\n",
    "    prefetch_factor=4,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "3b2a6099",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "7.68 s ± 764 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)\n"
     ]
    }
   ],
   "source": [
    "%%timeit\n",
    "\n",
    "for idx, (x_i, m_i, l_i, x_j, m_j, _, p) in enumerate(dloader):\n",
    "    continue"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "02a3ef33",
   "metadata": {},
   "outputs": [],
   "source": [
    "dloader = torch.utils.data.DataLoader(\n",
    "    dataset_train,\n",
    "    batch_size=64,\n",
    "    shuffle=True,\n",
    "    drop_last=True,\n",
    "    num_workers=2,\n",
    "    pin_memory=True,\n",
    "    prefetch_factor=2,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "b9a4c432",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "11 s ± 137 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)\n"
     ]
    }
   ],
   "source": [
    "%%timeit\n",
    "\n",
    "for idx, (x_i, m_i, l_i, x_j, m_j, _, p) in enumerate(dloader):\n",
    "    continue"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "8aa5ccd2",
   "metadata": {},
   "outputs": [],
   "source": [
    "dloader = torch.utils.data.DataLoader(\n",
    "    dataset_train,\n",
    "    batch_size=64,\n",
    "    shuffle=True,\n",
    "    drop_last=True,\n",
    "    num_workers=2,\n",
    "    pin_memory=True,\n",
    "    prefetch_factor=4,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "5423b59a",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "11.4 s ± 315 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)\n"
     ]
    }
   ],
   "source": [
    "%%timeit\n",
    "\n",
    "for idx, (x_i, m_i, l_i, x_j, m_j, _, p) in enumerate(dloader):\n",
    "    continue"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "98f7ff04",
   "metadata": {},
   "outputs": [],
   "source": [
    "dloader = torch.utils.data.DataLoader(\n",
    "    dataset_train,\n",
    "    batch_size=64,\n",
    "    shuffle=True,\n",
    "    drop_last=True,\n",
    "    num_workers=3,\n",
    "    pin_memory=True,\n",
    "    prefetch_factor=2,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "62c6beaf",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "8.64 s ± 365 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)\n"
     ]
    }
   ],
   "source": [
    "%%timeit\n",
    "\n",
    "for idx, (x_i, m_i, l_i, x_j, m_j, _, p) in enumerate(dloader):\n",
    "    continue"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "62502cad",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4c7f6d97",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "995f5a84",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "195681c8",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d7bea768",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "202128b7",
   "metadata": {},
   "outputs": [],
   "source": [
    "# with permutations"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7d7c8659",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "id": "fb5a60b3",
   "metadata": {},
   "source": [
    "# baseline: random data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "id": "2defcd7b",
   "metadata": {},
   "outputs": [],
   "source": [
    "from torch.utils.data import Dataset\n",
    "class RandomDataset(Dataset):\n",
    "    \"\"\"\n",
    "    This dataset produces random tensors of data\n",
    "    \"\"\"\n",
    "\n",
    "    # init\n",
    "    def __init__(\n",
    "        self,\n",
    "        windowsize=12,\n",
    "        tokensize=201,\n",
    "        samples=1000,\n",
    "        ):\n",
    "        super(RandomDataset, self).__init__()\n",
    "        self.windowsize=windowsize\n",
    "        self.tokensize=tokensize\n",
    "        self.samples = samples\n",
    "    \n",
    "    def __len__(self):\n",
    "        return self.samples\n",
    "    \n",
    "    def __getitem__(self,idx):\n",
    "        x_i = torch.randn(self.windowsize,self.tokensize)\n",
    "        m_i = torch.randn(self.windowsize,self.tokensize)\n",
    "        l_i = \"123\"\n",
    "        x_j = torch.randn(self.windowsize,self.tokensize)\n",
    "        m_j = torch.randn(self.windowsize,self.tokensize)\n",
    "        l_j = \"123\"\n",
    "        p = [(1,6) for _ in range(self.windowsize)]\n",
    "        return x_i, m_i, l_i, x_j, m_j, l_j, p"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "id": "6ad7a35d",
   "metadata": {},
   "outputs": [],
   "source": [
    "rdataset = RandomDataset(\n",
    "    windowsize=dataset_train.window,\n",
    "    tokensize=dataset_train.tokensize,\n",
    "    samples = dataset_train.__len__(),\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "id": "f59193e0",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "5.24 ms ± 140 µs per loop (mean ± std. dev. of 7 runs, 100 loops each)\n"
     ]
    }
   ],
   "source": [
    "%%timeit\n",
    "for idx in range(64):\n",
    "    kdx = random.randint(0, max_int)\n",
    "    (x_i, m_i, l_i, x_j, m_j, _, p) = rdataset.__getitem__(kdx)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "373918ce",
   "metadata": {},
   "source": [
    "random data access is about 1% of original dataset class"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "id": "dcb6d58f",
   "metadata": {},
   "outputs": [],
   "source": [
    "dloader = torch.utils.data.DataLoader(\n",
    "    rdataset,\n",
    "    batch_size=64,\n",
    "    shuffle=True,\n",
    "    drop_last=True,\n",
    "    num_workers=2,\n",
    "    pin_memory=True,\n",
    "    prefetch_factor=4,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "id": "22be3604",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1.48 s ± 169 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)\n"
     ]
    }
   ],
   "source": [
    "%%timeit\n",
    "\n",
    "for idx, (x_i, m_i, l_i, x_j, m_j, _, p) in enumerate(dloader):\n",
    "    continue"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "cc5057bb",
   "metadata": {},
   "source": [
    "# test ffcv dataset type"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "fdc9ba2c",
   "metadata": {},
   "outputs": [],
   "source": [
    "from ffcv.writer import DatasetWriter"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3a7ceb26",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "\n",
    "#     RGBImageField: Handles images including (optional) compression and resizing. Pass in a PyTorch Tensor.\n",
    "\n",
    "#     IntField and FloatField: Handle simple scalar fields. Pass in int or float.\n",
    "\n",
    "#     BytesField: Stores byte arrays of variable length. Pass in numpy byte array.\n",
    "\n",
    "#     JSONField: Encodes a JSON document. Pass in dict that can be JSON-encoded.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "3f8ba4b8",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.float32\n",
      "torch.float32\n",
      "torch.float32\n",
      "torch.float32\n",
      "torch.int64\n"
     ]
    }
   ],
   "source": [
    "(x_i, m_i, x_j, m_j, p) = next(iter(dloader))\n",
    "print(x_i.dtype)\n",
    "print(m_i.dtype)\n",
    "print(x_j.dtype)\n",
    "print(m_j.dtype)\n",
    "print(p.dtype)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "d21995f0",
   "metadata": {},
   "outputs": [
    {
     "ename": "TypeError",
     "evalue": "cannot create 'torch.dtype' instances",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mTypeError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[0;32mIn [20], line 1\u001b[0m\n\u001b[0;32m----> 1\u001b[0m \u001b[43mtorch\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mdtype\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[38;5;124;43mfloat32\u001b[39;49m\u001b[38;5;124;43m'\u001b[39;49m\u001b[43m)\u001b[49m\n",
      "\u001b[0;31mTypeError\u001b[0m: cannot create 'torch.dtype' instances"
     ]
    }
   ],
   "source": [
    "torch.dtype('float32')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "579119b5",
   "metadata": {},
   "outputs": [],
   "source": [
    "from ffcv.fields import NDArrayField, FloatField, TorchTensorField\n",
    "#  class ffcv.fields.TorchTensorField(dtype: torch.dtype, shape: Tuple[int, ...])[source]\n",
    "# Each field corresponds to an element of the data tuple returned by our dataset, \n",
    "# and specifies how the element should be written to (and later, read from) the FFCV dataset file. \n",
    "# In our case, the dataset has two fields, one for the (vector) input and the other for the corresponding \n",
    "# (scalar) label. Both of these fields already have default implementations in FFCV, which we use below:\n",
    "# (x_i, m_i, x_j, m_j, p)\n",
    "w= dataset_train.window\n",
    "t = dataset_train.tokensize\n",
    "write_path = Path('./ffcv_test_dataset.beton')\n",
    "writer = DatasetWriter(write_path, {\n",
    "    'x_i': TorchTensorField(shape=(w,t), dtype=torch.float32),\n",
    "    'm_i': TorchTensorField(shape=(w,t), dtype=torch.float32),\n",
    "    'x_j': TorchTensorField(shape=(w,t), dtype=torch.float32),\n",
    "    'm_j': TorchTensorField(shape=(w,t), dtype=torch.float32),\n",
    "    'p': TorchTensorField(shape=(w,2), dtype=torch.int64),\n",
    "\n",
    "\n",
    "}, num_workers=16)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "e9e9e4ad",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 2800/2800 [00:03<00:00, 735.12it/s]\n"
     ]
    }
   ],
   "source": [
    "writer.from_indexed_dataset(dataset_train)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ae197bbb",
   "metadata": {},
   "outputs": [],
   "source": [
    "# load dataloader"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "3ef64d6f",
   "metadata": {},
   "outputs": [],
   "source": [
    "from ffcv.loader import Loader, OrderOption\n",
    "from ffcv.fields.decoders import NDArrayDecoder, FloatDecoder\n",
    "\n",
    "# Our first step is instantiating the Loader class:\n",
    "\n",
    "BATCH_SIZE = 64\n",
    "NUM_WORKERS = 4\n",
    "\n",
    "# The order option in the loader initialization is similar to PyTorch DataLoader’s shuffle option, with some additional options. This argument takes an enum provided by ffcv.loader.OrderOption:\n",
    "\n",
    "from ffcv.loader import OrderOption\n",
    "\n",
    "# Truly random shuffling (shuffle=True in PyTorch)\n",
    "# ORDERING = OrderOption.RANDOM\n",
    "\n",
    "# Unshuffled (i.e., served in the order the dataset was written)\n",
    "# ORDERING = OrderOption.SEQUENTIAL\n",
    "\n",
    "# Memory-efficient but not truly random loading\n",
    "# Speeds up loading over RANDOM when the whole dataset does not fit in RAM!\n",
    "ORDERING = OrderOption.QUASI_RANDOM\n",
    "\n",
    "\n",
    "# In order to create a loader, we need to specify a path to the FFCV dataset, batch size, number of workers, as well as two less standard arguments, order and pipelines, which we discuss below:\n",
    "# Dataset ordering\n",
    "loader = Loader(write_path,\n",
    "                batch_size=BATCH_SIZE,\n",
    "                num_workers=NUM_WORKERS,\n",
    "                order=ORDERING,\n",
    "#                 pipelines=PIPELINES\n",
    "               )\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "49c78b4c",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "72.3 ms ± 1.77 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)\n"
     ]
    }
   ],
   "source": [
    "%%timeit\n",
    "\n",
    "for idx, (x_i, m_i, x_j, m_j, p) in enumerate(loader):\n",
    "    continue"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b7045c87",
   "metadata": {},
   "source": [
    "gave up, opencv installation is a pain in the ... "
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a64d5ed8",
   "metadata": {},
   "source": [
    "# improvement: tensorize data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "dc11d817",
   "metadata": {},
   "outputs": [],
   "source": [
    "# idee:\n",
    "# call tokenize at init\n",
    "# -> list of samples [n_samples,n_epochs]\n",
    "# -> each samlpe is [n_tokens,tokensize] (not list of tensors)\n",
    "# -> applying all operations on that one tensor"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "072fec30",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "from torch.utils.data import Dataset\n",
    "\n",
    "from pathlib import Path\n",
    "import random\n",
    "import copy\n",
    "\n",
    "import itertools\n",
    "from math import factorial\n",
    "\n",
    "from shrp.datasets.dataset_epochs import ModelDatasetBaseEpochs\n",
    "from shrp.git_re_basin.git_re_basin import (\n",
    "    PermutationSpec,\n",
    "    zoo_cnn_permutation_spec,\n",
    "    weight_matching,\n",
    "    apply_permutation,\n",
    ")\n",
    "\n",
    "import logging\n",
    "\n",
    "from typing import List\n",
    "\n",
    "from shrp.datasets.random_erasing import RandomErasingVector\n",
    "\n",
    "import ray\n",
    "from shrp.datasets.progress_bar import ProgressBar\n",
    "\n",
    "#####################################################################\n",
    "# Define Dataset class\n",
    "#####################################################################\n",
    "class SimCLRDatasetTensor(ModelDatasetBaseEpochs):\n",
    "    \"\"\"\n",
    "    This class inherits from the base ModelDatasetBaseEpochs class.\n",
    "    It extends it by permutations of the dataset in the init function.\n",
    "    \"\"\"\n",
    "\n",
    "    # init\n",
    "    def __init__(\n",
    "        self,\n",
    "        root,\n",
    "        epoch_lst=10,\n",
    "        mode=\"vector\",  # \"vector\", \"checkpoint\"\n",
    "        permutations_number=10,\n",
    "        permutation_spec: PermutationSpec = zoo_cnn_permutation_spec,\n",
    "        view_1_canonical: bool = False,\n",
    "        view_2_canonical: bool = False,\n",
    "        add_noise_view_1: float = 0.0,  # [('input', 0.15), ('output', 0.013)]\n",
    "        add_noise_view_2: float = 0.0,  # [('input', 0.15), ('output', 0.013)]\n",
    "        noise_multiplicative: bool = True,\n",
    "        erase_augment=None,  # {\"p\": 0.5,\"scale\":(0.02,0.33),\"value\":0,\"mode\":\"block\"}\n",
    "        windowsize: int = 5,\n",
    "        standardize: bool = True,  # wether or not to standardize the data\n",
    "        tokensize: int = 0,\n",
    "        train_val_test=\"train\",\n",
    "        ds_split=[0.7, 0.3],\n",
    "        weight_threshold: float = float(\"inf\"),\n",
    "        max_samples: int = 0,  # limit the number of models to integer number (full model trajectory, all epochs)\n",
    "        filter_function=None,  # gets sample path as argument and returns True if model needs to be filtered out\n",
    "        property_keys=None,\n",
    "        shuffle_path: bool = True,\n",
    "        num_threads=4,\n",
    "        verbosity=0,\n",
    "    ):\n",
    "        # call init of base class\n",
    "        super().__init__(\n",
    "            root=root,\n",
    "            epoch_lst=epoch_lst,\n",
    "            mode=\"checkpoint\",\n",
    "            train_val_test=train_val_test,\n",
    "            ds_split=ds_split,\n",
    "            weight_threshold=weight_threshold,\n",
    "            max_samples=max_samples,\n",
    "            filter_function=filter_function,\n",
    "            property_keys=property_keys,\n",
    "            num_threads=num_threads,\n",
    "            verbosity=verbosity,\n",
    "            shuffle_path=shuffle_path,\n",
    "        )\n",
    "        self.mode = mode\n",
    "        self.permutations_number = permutations_number\n",
    "        self.permutation_spec = permutation_spec\n",
    "        self.standardize = standardize\n",
    "        self.tokensize = tokensize\n",
    "\n",
    "        self.add_noise_view_1 = add_noise_view_1\n",
    "        assert isinstance(self.add_noise_view_1, float)\n",
    "        self.add_noise_view_2 = add_noise_view_2\n",
    "        assert isinstance(self.add_noise_view_1, float)\n",
    "        self.use_multiplicative_noise = noise_multiplicative\n",
    "\n",
    "        self.num_threads = num_threads\n",
    "\n",
    "        self.view_1_canonical = view_1_canonical\n",
    "        self.view_2_canonical = view_2_canonical\n",
    "        if view_1_canonical and view_2_canonical:\n",
    "            logging.info(\n",
    "                f\"both view 1 and view 2 are set to canonical. number of permutations is set to 0\"\n",
    "            )\n",
    "            self.permutations_number = 0\n",
    "\n",
    "        # set erase augmnet\n",
    "        self.set_erase(erase_augment)\n",
    "\n",
    "        if self.view_1_canonical or self.view_2_canonical:\n",
    "            logging.info(\"prepare canonical form\")\n",
    "            # TODO: load reference checkpoint in base class, share between train/val/test\n",
    "            self.map_models_to_canonical()\n",
    "\n",
    "        ### init len ###########################################################################################\n",
    "        logging.info(\"init dataset length\")\n",
    "        self.init_len()\n",
    "\n",
    "        ### get positions ###########################################################################################\n",
    "        logging.info(\"Get Positions\")\n",
    "        reference_checkpoint = self.reference_checkpoint\n",
    "        self.positions = get_position_mapping_from_checkpoint(reference_checkpoint)\n",
    "\n",
    "        ### vectorize data ###########################################################################################\n",
    "        if self.mode == \"vector\":\n",
    "            logging.info(\"vectorize data\")\n",
    "            # keep reference checkpoint\n",
    "            self.vectorize_data()\n",
    "\n",
    "        ### initialize permutations ##########################################################################################################################################\n",
    "        # list of permutations (list of list with indexes)\n",
    "        if self.permutations_number > 0:\n",
    "            logging.info(\"init permutations\")\n",
    "            self.precompute_permutations(\n",
    "                permutation_number=self.permutations_number,\n",
    "                perm_spec=self.permutation_spec,\n",
    "                num_threads=num_threads,\n",
    "            )\n",
    "\n",
    "        if self.standardize:\n",
    "            self.standardize_data()\n",
    "\n",
    "        ### set module window ##################################################################################################################################################################\n",
    "        self.set_module_window(windowsize=windowsize)\n",
    "\n",
    "        ### set tokensize ###################################################################################################################################################################\n",
    "        logging.info('Tokenize data')\n",
    "        self.set_token_size(tokensize=self.tokensize)\n",
    "        \n",
    "        self.tokenize_data()        \n",
    "\n",
    "    def set_module_window(self, windowsize: int = 5):\n",
    "        # check that window is within range 1,no-tokens\n",
    "        assert 0 < windowsize <= len(self.data[0][-1])\n",
    "        self.window = windowsize\n",
    "\n",
    "    def set_token_size(self, tokensize: int = 0):\n",
    "        \"\"\"\n",
    "        tokens are zero-padded to all have the same size at __getitem__\n",
    "        this function sets the size of the token either to a specific length (rest is cut off)\n",
    "        or discovers the maximum size\n",
    "        \"\"\"\n",
    "        if tokensize > 0:\n",
    "            self.tokensize = tokensize\n",
    "        else:\n",
    "            logging.info(\"Discover tokensize\")\n",
    "            # assumes data is already vectorized\n",
    "            max_len = 0\n",
    "            for tdx in self.data[0][-1]:\n",
    "                if tdx.shape[0] > max_len:\n",
    "                    max_len = tdx.shape[0]\n",
    "            self.tokensize = max_len\n",
    "\n",
    "            \n",
    "    def tokenize_data(self):\n",
    "        \"\"\"\n",
    "        cast samples as list of tokens to tensors to speed up processing\n",
    "        \"\"\"\n",
    "        # iterate over all samlpes\n",
    "        for idx in range(len(self.data)):\n",
    "            for jdx in range(len(self.data[idx])):\n",
    "                self.data[idx][jdx], mask = tokenize(\n",
    "                    self.data[idx][jdx],\n",
    "                    tokensize=self.tokensize,\n",
    "                    return_mask=True,\n",
    "                )\n",
    "        self.mask = mask\n",
    "        \n",
    "    ## get_weights ####################################################################################################################################################################\n",
    "    def __get_weights__(\n",
    "        self,\n",
    "    ):\n",
    "        \"\"\"\n",
    "        Returns:\n",
    "            torch.Tensor with full dataset as sequence of components [n_samples,n_tokens_per_sample,token_dim]\n",
    "        \"\"\"\n",
    "        #todo -> probably needs flattening or something\n",
    "        if not self.mode == \"vector\":\n",
    "            data_tmp = copy.deepcopy(self.data)\n",
    "            self.vectorize_data()\n",
    "            data_out = [\n",
    "                tokenize(\n",
    "                    self.data[idx][jdx],\n",
    "                    tokensize=self.tokensize,\n",
    "                    return_mask=True,\n",
    "                )[0]\n",
    "                for idx in range(len(self.data))\n",
    "                for jdx in range(len(self.data[idx]))\n",
    "            ]\n",
    "            mask_out = [\n",
    "                tokenize(\n",
    "                    self.data[idx][jdx],\n",
    "                    tokensize=self.tokensize,\n",
    "                    return_mask=True,\n",
    "                )[1]\n",
    "                for idx in range(len(self.data))\n",
    "                for jdx in range(len(self.data[idx]))\n",
    "            ]\n",
    "            data_out = torch.stack(data_out)\n",
    "            mask_out = torch.stack(mask_out)\n",
    "            self.data = data_tmp\n",
    "            return data_out, mask_out\n",
    "        data_out = [\n",
    "            tokenize(\n",
    "                self.data[idx][jdx],\n",
    "                tokensize=self.tokensize,\n",
    "                return_mask=True,\n",
    "            )[0]\n",
    "            for idx in range(len(self.data))\n",
    "            for jdx in range(len(self.data[idx]))\n",
    "        ]\n",
    "        mask_out = [\n",
    "            tokenize(\n",
    "                self.data[idx][jdx],\n",
    "                tokensize=self.tokensize,\n",
    "                return_mask=True,\n",
    "            )[1]\n",
    "            for idx in range(len(self.data))\n",
    "            for jdx in range(len(self.data[idx]))\n",
    "        ]\n",
    "        data_out = torch.stack(data_out)\n",
    "        mask_out = torch.stack(mask_out)\n",
    "        logging.debug(f\"shape of weight tensor: {data_out.shape}\")\n",
    "        return data_out, mask_out\n",
    "\n",
    "    ## getitem ####################################################################################################################################################################\n",
    "    def __getitem__(self, index):\n",
    "        \"\"\"\n",
    "        Args:\n",
    "            index (int): Index of the sample to be retrieved\n",
    "        Returns:\n",
    "            ddx_idx: torch.Tensor of neuron tokens with shape [n_tokens_per_sample/windowsize,token_dim] in view 1\n",
    "            mask_idx: torch.Tensor of the same shape as ddx_idx indicating the nonzero elements\n",
    "            label_idx: label for sample in view 1\n",
    "            ddx_jdx: torch.Tensor of neuron tokens with shape [n_tokens_per_sample/windowsize,token_dim] in view 2\n",
    "            mask_jdx: torch.Tensor of the same shape as ddx_jdx indicating the nonzero elements\n",
    "            label_jdx: label for sample in view 2\n",
    "            pos: positions [layer,token_in_layer] of sample in view 1 / view 2\n",
    "        \"\"\"\n",
    "        # get model and epoch index\n",
    "        mdx, edx = self._index[index]\n",
    "\n",
    "        # get permutation index -> pick random number from available perms\n",
    "        if self.permutations_number > 0:\n",
    "            perm_idx, perm_jdx = random.choices(\n",
    "                list(range(self.permutations_number)), k=2\n",
    "            )\n",
    "\n",
    "        ## mode \"vector has different workflow\"\n",
    "        if self.mode == \"vector\":\n",
    "            # get raw data, assume view 1 and view 2 are the same\n",
    "            ddx = self.data[mdx][edx]\n",
    "            label = self.labels[mdx][edx]\n",
    "\n",
    "            # slice window of size windowsize\n",
    "            idx_start = random.randint(0, len(ddx) - self.window)\n",
    "\n",
    "            # get position\n",
    "            pos = self.positions[idx_start : idx_start + self.window]\n",
    "\n",
    "            # get permutation index -> pick random number from available perms\n",
    "            if self.permutations_number > 0:\n",
    "                perm_idx, perm_jdx = random.choices(\n",
    "                    list(range(self.permutations_number)), k=2\n",
    "                )\n",
    "\n",
    "            # permutation\n",
    "            # permute data idx\n",
    "            if self.view_1_canonical:\n",
    "                ddx_idx = copy.deepcopy(ddx)\n",
    "                ddx_idx = ddx_idx[idx_start : idx_start + self.window]\n",
    "                label_idx = f\"{label}#_#canon\"\n",
    "            elif self.permutations_number > 0:\n",
    "                # get current permutation for slice of window\n",
    "                ddx_idx = copy.deepcopy(ddx)\n",
    "                permdx = self.permutations[perm_idx]\n",
    "                ddx_idx = permute_model_vector(\n",
    "                    vec=ddx_idx, perm=permdx, idx_start=idx_start, window=self.window\n",
    "                )\n",
    "                # update label\n",
    "                label_idx = f\"{label}#_#per_{perm_idx}\"\n",
    "            else:\n",
    "                ddx_idx = copy.deepcopy(ddx)\n",
    "                ddx_idx = ddx_idx[idx_start : idx_start + self.window]\n",
    "                label_idx = copy.deepcopy(label)\n",
    "\n",
    "            # permute data jdx\n",
    "            if self.view_2_canonical:\n",
    "                ddx_jdx = copy.deepcopy(ddx)\n",
    "                ddx_jdx = ddx_jdx[idx_start : idx_start + self.window]\n",
    "                label_jdx = f\"{label}#_#canon\"\n",
    "            elif self.permutations_number > 0:\n",
    "                # get current permutation for slice of window\n",
    "                ddx_jdx = copy.deepcopy(ddx)\n",
    "                permdx = self.permutations[perm_jdx]\n",
    "                ddx_jdx = permute_model_vector(\n",
    "                    vec=ddx_jdx, perm=permdx, idx_start=idx_start, window=self.window\n",
    "                )\n",
    "                # update label\n",
    "                label_jdx = f\"{label}#_#per_{perm_jdx}\"\n",
    "            else:\n",
    "                ddx_jdx = copy.deepcopy(ddx)\n",
    "                ddx_jdx = ddx_jdx[idx_start : idx_start + self.window]\n",
    "                label_jdx = copy.deepcopy(label)\n",
    "\n",
    "            # noise\n",
    "            if not self.add_noise_view_1 == False:\n",
    "                # add noise to input\n",
    "                # check sigma is larger than 0\n",
    "                if self.add_noise_view_1 > 0:\n",
    "                    if self.use_multiplicative_noise:\n",
    "                        # multiply each token with noise (uniform distribution around 1)\n",
    "                        # noise idx\n",
    "                        ddx_idx = ddx_idx * (1.0 + self.add_noise_view_1 * torch.randn(ddx_idx.shape))\n",
    "                    else:\n",
    "                        # add to each token noise (uniform distribution around 0)\n",
    "                        # noise idx\n",
    "                        ddx_idx = ddx_idx + self.add_noise_view_1 * torch.randn(ddx_idx.shape)\n",
    "\n",
    "            if not self.add_noise_view_2 == False:\n",
    "                # check sigma is number\n",
    "                assert isinstance(self.add_noise_view_2, float)\n",
    "                # add noise to input\n",
    "                # check sigma is larger than 0\n",
    "                if self.add_noise_view_2 > 0:\n",
    "                    if self.use_multiplicative_noise:\n",
    "                        # multiply each token with noise (uniform distribution around 1)\n",
    "                        # noise jdx\n",
    "                        ddx_jdx = ddx_jdx * (1.0 + self.add_noise_view_2 * torch.randn(ddx_jdx.shape))\n",
    "                    else:\n",
    "                        # add to each token noise (uniform distribution around 0)\n",
    "                        # noise jdx\n",
    "                        ddx_jdx = ddx_jdx + self.add_noise_view_2 * torch.randn(ddx_jdx.shape)\n",
    "\n",
    "            # erase_input/output augmentation\n",
    "            if self.erase_augment is not None:\n",
    "                # apply erase_augment on each token\n",
    "                ddx_idx = [self.erase_augment(iddx) for iddx in ddx_idx]\n",
    "                ddx_jdx = [self.erase_augment(iddx) for iddx in ddx_jdx]\n",
    "\n",
    "            # tokenize\n",
    "            ddx_idx, mask_idx = tokenize(t=ddx_idx, tokensize=self.tokensize)\n",
    "            ddx_jdx, mask_jdx = tokenize(t=ddx_jdx, tokensize=self.tokensize)\n",
    "\n",
    "            return ddx_idx, mask_idx, label_idx, ddx_jdx, mask_jdx, label_jdx, pos\n",
    "        ### end mode==\"vector\"\n",
    "\n",
    "        # TODO implement version on checkpoints, relatively straight-forward\n",
    "        raise NotImplementedError(\n",
    "            \"simclr dataset is not yet implemented for checkpoints. hang in there..\"\n",
    "        )\n",
    "\n",
    "    ### len ##################################################################################################################################################################\n",
    "    def init_len(self):\n",
    "        index = []\n",
    "        for idx, ddx in enumerate(self.data):\n",
    "            idx_tmp = [(idx, jdx) for jdx in range(len(ddx))]\n",
    "            index.extend(idx_tmp)\n",
    "        self._len = len(index)\n",
    "        self._index = index\n",
    "\n",
    "    def __len__(self):\n",
    "        return self._len\n",
    "\n",
    "    ### set erase ############################################################\n",
    "    def set_erase(self, erase=None):\n",
    "        if erase is not None:\n",
    "            assert (\n",
    "                self.mode == \"vectorize\" or self.mode == \"vector\"\n",
    "            ), \"erasing is only for vectorized mode implemenetd\"\n",
    "            erase = RandomErasingVector(\n",
    "                p=erase[\"p\"],\n",
    "                scale=erase[\"scale\"],\n",
    "                value=erase[\"value\"],\n",
    "                mode=erase[\"mode\"],\n",
    "            )\n",
    "        else:\n",
    "            erase = None\n",
    "        self.erase_augment = erase\n",
    "\n",
    "    ### vectorize_data #########################################################################################################################################################\n",
    "    def vectorize_data(self):\n",
    "        # iterate over models\n",
    "        for idx in range(len(self.data)):\n",
    "            # iterate over epochs\n",
    "            for jdx in range(len(self.data[idx])):\n",
    "                # get checkpoint\n",
    "                checkpoint = copy.deepcopy(self.data[idx][jdx])\n",
    "                # get vectorize\n",
    "                ddx = vectorize_checkpoint(checkpoint)\n",
    "                # overwrite data\n",
    "                self.data[idx][jdx] = ddx\n",
    "\n",
    "    ### standardize data #########################################################################################################################################################\n",
    "    def standardize_data(self):\n",
    "        \"\"\"\n",
    "        standardize data to zero-mean / unit std. per layer\n",
    "        store per-layer mean / std\n",
    "        \"\"\"\n",
    "        logging.info(\"Get layer mapping\")\n",
    "        # step 1: get token-layer index relation\n",
    "        layers = {}\n",
    "        # init vals\n",
    "        cur_layer = 0\n",
    "        cur_layer_start = 0\n",
    "        # iterate over models\n",
    "        for idx in range(self.positions.shape[0]):\n",
    "            ldx, kdx = self.positions[idx, 0].item(), self.positions[idx, 1].item()\n",
    "            # if new layer\n",
    "            if not ldx == cur_layer:\n",
    "                # add previous layer to dict\n",
    "                layer = str(cur_layer)\n",
    "                cur_layer_end = idx - 1\n",
    "                layers[layer] = {\n",
    "                    \"start_idx\": cur_layer_start,\n",
    "                    \"end_idx\": cur_layer_end,\n",
    "                }\n",
    "                # start new layer\n",
    "                cur_layer = ldx\n",
    "                cur_layer_start = idx\n",
    "        # last layer\n",
    "        layer = str(ldx)\n",
    "        cur_layer_end = idx\n",
    "        layers[layer] = {\n",
    "            \"start_idx\": cur_layer_start,\n",
    "            \"end_idx\": cur_layer_end,\n",
    "        }\n",
    "        logging.debug(f\"layer mapping: {layers}\")\n",
    "\n",
    "        logging.info(\"Get layer-wise mean and std\")\n",
    "\n",
    "        # iterate over layers\n",
    "        for layer in layers:\n",
    "            idx_start = layers[layer][\"start_idx\"]\n",
    "            idx_end = layers[layer][\"end_idx\"]\n",
    "\n",
    "            # collect all tokens within the layer for all models\n",
    "            tmp = []\n",
    "            # iterate over models\n",
    "            for idx in range(len(self.data)):\n",
    "                for jdx in range(len(self.data[idx])):\n",
    "                    tmp2 = [\n",
    "                        self.data[idx][jdx][ldx]\n",
    "                        for ldx in range(idx_start, idx_end + 1)\n",
    "                    ]\n",
    "                    tmp.extend(tmp2)\n",
    "\n",
    "            # stack / cat\n",
    "            tmp = torch.stack(tmp, dim=0)\n",
    "\n",
    "            # compute mean / std\n",
    "            mu = torch.mean(tmp)\n",
    "            sigma = torch.std(tmp)\n",
    "\n",
    "            # store in layer\n",
    "            layers[layer][\"mean\"] = mu\n",
    "            layers[layer][\"std\"] = sigma\n",
    "\n",
    "            # free memory\n",
    "            del tmp\n",
    "\n",
    "        self.layers = layers\n",
    "\n",
    "        logging.info(\"Apply standardization\")\n",
    "        # TODO: make more efficient, cut out the first two for loops with list expression?\n",
    "        # standardize:\n",
    "        # # iterate over models\n",
    "        for idx in range(len(self.data)):\n",
    "            for jdx in range(len(self.data[idx])):\n",
    "                # # iterate over tokens of that layer\n",
    "                for tdx in range(len(self.data[idx][jdx])):\n",
    "                    # get position\n",
    "                    pos = self.positions[tdx]\n",
    "                    ldx = pos[0].item()\n",
    "                    # get mu/sigma for that token\n",
    "                    mu = layers[str(ldx)][\"mean\"]\n",
    "                    std = layers[str(ldx)][\"std\"]\n",
    "                    # # standardize with mean / std\n",
    "                    self.data[idx][jdx][tdx] = (self.data[idx][jdx][tdx] - mu) / std\n",
    "\n",
    "    ### precompute_permutation_index #########################################################################################################################################################\n",
    "    def precompute_permutation_index(self):\n",
    "        # ASSUMES THAT DATA IS ALREADY VECTORIZED\n",
    "        permutation_index_list = []\n",
    "        # create index vector\n",
    "        # print(f\"vector shape: {self.data_in[0].shape}\")\n",
    "        index_vector = torch.tensor(list(range(self.data_in[0].shape[0])))\n",
    "        # cast index vector to double\n",
    "        index_vector = index_vector.double()\n",
    "        # print(f\"index vector: {index_vector}\")\n",
    "        # reference checkpoint\n",
    "        reference_checkpoint = copy.deepcopy(self.reference_checkpoint)\n",
    "        # cast index vector to checkpoint\n",
    "        index_checkpoint = vector_to_checkpoint(\n",
    "            checkpoint=copy.deepcopy(reference_checkpoint),\n",
    "            vector=copy.deepcopy(index_vector),\n",
    "            layer_lst=self.layer_lst,\n",
    "            use_bias=self.use_bias,\n",
    "        )\n",
    "\n",
    "        ## init multiprocessing environment ############\n",
    "        ray.init(num_cpus=self.num_threads)\n",
    "\n",
    "        ### gather data #############################################################################################\n",
    "        logging.info(f\"preparing permutation indices from {self.root}\")\n",
    "        pb = ProgressBar(total=self.permutations_number)\n",
    "        pb_actor = pb.actor\n",
    "\n",
    "        # loop over all permutations in self.permutations_number\n",
    "        for pdx in range(self.permutations_number):\n",
    "            # get perm dict\n",
    "            prmt_dct = self.permutations_dct_lst[pdx]\n",
    "            #\n",
    "            index_p = compute_single_index_vector_remote.remote(\n",
    "                index_checkpoint=copy.deepcopy(index_checkpoint),\n",
    "                prmt_dct=prmt_dct,\n",
    "                layer_lst=self.layer_lst,\n",
    "                permute_layers=self.permute_layers,\n",
    "                use_bias=self.use_bias,\n",
    "                pba=pb_actor,\n",
    "            )\n",
    "            # append to permutation_index_list\n",
    "            permutation_index_list.append(index_p)\n",
    "\n",
    "        # update progress bar\n",
    "        pb.print_until_done()\n",
    "\n",
    "        # collect actual data\n",
    "        permutation_index_list = ray.get(permutation_index_list)\n",
    "\n",
    "        ray.shutdown()\n",
    "\n",
    "        self.permutation_index_list = permutation_index_list\n",
    "\n",
    "    ### map data to canoncial #############################################################################################\n",
    "    def map_models_to_canonical(self):\n",
    "        \"\"\"\n",
    "        define reference model\n",
    "        iterate over all models\n",
    "        get permutation w.r.t last epoch (best convergence)\n",
    "        apply same permutation on all epochs (on raw data)\n",
    "        \"\"\"\n",
    "        # use first model / last epoch as reference model (might be sub-optimal)\n",
    "        reference_model = self.reference_checkpoint\n",
    "\n",
    "        ## init multiprocessing environment ############\n",
    "        ray.init(num_cpus=self.num_threads)\n",
    "\n",
    "        ### gather data #############################################################################################\n",
    "        print(f\"preparing computing canon form...\")\n",
    "        pb = ProgressBar(total=len(self.data))\n",
    "        pb_actor = pb.actor\n",
    "\n",
    "        for idx in range(len(self.data)):\n",
    "            # align models using git-re-basin\n",
    "            perm_spec = self.permutation_spec\n",
    "            # get second\n",
    "            model_curr = self.data[idx]\n",
    "            model_curr = compute_single_canon_form.remote(\n",
    "                reference_model=reference_model,\n",
    "                data_curr=model_curr,\n",
    "                perm_spec=perm_spec,\n",
    "                pba=pb_actor,\n",
    "            )\n",
    "            self.data[idx] = model_curr\n",
    "\n",
    "        # update progress bar\n",
    "        pb.print_until_done()\n",
    "\n",
    "        self.data = [ray.get(self.data[idx]) for idx in range(len(self.data))]\n",
    "        ray.shutdown()\n",
    "\n",
    "    ### precompute_permutations #############################################################################################\n",
    "    def precompute_permutations(self, permutation_number, perm_spec, num_threads=6):\n",
    "        \"\"\"\n",
    "        - get permutation_dict as template\n",
    "        - generate random permutations\n",
    "        - generate index checkpoint\n",
    "            - copy actual checkpoint\n",
    "            - flatten tensor\n",
    "            - generate list with indices of same shape\n",
    "            - put on tensor\n",
    "            - reshape to original view\n",
    "        - apply permutation on checkpoint: store checkpoint as permutation dict.\n",
    "\n",
    "        - permuting modules translates to:\n",
    "            - get right index for current module\n",
    "            - .flatten()\n",
    "            - apply permutation /slice\n",
    "            - .view()\n",
    "        \"\"\"\n",
    "        logging.info(\"start precomputing permutations\")\n",
    "        # model_curr = self.data[0][-1]\n",
    "        model_curr = self.reference_checkpoint\n",
    "        # find permutation of model to itself as reference\n",
    "        reference_permutation = weight_matching(\n",
    "            ps=perm_spec, params_a=model_curr, params_b=model_curr\n",
    "        )\n",
    "\n",
    "        logging.info(\"get random permutation dicts\")\n",
    "        # compute random permutations\n",
    "        permutation_dicts = []\n",
    "        for ndx in range(permutation_number):\n",
    "            perm = copy.deepcopy(reference_permutation)\n",
    "            for key in perm.keys():\n",
    "                # get permuted indecs for current layer\n",
    "                perm[key] = torch.randperm(perm[key].shape[0]).float()\n",
    "            # append to list of permutation dicts\n",
    "            permutation_dicts.append(perm)\n",
    "\n",
    "        self.permutation_dicts = permutation_dicts\n",
    "\n",
    "        if self.mode == \"vector\":\n",
    "            logging.info(\"get permutation indices\")\n",
    "            # get permutation data\n",
    "            ## get reference checkpoint\n",
    "            ref_checkpoint = copy.deepcopy(model_curr)\n",
    "            ## vectoirze\n",
    "            ref_vec_global = vectorize_checkpoint(ref_checkpoint)\n",
    "            ref_vec_kernel = copy.deepcopy(ref_vec_global)\n",
    "            ## get reference index vec\n",
    "            for idx, module in enumerate(ref_vec_global):\n",
    "                # get global index of permutation between kernels\n",
    "                index_global = torch.ones(module.numel()) * idx\n",
    "                index_global = index_global.view(module.shape)\n",
    "                ref_vec_global[idx] = index_global\n",
    "                # got local index of permutation within kernels\n",
    "                index_kernel = torch.tensor(list(range(module.numel())))\n",
    "                index_kernel = index_kernel.view(module.shape)\n",
    "                ref_vec_kernel[idx] = index_kernel\n",
    "            ## map to checkpoint\n",
    "            ref_checkpoint_global = vector_to_checkpoint(\n",
    "                vector=ref_vec_global, reference_checkpoint=ref_checkpoint\n",
    "            )\n",
    "            ref_checkpoint_kernel = vector_to_checkpoint(\n",
    "                vector=ref_vec_kernel, reference_checkpoint=ref_checkpoint\n",
    "            )\n",
    "\n",
    "            ## init multiprocessing environment ############\n",
    "            ray.init(num_cpus=num_threads)\n",
    "            pb = ProgressBar(total=permutation_number)\n",
    "            pb_actor = pb.actor\n",
    "            # get permutations\n",
    "            permutations_global = []\n",
    "            permutations_kernel = []\n",
    "            for perm_dict in permutation_dicts:\n",
    "                perm_curr_global = compute_single_perm.remote(\n",
    "                    reference_checkpoint=ref_checkpoint_global,\n",
    "                    permutation_dict=perm_dict,\n",
    "                    perm_spec=perm_spec,\n",
    "                    pba=pb_actor,\n",
    "                )\n",
    "\n",
    "                perm_curr_kernel = compute_single_perm.remote(\n",
    "                    reference_checkpoint=ref_checkpoint_kernel,\n",
    "                    permutation_dict=perm_dict,\n",
    "                    perm_spec=perm_spec,\n",
    "                    pba=pb_actor,\n",
    "                )\n",
    "\n",
    "                permutations_global.append(perm_curr_global)\n",
    "                permutations_kernel.append(perm_curr_kernel)\n",
    "\n",
    "            permutations_global = ray.get(permutations_global)\n",
    "            permutations_kernel = ray.get(permutations_kernel)\n",
    "\n",
    "            permutations_global = [\n",
    "                torch.tensor([perm[0].item() for perm in perm_g]).int()\n",
    "                for perm_g in permutations_global\n",
    "            ]\n",
    "\n",
    "            permutations = [\n",
    "                (perm_g, perm_k)\n",
    "                for (perm_g, perm_k) in zip(permutations_global, permutations_kernel)\n",
    "            ]\n",
    "\n",
    "            ray.shutdown()\n",
    "\n",
    "            self.permutations = permutations\n",
    "\n",
    "\n",
    "def permute_model_vector(\n",
    "    vec: List[torch.Tensor], perm: List[tuple], idx_start: int, window: int\n",
    "):\n",
    "    \"\"\"\n",
    "    performs permutation on vectorized model and returns slice of tokens\n",
    "    input vec: list(torch.tensor) with the weights per output channel of the full model\n",
    "    input perm: contains two pieces of information for permutation. perm[0] is the global permutation of the tokens, perm[1] contains the permutaion mappings within tokens per token\n",
    "    input idx_start: int marks the start of a slice of tokens to keep\n",
    "    input window: int marks the size of the slice to keep\n",
    "    return vec: list(torch.tensor) of permuted and sliced tokens\n",
    "    \"\"\"\n",
    "    # create index vector of tokens\n",
    "    index = list(range(len(vec)))\n",
    "\n",
    "    # apply global permutation on index\n",
    "    # using (slices of) the permuted index to access tokens equals permuting all tokens and slicing after\n",
    "    perm_glob = perm[0]\n",
    "    index = [index[idx] for idx in perm_glob]\n",
    "\n",
    "    # slice index\n",
    "    idx_end = idx_start + window\n",
    "    index = index[idx_start:idx_end]\n",
    "\n",
    "    # slice token sequence\n",
    "    vec = [vec[idx] for idx in index]\n",
    "\n",
    "    # slice permutations\n",
    "    perm_loc = perm[1]\n",
    "    perm_loc = [perm_loc[idx] for idx in index]\n",
    "\n",
    "    # apply token permutation\n",
    "    vec = [vecdx[permdx] for (vecdx, permdx) in zip(vec, perm_loc)]\n",
    "\n",
    "    # return tokens\n",
    "    return vec\n",
    "\n",
    "\n",
    "### helper parallel function #############################################################################################\n",
    "@ray.remote(num_returns=1)\n",
    "def compute_single_perm(reference_checkpoint, permutation_dict, perm_spec, pba):\n",
    "    # copy reference checkpoint\n",
    "    index_check = copy.deepcopy(reference_checkpoint)\n",
    "    # apply permutation on checkpoint\n",
    "    index_check_perm = apply_permutation(\n",
    "        ps=perm_spec, perm=permutation_dict, params=index_check\n",
    "    )\n",
    "    # vectorize\n",
    "    index_perm = vectorize_checkpoint(index_check_perm)\n",
    "    # update counter\n",
    "    pba.update.remote(1)\n",
    "    # return list\n",
    "    return index_perm\n",
    "\n",
    "\n",
    "### helper parallel function #############################################################################################\n",
    "@ray.remote(num_returns=1)\n",
    "def compute_single_canon_form(reference_model, data_curr, perm_spec, pba):\n",
    "    # get second\n",
    "    model_curr = data_curr[-1]\n",
    "    # find permutation to match params_b to params_a\n",
    "    logging.debug(\n",
    "        f\"compute canonical form: params a {type(reference_model)} params b {type(model_curr)}\"\n",
    "    )\n",
    "    match_permutation = weight_matching(\n",
    "        ps=perm_spec, params_a=reference_model, params_b=model_curr\n",
    "    )\n",
    "    # apply permutation on all epochs\n",
    "    for jdx in range(len(data_curr)):\n",
    "        model_curr = data_curr[jdx]\n",
    "        model_curr_perm = apply_permutation(\n",
    "            ps=perm_spec, perm=match_permutation, params=model_curr\n",
    "        )\n",
    "        # put back in data\n",
    "        data_curr[jdx] = model_curr_perm\n",
    "\n",
    "    # update counter\n",
    "    pba.update.remote(1)\n",
    "    # return list\n",
    "    return data_curr\n",
    "\n",
    "\n",
    "### helper parallel function #############################################################################################\n",
    "@ray.remote(num_returns=1)\n",
    "def compute_single_index_vector_remote(\n",
    "    index_checkpoint, prmt_dct, layer_lst, permute_layers, use_bias, pba\n",
    "):\n",
    "    # apply permutation on copy of unit checkpoint\n",
    "    # TODO: fix function,\n",
    "    chkpt_p = permute_checkpoint(\n",
    "        index_checkpoint,\n",
    "        layer_lst,\n",
    "        permute_layers,\n",
    "        prmt_dct,\n",
    "    )\n",
    "\n",
    "    # cast back to vector\n",
    "    vector_p = vectorize_checkpoint(copy.deepcopy(chkpt_p), layer_lst, use_bias)\n",
    "    # cast vector back to int\n",
    "    vector_p = vector_p.int()\n",
    "    # we specifically don't check for uniqueness of indices. we'd rather let this run into index errors to catch the issue\n",
    "    index_p = copy.deepcopy(vector_p.tolist())\n",
    "    # update counter\n",
    "    pba.update.remote(1)\n",
    "    # return list\n",
    "    return index_p\n",
    "\n",
    "\n",
    "def vectorize_checkpoint(checkpoint):\n",
    "    out = []\n",
    "    # use only weights and biases\n",
    "    for key in checkpoint.keys():\n",
    "        if \"weight\" in key:\n",
    "            w = checkpoint[key]\n",
    "            # flatten to out_channels x n\n",
    "            w = w.view(w.shape[0], -1)\n",
    "            # cat biases to channels if they exist in checkpoint\n",
    "            if key.replace(\"weight\", \"bias\") in checkpoint:\n",
    "                b = checkpoint[key.replace(\"weight\", \"bias\")]\n",
    "                w = torch.cat([w, b.unsqueeze(dim=1)], dim=1)\n",
    "            # split weights in slices along output channel dims\n",
    "            w = torch.split(w, w.shape[0])\n",
    "            # extend out with new tokens, zero's (and only entry) is a list\n",
    "            out.extend(w[0])\n",
    "\n",
    "    return out\n",
    "\n",
    "\n",
    "def vector_to_checkpoint(vector, reference_checkpoint):\n",
    "    # make copy to prevent memory management issues\n",
    "    checkpoint = copy.deepcopy(reference_checkpoint)\n",
    "    # use only weights and biases\n",
    "    idx_start = 0\n",
    "    for key in checkpoint.keys():\n",
    "        if \"weight\" in key:\n",
    "            # get correct slice of modules out of vec sequence\n",
    "            out_channels = checkpoint[key].shape[0]\n",
    "            idx_end = idx_start + out_channels\n",
    "            w = vector[idx_start:idx_end]\n",
    "            # get weight matrix from list of vectors\n",
    "            try:\n",
    "                w = torch.stack(w, dim=0)\n",
    "            except Exception as e:\n",
    "                logging.error(\n",
    "                    f\"vector_to_checkpoint: layer {key}, idx: [{idx_start},{idx_end}] created weight {w.shape} for checkpoint weight {checkpoint[key].shape}\"\n",
    "                )\n",
    "                logging.error(e)\n",
    "            # extract bias\n",
    "            if key.replace(\"weight\", \"bias\") in checkpoint:\n",
    "                b = w[:, -1]\n",
    "                checkpoint[key.replace(\"weight\", \"bias\")] = b\n",
    "                w = w[:, :-1]\n",
    "            # reshape weight vector\n",
    "            w = w.view(checkpoint[key].shape)\n",
    "            logging.debug(\n",
    "                f\"vector_to_checkpoint: layer {key}, idx: [{idx_start},{idx_end}] tried to create weights from {[wdx.shape] for wdx in w} for checkpoint weight {checkpoint[key].shape}\"\n",
    "            )\n",
    "            checkpoint[key] = w\n",
    "            # update start\n",
    "            idx_start = idx_end\n",
    "    return checkpoint\n",
    "\n",
    "\n",
    "def get_position_mapping_from_checkpoint(checkpoint) -> list:\n",
    "    \"\"\"\n",
    "    Args:\n",
    "        checkpoint: Collections.OrderedDict model checkpoint\n",
    "    Returns:\n",
    "        output tensor with 2d positions for every token in the vectorized model sequence\n",
    "    \"\"\"\n",
    "    out = []\n",
    "    # start layer index counter at 0\n",
    "    idx = 0\n",
    "    # iterate over modules\n",
    "    for key in checkpoint.keys():\n",
    "        # if module with weights is found -> add index\n",
    "        if \"weight\" in key:\n",
    "            w = checkpoint[key]\n",
    "            # create tuple (layer_idx, channel_idx) for every channel\n",
    "            idx_layer = [torch.tensor([idx, jdx]) for jdx in range(w.shape[0])]\n",
    "            # add to overall position\n",
    "            out.extend(idx_layer)\n",
    "            # increase layer counter\n",
    "            idx += 1\n",
    "    out = torch.stack(out, dim=0)\n",
    "    return out\n",
    "\n",
    "\n",
    "def tokenize(t: List[torch.tensor], tokensize: int, return_mask: bool = True):\n",
    "    \"\"\"\n",
    "    transforms list of tokens of differen lenght to tensor\n",
    "    Args:\n",
    "        t: List[torch.tensor]: list of 1d input tokens of different lenghts\n",
    "        tokensize: int output dimension of each token\n",
    "        return_mask: bool wether to return the mask of nonzero values\n",
    "    Returns:\n",
    "        tokens: torch.tensor with tokens stacked along dim=0\n",
    "        mask: torch.tensor indicating the shape of the original tokens\n",
    "    \"\"\"\n",
    "    # init output with zeros\n",
    "    tokens = torch.zeros(len(t), tokensize)\n",
    "    mask = torch.zeros(len(t), tokensize)\n",
    "    # iterate over inputs\n",
    "    for idx, tdx in enumerate(t):\n",
    "        # get end of token, either the length of the input or them maximum length\n",
    "        tdx_end = min(tdx.shape[0], tokensize)\n",
    "        # put at position idx\n",
    "        tokens[idx, :tdx_end] = tdx[:tdx_end]\n",
    "        mask[idx, :tdx_end] = torch.ones(tdx_end)\n",
    "\n",
    "    # return\n",
    "    if return_mask:\n",
    "        return tokens, mask\n",
    "    else:\n",
    "        return tokens\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f8188133",
   "metadata": {},
   "outputs": [],
   "source": [
    "def permute_model_tensor(\n",
    "    vec: List[torch.Tensor], perm: List[tuple], idx_start: int, window: int\n",
    "):\n",
    "    \"\"\"\n",
    "    performs permutation on vectorized model and returns slice of tokens\n",
    "    input vec: list(torch.tensor) with the weights per output channel of the full model\n",
    "    input perm: contains two pieces of information for permutation. perm[0] is the global permutation of the tokens, perm[1] contains the permutaion mappings within tokens per token\n",
    "    input idx_start: int marks the start of a slice of tokens to keep\n",
    "    input window: int marks the size of the slice to keep\n",
    "    return vec: list(torch.tensor) of permuted and sliced tokens\n",
    "    \"\"\"\n",
    "    # create index vector of tokens\n",
    "    index = list(range(len(vec)))\n",
    "\n",
    "    # apply global permutation on index\n",
    "    # using (slices of) the permuted index to access tokens equals permuting all tokens and slicing after\n",
    "    perm_glob = perm[0]\n",
    "    index = [index[idx] for idx in perm_glob]\n",
    "\n",
    "    # slice index\n",
    "    idx_end = idx_start + window\n",
    "    index = index[idx_start:idx_end]\n",
    "\n",
    "    # slice token sequence\n",
    "    vec = [vec[idx] for idx in index]\n",
    "\n",
    "    # slice permutations\n",
    "    perm_loc = perm[1]\n",
    "    perm_loc = [perm_loc[idx] for idx in index]\n",
    "\n",
    "    # apply token permutation\n",
    "    vec = [vecdx[permdx] for (vecdx, permdx) in zip(vec, perm_loc)]\n",
    "\n",
    "    # return tokens\n",
    "    return vec"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "0e595c70",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2023-04-12 03:46:52,774\tINFO worker.py:1553 -- Started a local Ray instance.\n",
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 2800/2800 [00:08<00:00, 337.88it/s]\n",
      "700it [00:12, 57.53it/s]\n",
      "2023-04-12 03:47:29,804\tINFO worker.py:1553 -- Started a local Ray instance.\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "preparing computing canon form...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 700/700 [00:02<00:00, 272.80it/s]\n"
     ]
    }
   ],
   "source": [
    "config_key_list = []\n",
    "result_key_list = [\n",
    "    \"test_acc\",\n",
    "    \"training_iteration\",\n",
    "    \"ggap\",\n",
    "    #     \"sparsity_ratio\",\n",
    "]\n",
    "property_keys = {\n",
    "    \"result_keys\": result_key_list,\n",
    "    \"config_keys\": config_key_list,\n",
    "}\n",
    "\n",
    "path_root = Path(\"/netscratch2/dtaskiran/zoos/SVHN/tune_zoo_svhn_uniform/\")\n",
    "\n",
    "dataset_tensor = SimCLRDatasetTensor(\n",
    "    root=path_root,\n",
    "    epoch_lst=[1, 5, 20, 25],\n",
    "#     epoch_lst=list(range(1, 26)),\n",
    "    #     mode=\"checkpoint\",\n",
    "    mode=\"vector\",\n",
    "    permutations_number=0,\n",
    "    permutation_spec=zoo_cnn_permutation_spec(),\n",
    "    #     view_1_canonical = False,\n",
    "    view_1_canonical=True,\n",
    "    view_2_canonical=True,\n",
    "    #     view_2_canonical = False,\n",
    "    add_noise_view_1=0.1,  # [('input', 0.15), ('output', 0.013)]\n",
    "    add_noise_view_2=0.0,  # [('input', 0.15), ('output', 0.013)]\n",
    "    noise_multiplicative=True,\n",
    "    erase_augment=None,  # {\"p\": 0.5,\"scale\":(0.02,0.33),\"value\":0,\"mode\":\"block\"}\n",
    "    windowsize=12,\n",
    "    standardize=True,\n",
    "    train_val_test=\"train\",  # determines whcih dataset split to use\n",
    "    ds_split=[0.7, 0.15, 0.15],  #\n",
    "    max_samples=1000,\n",
    "    # weight_threshold=float(\"inf\"),\n",
    "    weight_threshold=15,\n",
    "    filter_function=None,  # gets sample path as argument and returns True if model needs to be filtered out\n",
    "    property_keys=property_keys,\n",
    "    num_threads=6,\n",
    "    shuffle_path=True,\n",
    "    verbosity=3,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "e0d0577a",
   "metadata": {},
   "outputs": [],
   "source": [
    "max_int = len(dataset_tensor)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "578ed3b9",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "91 µs ± 645 ns per loop (mean ± std. dev. of 7 runs, 10,000 loops each)\n"
     ]
    }
   ],
   "source": [
    "%%timeit\n",
    "for idx in range(64):\n",
    "    kdx = random.randint(0, max_int)\n",
    "#     (x_i, m_i, l_i, x_j, m_j, _, p) = dataset_tensor.__getitem__(kdx)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "5f072f44",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "4.41 s ± 531 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)\n"
     ]
    }
   ],
   "source": [
    "%%timeit\n",
    "for idx in range(64):\n",
    "    kdx = random.randint(0, max_int-1)\n",
    "    (x_i, m_i, l_i, x_j, m_j, _, p) = dataset_tensor.__getitem__(kdx)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "559dcf0b",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.8.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
