{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "68de9170-e954-4358-911c-a8d54e21a84a",
   "metadata": {},
   "outputs": [],
   "source": [
    "import sys\n",
    "import torch\n",
    "import numpy as np\n",
    "from matplotlib import pyplot as plt\n",
    "import json\n",
    "from itertools import product\n",
    "sys.path.append('/home/XXXX-1/Finite-groups/src')\n",
    "from model import MLP3\n",
    "from evals import load_model\n",
    "from utils import dotdict\n",
    "from group_data import *\n",
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "4f8cdf72-7889-4a7b-ab5b-45ca2e6c725b",
   "metadata": {},
   "outputs": [],
   "source": [
    "params = dotdict(dict(\n",
    "    group_string='Z(100);Z(50,2)',\n",
    "    delta_frac=0.,\n",
    "    intersect_frac=1.,\n",
    "    train_frac=1.,\n",
    "))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "e721aaf4-d161-4bd7-a51e-d28f9f686b07",
   "metadata": {},
   "outputs": [
    {
     "ename": "KeyError",
     "evalue": "'instances'",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyError\u001b[0m                                  Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[5], line 1\u001b[0m\n\u001b[0;32m----> 1\u001b[0m m \u001b[38;5;241m=\u001b[39m \u001b[43mMLP3\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m5\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mparams\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m~/Finite-groups/src/model.py:80\u001b[0m, in \u001b[0;36mMLP3.__init__\u001b[0;34m(self, N, params)\u001b[0m\n\u001b[1;32m     75\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mparams \u001b[38;5;241m=\u001b[39m params\n\u001b[1;32m     76\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mN \u001b[38;5;241m=\u001b[39m N\n\u001b[1;32m     78\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39membedding_left \u001b[38;5;241m=\u001b[39m custom_kaiming(\n\u001b[1;32m     79\u001b[0m     [\n\u001b[0;32m---> 80\u001b[0m         \u001b[43mparams\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43minstances\u001b[49m,\n\u001b[1;32m     81\u001b[0m         \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mN,\n\u001b[1;32m     82\u001b[0m         params\u001b[38;5;241m.\u001b[39membed_dim,\n\u001b[1;32m     83\u001b[0m     ]\n\u001b[1;32m     84\u001b[0m )\n\u001b[1;32m     86\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39membedding_right \u001b[38;5;241m=\u001b[39m custom_kaiming(\n\u001b[1;32m     87\u001b[0m     [params\u001b[38;5;241m.\u001b[39minstances, \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mN, params\u001b[38;5;241m.\u001b[39membed_dim]\n\u001b[1;32m     88\u001b[0m )\n\u001b[1;32m     90\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mlinear \u001b[38;5;241m=\u001b[39m custom_kaiming(\n\u001b[1;32m     91\u001b[0m     [params\u001b[38;5;241m.\u001b[39minstances, params\u001b[38;5;241m.\u001b[39membed_dim, params\u001b[38;5;241m.\u001b[39mhidden_size]\n\u001b[1;32m     92\u001b[0m )\n",
      "\u001b[0;31mKeyError\u001b[0m: 'instances'"
     ]
    }
   ],
   "source": [
    "gd = GroupData(params)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "2fab4d00-567e-4c9c-941f-fad930abd3e8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA1IAAAIjCAYAAAAJLyrXAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjkuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/TGe4hAAAACXBIWXMAAA9hAAAPYQGoP6dpAAA/g0lEQVR4nO3deXSURb7G8afJTkgIWzbZAqJsomxiBMeFYAD1EmDGQcOwiDKjQUF0EGZkCYosDoggENcgMwjKCKhcUTFsghDZ3DGAoiAhAQUSFhMgXfcPD31pk0gqhHQn+X7O6XPseqvf/nWn5tVnqt8qhzHGCAAAAABQYtU8XQAAAAAAVDQEKQAAAACwRJACAAAAAEsEKQAAAACwRJACAAAAAEsEKQAAAACwRJACAAAAAEsEKQAAAACwRJACAAAAAEsEKQBAlbJ27Vo5HA6tXbvW06W4+fe//63mzZvLz89PYWFhni4HAHABBCkAuITmz58vh8NR5GP06NGeLq9S+e13HRgYqCuuuELDhg1TdnZ2mbzHu+++qwkTJpTJuc73zTffaNCgQWratKlefPFFvfDCC8X2nTBhghwOh3766Sfr98nMzNSECRP06aefXkS1AABJ8vV0AQBQFUycOFExMTFuba1bt/ZQNZXbue86Ly9PGzZs0Lx58/Tuu+/qyy+/VPXq1S/q3O+++67mzJlT5mFq7dq1cjqdevbZZ3X55ZeX6bnPl5mZqeTkZDVu3FjXXHPNJXsfAKgKCFIAUA569OihDh06lKhvXl6e/P39Va0aPxoojfO/63vvvVd16tTRjBkz9NZbb+muu+7ycHVFO3TokCTxk77znDp16qKDLwBcSvxbGgA86Nz9OosXL9bjjz+uyy67TNWrV1dubq4kKT09Xd27d1fNmjVVvXp13Xjjjdq4cWOh82zYsEEdO3ZUYGCgmjZtqueff971E7Bzvv/+ezkcDs2fP7/Q6x0OR6FZlgMHDuiee+5RRESEAgIC1KpVK73yyitF1v/GG29o0qRJql+/vgIDA9W1a1ft2bOn0Pukp6erZ8+eqlWrloKDg9WmTRs9++yzkqTU1FQ5HA7t2LGj0Oueeuop+fj46MCBAxf8Tn/rlltukSTt3bv3d/stWbJE7du3V1BQkOrWrav+/fu7vd+gQYM0Z84cSXL7CeGFzJ07V61atVJAQICio6OVlJSkY8eOuY43btxY48ePlyTVq1evyL/Fhdx0001q3bq1vv76a918882qXr26LrvsMk2bNs3VZ+3aterYsaMkafDgwa76zx8PJR1va9euVYcOHX53vJ3zn//8x/W91q5dW/369dP+/fuLrH/btm36wx/+oOrVq+sf//iHJGnr1q2Kj49X3bp1FRQUpJiYGN1zzz1W3w8AXArMSAFAOcjJySl0T0vdunVd//zEE0/I399fjz76qPLz8+Xv76/Vq1erR48eat++vcaPH69q1aopNTVVt9xyiz766CNde+21kqQvvvhCt956q+rVq6cJEybo7NmzGj9+vCIiIkpdb3Z2tq677jo5HA4NGzZM9erV08qVKzVkyBDl5uZqxIgRbv2nTJmiatWq6dFHH1VOTo6mTZumxMREpaenu/qsWrVKt99+u6KiojR8+HBFRkZq586dWrFihYYPH64//vGPSkpK0sKFC9W2bVu38y9cuFA33XSTLrvsMuvP8u2330qS6tSpU2yf+fPna/DgwerYsaMmT56s7OxsPfvss9q4caN27NihsLAw/fWvf1VmZqZWrVqlf//73yV67wkTJig5OVlxcXG6//77lZGRoXnz5mnLli3auHGj/Pz8NHPmTC1YsEDLli3TvHnzVKNGDbVp08b6cx49elTdu3dXnz59dOedd+q///2vHnvsMV111VXq0aOHWrRooYkTJ2rcuHEaOnSobrjhBknS9ddfL0klHm87duxQ9+7dFRUVpeTkZBUUFGjixImqV69eoZomTZqksWPH6s4779S9996rw4cPa/bs2frDH/7g+l7P+fnnn9WjRw/169dP/fv3V0REhA4dOuQa26NHj1ZYWJi+//57LV261Pr7AYAyZwAAl0xqaqqRVOTDGGPWrFljJJkmTZqYU6dOuV7ndDpNs2bNTHx8vHE6na72U6dOmZiYGNOtWzdXW0JCggkMDDQ//PCDq+3rr782Pj4+5vzL/N69e40kk5qaWqhOSWb8+PGu50OGDDFRUVHmp59+cuvXr18/U7NmTVet5+pv0aKFyc/Pd/V79tlnjSTzxRdfGGOMOXv2rImJiTGNGjUyR48edTvn+Z/vrrvuMtHR0aagoMDVtn379mLrPt+57/rDDz80hw8fNvv37zeLFy82derUMUFBQebHH390q3nNmjXGGGNOnz5twsPDTevWrc0vv/ziOt+KFSuMJDNu3DhXW1JSkinpvzoPHTpk/P39za233ur2eZ577jkjybzyyiuutvHjxxtJ5vDhwxc8b1F9b7zxRiPJLFiwwNWWn59vIiMjTd++fV1tW7ZsKfK7tBlvd9xxh6levbo5cOCAq2337t3G19fX7bv5/vvvjY+Pj5k0aZLbe33xxRfG19fXrf1c/SkpKW59ly1bZiSZLVu2XPB7AYDyxk/7AKAczJkzR6tWrXJ7nG/gwIEKCgpyPf/000+1e/du3X333fr555/1008/6aefftLJkyfVtWtXrV+/Xk6nUwUFBXr//feVkJCghg0bul7fokULxcfHl6pWY4zefPNN3XHHHTLGuN77p59+Unx8vHJycrR9+3a31wwePFj+/v6u5+dmO7777jtJv85i7N27VyNGjCh0H9D5PwcbMGCAMjMztWbNGlfbwoULFRQUpL59+5ao/ri4ONWrV08NGjRQv379VKNGDS1btqzY2aytW7fq0KFDeuCBBxQYGOhqv+2229S8eXP97//+b4ne97c+/PBDnT59WiNGjHC73+2+++5TaGhoqc9bnBo1aqh///6u5/7+/rr22mtdf4PfYzPePvzwQyUkJCg6Otr1+ssvv1w9evRwO+fSpUvldDp15513uo2hyMhINWvWzO1vLEkBAQEaPHiwW9u5sbJixQqdOXPG9isBgEuKn/YBQDm49tprf3exid+u6Ld7925Jvwas4uTk5Cg/P1+//PKLmjVrVuj4lVdeqXfffde61sOHD+vYsWN64YUXil2G+9ziCOecH+IkqVatWpJ+/bmZ9P8/r7vQSoXdunVTVFSUFi5cqK5du8rpdGrRokXq1auXQkJCSlT/nDlzdMUVV8jX11cRERG68sorf3fhjh9++EHSr9/XbzVv3lwbNmwo0fuW9Lz+/v5q0qSJ63hZqV+/fqF7lGrVqqXPP//8gq8t6XjLy8vTL7/8UuTKgr9t2717t4wxRY5NSfLz83N7ftlll7mFcUm68cYb1bdvXyUnJ+uZZ57RTTfdpISEBN19990KCAi44OcCgEuJIAUAXuD82ShJcjqdkqSnn3662GWqa9Soofz8/BK/R3ELIxQUFBT53v379y/2P6x/ew+Pj49Pkf2MMSWu79x57r77br344ouaO3euNm7cqMzMTLeZlgu5UGitrC7mb1DS8ZaXl1fiepxOpxwO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+uqrGjlypNLT07Vp0yYNGjRInTt3Vrdu3bRlyxaFh4crNTVV3bt3l4+PjyTpo48+0oABAzRr1izdcMMN+vbbbzV06FBJ0vjx41VQUKCEhAQ1bNhQ6enpOn78uB555BG39z1z5ozi4+MVGxurjz76SL6+vnryySfVvXt3ff755/L395ckpaWlKTQ0VKtWrZIkbd26VQ899JD+/e9/6/rrr9eRI0f00UcfleM3BgC4FAhSAIAKpU2bNho/frwkqVmzZnruueeUlpambt26qV69epKksLAwRUZGul6TnJys0aNHa+DAgZKkJk2a6IknntCoUaM0fvx4rVq1St9++63Wrl3ret2kSZPUrVs31zlef/11OZ1OvfTSS3I4HJKk1NRUhYWFae3atbr11lslScHBwXrppZdcwWrp0qUKDg7W7bffrpCQEDVq1Eht27a9xN8SAOBSI0gBACqUNm3auD2PiorSoUOHfvc1n332mTZu3KhJkya52goKCpSXl6dTp04pIyNDDRo0cAtf1157baFz7NmzRyEhIW7teXl5+vbbb13Pr7rqKleIkqRu3bqpUaNGatKkibp3767u3burd+/eql69esk/NADA6xCkAAAVip+fn9tzh8Mhp9P5u685ceKEkpOT1adPn0LHAgMDS/S+J06cUPv27bVw4cJCx87NhEm/zkidLyQkRNu3b9fatWv1wQcfaNy4cZowYYK2bNmisLCwEr03AMD7EKQAAJWKn5+fCgoK3NratWunjIwMXX755UW+5sorr9T+/fuVnZ2tiIgISdKWLVsKneP1119XeHi4QkNDrWry9fVVXFyc4uLiNH78eIWFhWn16tVFBjsAQMXAqn0AgEqlcePGSktLU1ZWlo4ePSpJGjdunBYsWKDk5GR99dVX2rlzpxYvXqzHH39c0q8/v2vatKkGDhyozz//XBs3bnQdO3c/VGJiourWratevXrpo48+0t69e7V27Vo99NBD+vHHH4utZ8WKFZo1a5Y+/fRT/fDDD1qwYIGcTqdrRT8AQMVEkAIAVCrTp0/XqlWr1KBBA9eiDvHx8VqxYoU++OADdezYUdddd52eeeYZNWrUSJLk4+Oj5cuX68SJE+rYsaPuvfde/fOf/5T0/z/9q169utavX6+GDRuqT58+atGihYYMGaK8vLzfnaEKCwvT0qVLdcstt6hFixZKSUnRokWL1KpVq0v8TQAALiWHMcZ4uggAALzNxo0b1aVLF+3Zs0dNmzb1dDkAAC9DkAIAQNKyZctUo0YNNWvWTHv27NHw4cNVq1YtbdiwwdOlAQC8EItNAAAg6fjx43rssce0b98+1a1bV3FxcZo+fbqnywIAeClmpAAAAADAEotNAAAAAIAlghQAAAAAWCJIAQAAAIAlghQAAAAAWCJIAQAAAIAlghQAAAAAWCJIAQAAAIAlghQAAAAAWPo/wlrHqy0iRSAAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 1000x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "gd.frequency_histogram()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "46be826a",
   "metadata": {},
   "outputs": [],
   "source": [
    "prefix = '/workspace/Finite-groups/src/models'\n",
    "#path = f'{prefix}/2024_07_21_21_21_51_Z_50__clamped_50_'\n",
    "#path = f'{prefix}/2024_07_21_22_38_07_Z_100__Z_2_50_'\n",
    "path = f'{prefix}/2024_07_21_22_48_54_Z_48_2__twZ_48_'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "2d01bfef",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Intersection size: 6912/9216 (0.75)\n",
      "Added 6912 elements from intersection\n",
      "Added 0 elements from group 0: Z(48,2)\n",
      "Added 0 elements from group 1: twZ(48)\n",
      "Taking random subset: 6912/6912 (1.00)\n",
      "Train set size: 6912\n"
     ]
    }
   ],
   "source": [
    "fig = load_model(path, 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "404c979a",
   "metadata": {},
   "outputs": [],
   "source": [
    "fig.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "5a7d87bc-e44c-49ad-a8ad-50a2b46b5080",
   "metadata": {},
   "outputs": [],
   "source": [
    "loss_dict = torch.load('/workspace/Finite-groups/src/models/2024-07-13_00-37-04_long_run/losses/007940.pt')\n",
    "loss_dict = {k: v.detach().cpu().numpy() for k, v in loss_dict.items()}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a9d8fa1e",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "96abb17b-043a-45e2-b469-bd1015c1cbc3",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "dict_keys(['G1_loss', 'G2_loss', 'G1_accuracy', 'G2_accuracy', 'epoch_train_loss', 'epoch_train_acc', 'epoch_train_margin'])"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "loss_dict.keys()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "dcd7a412-46e8-4767-8c51-18a3f6543c98",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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oz0cDGP06NFqamrSn//8Z9XV1amiokK333675s6dG+2hYYArV67okUce0datW5WZmRnt4cQsQg3CLjMzU0lJSWpra/N4vK2tjSLgGBTM+/XCCy/o2Wef1f79+/XpT386nMPEDQJ9z6xWq+644w5JUn5+vo4fP67q6mpCTQQYfc9OnTqlM2fOaNGiRe7HHA6HJGnEiBE6ceKEpk6dGt5BxwFqahB2ycnJKigoUF1dnfsxh8Ohuro6FRcXR3Fk8CbQ9+uf//mf9cwzz6impkazZs2KxFDxV6H6/5jD4VBPT084hogBjL5n06dP1+9+9zs1NTW5vxYvXqzPf/7zampqUk5OTiSHH7OYqUFEVFRUaOnSpZo1a5YKCwu1YcMGdXV1admyZZKkJUuWaOLEiaqurpbUX0R37Ngx9/9uaWlRU1OTbrnlFvcnS4SP0ffrueee07p167R9+3ZNnjxZdrtdknTLLbfolltuidrvkUiMvmfV1dWaNWuWpk6dqp6eHr3xxhv6+c9/rs2bN0fz10goRt6z1NRU5eXleTx/zJgxkjTo8URGqEFElJWV6eLFi1q3bp3sdrvy8/NVU1PjLpI7d+6crNaPJw7Pnz+vO++80/39Cy+8oBdeeEH33nuvfv3rX0d6+AnH6Pu1efNmXbt2TQ888IDH61RVVenpp5+O5NATltH3rKurS9/61rf0pz/9STfffLOmT5+ubdu2qaysLFq/QsIx+p7BN4vT6XRGexAAAADBIgICAABTINQAAABTINQAAABTINQAAABTINQAAABTINQAAABTINQAAABTINQAAABTINQAAABTINQAAABTINQAAABTINQAAABT+P8Bv2a6OBqEufcAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.scatter(loss_dict['G1_loss'], loss_dict['epoch_train_loss'])\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "4d156a45-95ce-4c6c-9990-8d8bf2fb23f8",
   "metadata": {},
   "outputs": [],
   "source": [
    "path = '/home/XXXX-1/Finite-groups/src/models/2024-07-13_19-30-43_four'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "e7d39bce-0448-4dfd-af65-32a712e39e13",
   "metadata": {},
   "outputs": [],
   "source": [
    "params = json.load(open(f'{path}/params.json', 'r'))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "f3a357bd-e526-49e5-9ba9-5da31956ac16",
   "metadata": {},
   "outputs": [],
   "source": [
    "params = dotdict(params)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "15b15cf4-7d24-488e-ba67-5b57a57a563c",
   "metadata": {},
   "outputs": [],
   "source": [
    "test_inputs = torch.tensor(list(product(range(params.N), repeat=2)), device='cpu')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "715f5f80-f6a6-4f67-b5fa-41d3b18c6ed2",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<All keys matched successfully>"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model = MLP3(params)\n",
    "state = torch.load(f'{path}/ckpts/001998.pt')\n",
    "model.load_state_dict(state)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "07dc1aa6-cfb0-4a70-b5ad-ee5d78241fdf",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([100, 2, 8])"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model.linear.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "695f07c7-5c06-43cb-a1b2-a87cbcdb5206",
   "metadata": {},
   "outputs": [],
   "source": [
    "logits = model(test_inputs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "5d947816-299e-4c57-aa96-aeb4ab64bd5f",
   "metadata": {},
   "outputs": [],
   "source": [
    "loss = torch.load(f'{path}/losses/001998.pt')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "2f512fa5-d73e-4650-97a0-f6db2fdccd0d",
   "metadata": {},
   "outputs": [],
   "source": [
    "good = (loss['epoch_train_acc'] == 1).nonzero()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "3430bb2a-fd56-450c-b3ea-02aea60e507f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 2, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 0, 3, 2],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 0]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 0],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 0, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 0, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 0, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 0],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 2]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 2, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 2],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 0, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 2, 3, 0],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 0],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 2]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 2],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 2, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 0]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 0, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 0, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 2, 3, 2],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 0],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 0, 1, 0]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 0, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 0, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 2]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 2],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 0, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 2],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 0],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 2, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 0, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 2]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 0, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 2, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 0, 1, 2]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 2],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 2, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 0, 3, 0],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 0]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 2, 3, 2],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 2]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 2]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 2, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 0, 1, 1]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 2, 3, 0],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 2, 1, 2]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 1, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 0]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 1],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 3, 1, 3]])\n",
      "tensor([[0, 1, 2, 3],\n",
      "        [1, 3, 3, 3],\n",
      "        [2, 3, 0, 1],\n",
      "        [3, 1, 1, 2]])\n"
     ]
    }
   ],
   "source": [
    "for i in good.cpu().numpy().flatten():\n",
    "    print(logits.max(dim=2).indices[:,i].reshape((4, 4)))#[1::2,1::2])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "id": "dba5ac08-8b1a-4ed4-9919-5689fad7d9f7",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([0, 0], device='cuda:0')"
      ]
     },
     "execution_count": 47,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "test_inputs[0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 51,
   "id": "ebb66e7f-08a6-45d0-8fee-6bcfdfe397a8",
   "metadata": {},
   "outputs": [],
   "source": [
    "mult_table = np.zeros((96, 96))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 53,
   "id": "32d33b3b-5ab1-4f2a-beeb-7ef754620ef2",
   "metadata": {},
   "outputs": [
    {
     "ename": "RuntimeError",
     "evalue": "a Tensor with 96 elements cannot be converted to Scalar",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mRuntimeError\u001b[0m                              Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[53], line 5\u001b[0m\n\u001b[1;32m      3\u001b[0m i \u001b[38;5;241m=\u001b[39m i\u001b[38;5;241m.\u001b[39mitem()\n\u001b[1;32m      4\u001b[0m j \u001b[38;5;241m=\u001b[39m j\u001b[38;5;241m.\u001b[39mitem()\n\u001b[0;32m----> 5\u001b[0m mult_table[i, j] \u001b[38;5;241m=\u001b[39m \u001b[43mlogits\u001b[49m\u001b[43m[\u001b[49m\u001b[43mn\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43minstance\u001b[49m\u001b[43m]\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mitem\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n",
      "\u001b[0;31mRuntimeError\u001b[0m: a Tensor with 96 elements cannot be converted to Scalar"
     ]
    }
   ],
   "source": [
    "instance = 0\n",
    "for n, (i, j) in enumerate(test_inputs):\n",
    "    i = i.item()\n",
    "    j = j.item()\n",
    "    mult_table[i, j] = logits[n, instance].item()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "465b74f3-bf6c-429a-b3cf-0a52fb09beb3",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "277304e0-8886-4d81-9409-693ec7459774",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "337d0370-0923-4bba-b908-2828416a5d98",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2a59fa27-5c8c-4d3a-aeb8-4cefd680aa79",
   "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.10.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
