{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-05-01T22:03:57.044082Z",
     "start_time": "2023-05-01T22:03:56.079501Z"
    }
   },
   "outputs": [],
   "source": [
    "import os\n",
    "import sys\n",
    "import logging\n",
    "import torch\n",
    "import transformers\n",
    "from datasets import load_dataset\n",
    "\n",
    "module_path = os.path.abspath(os.path.join('..'))\n",
    "if module_path not in sys.path:\n",
    "    sys.path.append(os.path.join(module_path, 'src'))\n",
    "\n",
    "logging.basicConfig(level=logging.INFO)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-05-01T22:03:59.532197Z",
     "start_time": "2023-05-01T22:03:57.571564Z"
    }
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "WARNING:datasets.builder:Found cached dataset imagenet-1k (/bigdata/cache/huggingface/imagenet-1k/default/1.0.0/a1e9bfc56c3a7350165007d1176b15e9128fcaf9ab972147840529aed3ae52bc)\n",
      "WARNING:datasets.arrow_dataset:Loading cached processed dataset at /bigdata/cache/huggingface/imagenet-1k/default/1.0.0/a1e9bfc56c3a7350165007d1176b15e9128fcaf9ab972147840529aed3ae52bc/cache-602ab40338769b06.arrow\n"
     ]
    }
   ],
   "source": [
    "# Load imagenet 1k\n",
    "dataset = load_dataset(\"imagenet-1k\", split=\"train[:2000]\", cache_dir=\"/bigdata/cache/huggingface\")\n",
    "# Filter the dataset and only take 3-channel images, images are PIL images\n",
    "dataset = dataset.filter(lambda x: x['image'].mode == 'RGB')\n",
    "batch_size = 128"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-05-01T22:04:03.195671Z",
     "start_time": "2023-05-01T22:04:02.021637Z"
    }
   },
   "outputs": [],
   "source": [
    "# Get a pretrained VIT with intermediate features\n",
    "from transformers import AutoProcessor, AutoModel, ViTFeatureExtractor\n",
    "feature_extractor = transformers.ViTFeatureExtractor.from_pretrained('google/vit-base-patch16-224-in21k', cache_dir='/bigdata/cache/models')\n",
    "trained_vit = transformers.ViTModel.from_pretrained(\n",
    "    'google/vit-base-patch16-224-in21k', return_dict=True, output_attentions=True, output_hidden_states=True, cache_dir='/bigdata/cache/models'\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-05-01T22:04:03.487471Z",
     "start_time": "2023-05-01T22:04:03.423149Z"
    }
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "WARNING:datasets.arrow_dataset:Loading cached processed dataset at /bigdata/cache/huggingface/imagenet-1k/default/1.0.0/a1e9bfc56c3a7350165007d1176b15e9128fcaf9ab972147840529aed3ae52bc/cache-c812455f93825cdf.arrow\n"
     ]
    }
   ],
   "source": [
    "def preprocess(examples):\n",
    "    # take a list of PIL images and turn them to pixel values\n",
    "    images = examples['image']\n",
    "    inputs = feature_extractor(images=images, return_tensors='pt')\n",
    "    inputs['label'] = examples['label']\n",
    "    return inputs\n",
    "\n",
    "# preprocess the dataset by calling map function, but do it one by one\n",
    "prepared_train = dataset.map(preprocess, batched=True, batch_size=batch_size).with_format('torch')"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Test ViT/Attnention"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-05-01T22:04:33.897080Z",
     "start_time": "2023-05-01T22:04:33.194059Z"
    }
   },
   "outputs": [],
   "source": [
    "from gnn_models import CustomAttention\n",
    "import torch.nn as nn\n",
    "\n",
    "\n",
    "device = 'cuda'\n",
    "untrained_vit = transformers.ViTModel(\n",
    "    config=trained_vit.config,\n",
    "    add_pooling_layer=True,\n",
    "    use_mask_token=False,\n",
    ")\n",
    "untrained_vit.eval()\n",
    "trained_vit.eval()\n",
    "# cls_token = untrained_vit.embeddings.cls_token\n",
    "d_model = untrained_vit.config.hidden_size\n",
    "att_layer = CustomAttention(\n",
    "    embed_dim=d_model, num_heads=1, batch_first=True,\n",
    "    dv=d_model, do=d_model,\n",
    "    linear_attention=False\n",
    ")\n",
    "att_layer.eval()\n",
    "before_att_ln = nn.LayerNorm(d_model, elementwise_affine=False)\n",
    "\n",
    "trained_vit = trained_vit.to(device)\n",
    "untrained_vit = untrained_vit.to(device)\n",
    "before_att_ln = before_att_ln.to(device)\n",
    "att_layer = att_layer.to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-05-01T22:06:09.338659Z",
     "start_time": "2023-05-01T22:04:35.185373Z"
    }
   },
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "0585136369e74154a7a75171290f3058",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/16 [00:00<?, ?ba/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "def process_embeddings(inputs):\n",
    "    pixel_values = inputs['pixel_values'].to(device)\n",
    "    embeddings = untrained_vit.embeddings(pixel_values=pixel_values)\n",
    "    n = embeddings.size(1)\n",
    "\n",
    "    rand_vit_outputs = untrained_vit(pixel_values=pixel_values, output_hidden_states=True, return_dict=True)\n",
    "    trained_vit_outputs = trained_vit(pixel_values=pixel_values, output_hidden_states=True, return_dict=True)\n",
    "\n",
    "    embeddings_normalized = before_att_ln(embeddings)\n",
    "    rand_attn_encoded, rand_attn_weights = att_layer(query=embeddings_normalized, key=embeddings_normalized, value=embeddings_normalized, attn_mask=torch.zeros((n, n), dtype=torch.float, device=device))\n",
    "\n",
    "    inputs['rand_vit_embeddings'] = embeddings.detach().clone().cpu()\n",
    "    inputs['rand_vit_encoded'] = rand_vit_outputs.hidden_states[1].detach().clone().cpu()\n",
    "    inputs['trained_vit_encoded'] = trained_vit_outputs.hidden_states[1].detach().clone().cpu()\n",
    "    inputs['rand_att_encoded'] = inputs['rand_vit_embeddings'] + rand_attn_encoded.detach().clone().cpu()\n",
    "    inputs['attn_weights'] = rand_attn_weights.detach().clone().cpu()\n",
    "    return inputs\n",
    "\n",
    "with torch.no_grad():\n",
    "    embedded_dataset = prepared_train.map(\n",
    "        process_embeddings,\n",
    "        batched=True, batch_size=batch_size\n",
    "    )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-05-01T22:06:19.873901Z",
     "start_time": "2023-05-01T22:06:19.850271Z"
    }
   },
   "outputs": [],
   "source": [
    "from tqdm import tqdm\n",
    "\n",
    "def get_context_ranks(dataset, key):\n",
    "    # get the ranks of the context\n",
    "    ranks = []\n",
    "    for i in tqdm(range(len(dataset))):\n",
    "        context = dataset[i][key]\n",
    "        ranks.append(torch.linalg.matrix_rank(context))\n",
    "    return torch.tensor(ranks)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-05-01T22:16:53.257163Z",
     "start_time": "2023-05-01T22:06:24.120640Z"
    }
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 1973/1973 [01:17<00:00, 25.40it/s]\n",
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 1973/1973 [01:19<00:00, 24.79it/s]\n",
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 1973/1973 [01:19<00:00, 24.86it/s]\n",
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 1973/1973 [01:19<00:00, 24.75it/s]\n"
     ]
    }
   ],
   "source": [
    "rank_dict = {}\n",
    "for key in ['rand_vit_embeddings', 'rand_vit_encoded', 'rand_att_encoded', 'trained_vit_encoded']:\n",
    "    rank_dict[key] = get_context_ranks(embedded_dataset, key)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-05-01T21:46:50.487881Z",
     "start_time": "2023-05-01T21:46:49.497069Z"
    }
   },
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAOcAAADcCAYAAACLSbZOAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjYuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8o6BhiAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAeoElEQVR4nO3dfVhTdf8H8PcYsiG4rcnDIAFBDYFMu7B0BlpKDvAhjUqEFJH0ttAuQzPpQaBMyp5MbzW7Te3BB6zLvG/RGyXQKENNdGJo/sQwMBsk4CaKE9z394cX5/Y4VKYDvtDndV27Ls73fM85n53tzc7Ddo6EMcZACOGOQ3sXQAhpHoWTEE5ROAnhFIWTEE5ROAnhFIWTEE5ROAnhFIWTEE5ROAnhFIXzDkkkEqSnp9t9vqdPn4ZEIsG6devsPu/WtGfPHkgkEnzzzTetvqz09HRIJJIW9b3xdVq3bh0kEglOnz7dOsXZEYWzA9ixY0er/CMgfKNwcsbPzw/19fWYNGmS0LZjxw5kZGS0Y1Wdx6RJk1BfXw8/P7/2LuW2Ok04L1682N4l2IVEIoFcLodUKm3vUjolqVQKuVze4s3i9tQhw9m0z3Hs2DHExcXhnnvuQVhYGIqLizFlyhQEBARALpdDo9Fg6tSpqK6ubnb60tJSTJkyBSqVCkqlEomJibh06ZKor9lsxksvvQR3d3d069YNY8eOxZkzZ2yqt6GhAWq1GomJiVbjTCYT5HI55s6dC8B6n3PKlClYvnw5gGvBbXrYYv/+/YiMjIRSqUTXrl0xbNgw7N27V9SnaZ383//9H5599lkolUq4u7vjjTfeAGMMFRUVeOKJJ6BQKKDRaPDBBx80u6yrV6/i1VdfhUajgYuLC8aOHYuKioo7qgkAfvzxRzz00EOQy+Xo1asXVq1a1exyW/o6NbfP2bNnT4wePRo//vgjHn74YcjlcgQEBOCLL76wmr64uBjDhg2Ds7MzevTogYULF2Lt2rVW8zx48CB0Oh3c3Nzg7OwMf39/TJ06tdnab8bRpt6cefrpp9GnTx8sWrQIjDHk5ubit99+Q2JiIjQaDUpKSvDpp5+ipKQE+/bts3pTP/PMM/D390dmZiYOHTqE1atXw8PDA++++67Q57nnnsNXX32FuLg4DBkyBPn5+Rg1apRNdXbp0gXjx4/Hli1bsGrVKjg5OQnjtm7dCrPZjNjY2Gan/cc//oGzZ88iNzcXX375pU3LBYD8/HxERUUhNDQUaWlpcHBwwNq1azF8+HD88MMPePjhh0X9J0yYgKCgILzzzjvYvn07Fi5cCLVajVWrVmH48OF49913sX79esydOxcPPfQQhg4dKpr+7bffhkQiwSuvvIKqqiosWbIEERER0Ov1cHZ2tqmmo0ePYuTIkXB3d0d6ejoaGxuRlpYGT09Pq+d5t69TaWkpnnrqKSQlJSEhIQFr1qzBlClTEBoaipCQEADAH3/8gcceewwSiQSpqalwcXHB6tWrIZPJRPOqqqoS6p4/fz5UKhVOnz6NLVu2tLgeAADrgNLS0hgANnHiRFH7pUuXrPpu3LiRAWAFBQVW00+dOlXUd/z48ax79+7CsF6vZwDYCy+8IOoXFxfHALC0tLQW17xz504GgG3btk3UHh0dzQICAoThsrIyBoCtXbtWaEtOTmZ38lJZLBbWp08fptPpmMViEdovXbrE/P392eOPPy60Na2T6dOnC22NjY2sR48eTCKRsHfeeUdor62tZc7OziwhIUFo2717NwPA7r33XmYymYT2zZs3MwDs448/trmmcePGMblczn7//Xeh7dixY0wqlYrWhy2v09q1axkAVlZWJrT5+flZvUeqqqqYTCZjc+bMEdpmzZrFJBIJO3z4sNBWXV3N1Gq1aJ7ffvstA8B+/vlndjc65GZtkxkzZoiGm/4zA8Dly5dx7tw5DB48GABw6NCh204fHh6O6upqmEwmANcOxADAiy++KOo3e/Zsm2sdPnw43NzckJWVJbTV1tYiNzcXEyZMsHl+LaHX63Hy5EnExcWhuroa586dw7lz53Dx4kWMGDECBQUFsFgsommee+454W+pVIqBAweCMYakpCShXaVSITAwEL/99pvVMidPnoxu3boJw0899RS8vLyEddnSmq5evYqdO3di3Lhx8PX1FeYXFBQEnU4nWqY9Xqfg4GCEh4cLw+7u7lbPMScnB1qtFgMGDBDa1Go14uPjRfNSqVQAgOzsbDQ0NLS4hht16M1af39/0XBNTQ0yMjKwadMmVFVVicYZjUar6a9/0QHgnnvuAXAtNAqFAr///jscHBzQq1cvUb/AwECba3V0dERMTAw2bNgAs9kMmUyGLVu2oKGhodXCefLkSQBAQkLCTfsYjUbheQPW60SpVEIul8PN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",
      "text/plain": [
       "<Figure size 200x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 200x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 200x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 200x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "\n",
    "for key, arr in rank_dict.items():\n",
    "    n = embedded_dataset[0]['rand_vit_embeddings'].shape[0]\n",
    "    plt.figure(figsize=(2, 2))\n",
    "    plt.bar(['$<n$', '$=n$'], [sum([r < n for r in arr]), sum([r == n for r in arr])])\n",
    "    plt.title(key)\n",
    "    plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-05-01T22:32:10.657482Z",
     "start_time": "2023-05-01T22:32:10.651400Z"
    }
   },
   "outputs": [],
   "source": [
    "def check_n_kruskal_rank(all_dataset, key, n_rank=None, total_exps=5000, device='cuda'):\n",
    "    encoded_tensor = all_dataset[key]\n",
    "    T, n, d = encoded_tensor.shape\n",
    "    if n_rank is None:\n",
    "        n_rank = n\n",
    "    cls_tensor = encoded_tensor[:, 0, :].to(device)\n",
    "    cnt = 0\n",
    "    rank_dist = {}\n",
    "    for _ in tqdm(range(total_exps)):\n",
    "        # Take a samples of n examples from dataset\n",
    "        random_perm = torch.randperm(T)[:n_rank]\n",
    "        sample = cls_tensor[random_perm, :]\n",
    "        rank = torch.linalg.matrix_rank(sample).item()\n",
    "        if rank == n_rank:\n",
    "            cnt += 1\n",
    "        rank_dist[rank] = rank_dist.get(rank, 0) + 1\n",
    "    return cnt / total_exps, rank_dist"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-05-01T22:35:48.618560Z",
     "start_time": "2023-05-01T22:32:11.002425Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "rand_vit_encoded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:14<00:00, 354.00it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(1.0, {197: 5000})\n",
      "--------------------\n",
      "rand_att_encoded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:14<00:00, 353.07it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(1.0, {197: 5000})\n",
      "--------------------\n",
      "trained_vit_encoded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:14<00:00, 354.37it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(0.9982, {197: 4991, 196: 9})\n",
      "--------------------\n",
      "rand_vit_embeddings\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [02:56<00:00, 28.27it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(0.0, {1: 5000})\n",
      "--------------------\n"
     ]
    }
   ],
   "source": [
    "all_embedded_dataset = embedded_dataset[:]\n",
    "for key in ['rand_vit_encoded', 'rand_att_encoded', 'trained_vit_encoded', 'rand_vit_embeddings']:\n",
    "    print(key)\n",
    "    print(check_n_kruskal_rank(all_embedded_dataset, key))\n",
    "    print('-' * 20)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-05-01T22:31:34.203448Z",
     "start_time": "2023-05-01T22:31:31.400350Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking 768 Kruskal Rank\n",
      "rand_vit_encoded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:54<00:00, 43.64it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(0.0, {715: 887, 719: 341, 714: 482, 718: 750, 713: 197, 717: 1060, 716: 1040, 720: 131, 722: 3, 721: 26, 712: 60, 711: 17, 710: 4, 723: 2})\n",
      "--------------------\n",
      "rand_att_encoded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:49<00:00, 45.65it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(0.0, {472: 769, 473: 717, 470: 565, 469: 388, 468: 227, 467: 123, 478: 69, 474: 591, 466: 46, 471: 676, 476: 243, 475: 388, 477: 133, 479: 27, 465: 20, 481: 3, 463: 3, 482: 1, 480: 8, 464: 3})\n",
      "--------------------\n",
      "trained_vit_encoded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:51<00:00, 44.91it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(0.0, {312: 1105, 311: 399, 314: 1178, 310: 97, 313: 1596, 315: 500, 316: 100, 309: 10, 317: 14, 318: 1})\n",
      "--------------------\n",
      "rand_vit_embeddings\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [02:01<00:00, 41.07it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(0.0, {1: 5000})\n",
      "--------------------\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "all_embedded_dataset = embedded_dataset[:]\n",
    "d_data = all_embedded_dataset['rand_vit_embeddings'][0].shape[-1]\n",
    "print(f\"Checking {d_data} Kruskal Rank\")\n",
    "for key in ['rand_vit_encoded', 'rand_att_encoded', 'trained_vit_encoded', 'rand_vit_embeddings']:\n",
    "    print(key)\n",
    "    print(check_n_kruskal_rank(all_embedded_dataset, key, n_rank=d_data))\n",
    "    print('-' * 20)"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Check the rank for [1,d]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 54,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:01<00:00, 4775.74it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=11\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:03<00:00, 1657.66it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=21\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:04<00:00, 1225.52it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=31\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:04<00:00, 1030.83it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=41\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:07<00:00, 654.08it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=51\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:08<00:00, 575.93it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=61\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:09<00:00, 527.44it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=71\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:12<00:00, 385.29it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=81\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:13<00:00, 359.04it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=91\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:14<00:00, 337.78it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=101\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:19<00:00, 262.27it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=111\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:20<00:00, 248.47it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=121\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:21<00:00, 236.55it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=131\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:22<00:00, 221.22it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=141\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:23<00:00, 216.24it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=151\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:23<00:00, 210.41it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=161\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:28<00:00, 174.93it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=171\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:29<00:00, 169.44it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=181\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:30<00:00, 164.99it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=191\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:30<00:00, 162.05it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=197\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:35<00:00, 140.29it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=201\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:35<00:00, 139.16it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=211\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:36<00:00, 136.39it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=221\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:37<00:00, 134.09it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=231\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:42<00:00, 117.34it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=241\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:43<00:00, 115.46it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=251\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:44<00:00, 113.19it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=261\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:53<00:00, 92.72it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=271\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:54<00:00, 91.34it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=281\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:55<00:00, 89.64it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=291\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:01<00:00, 80.71it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=301\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:02<00:00, 79.78it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=311\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:03<00:00, 78.79it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=321\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:08<00:00, 72.64it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=331\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:09<00:00, 71.67it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=341\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:10<00:00, 70.97it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=351\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:11<00:00, 70.34it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=361\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:17<00:00, 64.25it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=371\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:18<00:00, 63.64it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=381\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:19<00:00, 63.00it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=391\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:32<00:00, 53.97it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=401\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:33<00:00, 53.41it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=411\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:34<00:00, 52.76it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=421\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:42<00:00, 48.80it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=431\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:43<00:00, 48.39it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=441\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:44<00:00, 47.81it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=451\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:51<00:00, 44.99it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=461\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:52<00:00, 44.59it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=471\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [01:53<00:00, 44.16it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=481\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [02:00<00:00, 41.45it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=491\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [02:01<00:00, 41.19it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=501\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [02:02<00:00, 40.67it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=511\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [02:03<00:00, 40.43it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=521\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [02:23<00:00, 34.95it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=531\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [02:23<00:00, 34.80it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=541\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [02:24<00:00, 34.64it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=551\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [02:32<00:00, 32.73it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=561\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [02:33<00:00, 32.58it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=571\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [02:34<00:00, 32.41it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=581\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [02:42<00:00, 30.77it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=591\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [02:43<00:00, 30.59it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=601\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [02:45<00:00, 30.27it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=611\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [02:54<00:00, 28.61it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=621\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [02:56<00:00, 28.35it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=631\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [02:58<00:00, 27.99it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=641\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [03:21<00:00, 24.87it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=651\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [03:22<00:00, 24.71it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=661\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [03:23<00:00, 24.52it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=671\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [03:24<00:00, 24.41it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=681\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [03:34<00:00, 23.29it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=691\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [03:35<00:00, 23.18it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=701\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [03:36<00:00, 23.06it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=711\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [03:46<00:00, 22.03it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=721\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [03:48<00:00, 21.92it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=731\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [03:51<00:00, 21.62it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=741\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [04:02<00:00, 20.61it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=751\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [04:10<00:00, 19.98it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=761\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [04:38<00:00, 17.98it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checking d=768\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [05:34<00:00, 14.96it/s]\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "from tqdm import tqdm\n",
    "\n",
    "# all_embedded_dataset = embedded_dataset[:]\n",
    "# _, n_data, d_data = all_embedded_dataset['rand_vit_embeddings'].shape\n",
    "# kruskal_dist = {\n",
    "#     'rand_att_encoded': []\n",
    "# }\n",
    "# for key in kruskal_dist:\n",
    "#     for d in sorted(list(set(range(1, d_datt\n",
    "#                                    a+1, 10)) | set([n_data, d_data]))):\n",
    "#         print(f\"Checking d={d}\")\n",
    "#         fract, _ = check_n_kruskal_rank(all_embedded_dataset, key, n_rank=d)\n",
    "#         kruskal_dist['rand_att_encoded'].append((d, fract))\n",
    "# np.save('kruskal.npy', kruskal_dist)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 96,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 450x300 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "results = np.load('kruskal.npy', allow_pickle=True).item()\n",
    "translations = {\n",
    "    'rand_att_encoded': 'Random Attention'\n",
    "}\n",
    "plt.figure(figsize=(4.5, 3))\n",
    "colors = ['#19647e']\n",
    "for i, key in enumerate(results.keys()):\n",
    "    x, y = zip(*tuple(results[key]))\n",
    "    plt.plot(x, y, color=colors[i], marker='.', markersize=8)\n",
    "plt.axvline(x=n_data, color='g', linestyle='--', label='$n$')\n",
    "plt.axvline(x=d_data, color='r', linestyle='--', label='$d$')\n",
    "plt.xlabel('Kruskal Rank Candidate')\n",
    "plt.ylabel('Satisfied Fraction')\n",
    "plt.legend()\n",
    "plt.tight_layout()\n",
    "plt.savefig('/data/smahdavi/projects/in-context/aux/rand_att_krus_rank.pdf')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 97,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "197\n"
     ]
    }
   ],
   "source": [
    "print(n_data)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 98,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "768\n"
     ]
    }
   ],
   "source": [
    "print(d_data)"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Check Assumption 3"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 106,
   "metadata": {},
   "outputs": [],
   "source": [
    "assp3_out = (\n",
    "    all_embedded_dataset['rand_vit_embeddings'][:, 0, :] + all_embedded_dataset['rand_vit_embeddings'].mean(dim=1)\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 112,
   "metadata": {},
   "outputs": [],
   "source": [
    "def minimal_check_n_kruskal_rank(arr, n_rank, total_exps=5000, device='cuda'):\n",
    "    arr = arr.to(device)\n",
    "    T = arr.shape[0]\n",
    "    cnt = 0\n",
    "    rank_dist = {}\n",
    "    for _ in tqdm(range(total_exps)):\n",
    "        random_perm = torch.randperm(T)[:n_rank]\n",
    "        sample = arr[random_perm, :]\n",
    "        rank = torch.linalg.matrix_rank(sample).item()\n",
    "        if rank == n_rank:\n",
    "            cnt += 1\n",
    "        rank_dist[rank] = rank_dist.get(rank, 0) + 1\n",
    "    return cnt / total_exps, rank_dist"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 113,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 5000/5000 [00:35<00:00, 140.90it/s]\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "(1.0, {197: 5000})"
      ]
     },
     "execution_count": 113,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "minimal_check_n_kruskal_rank(assp3_out, n_rank=n_data)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "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": 1
}
