{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "12ad3103-438b-4539-bb06-0e916168fe99",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.optim as optim\n",
    "from torchvision import models\n",
    "from torchvision.models import ResNet18_Weights\n",
    "from tqdm.notebook import tqdm\n",
    "import numpy as np\n",
    "import torchvision\n",
    "from torch.utils.data import DataLoader, random_split\n",
    "from matplotlib import pyplot as plt\n",
    "import random\n",
    "import os\n",
    "import math\n",
    "from torch.utils.data import TensorDataset, random_split\n",
    "from pprint import pprint"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "40baca94-3599-4867-b6ac-c13d78fc4e61",
   "metadata": {},
   "outputs": [],
   "source": [
    "seed = 42\n",
    "torch.manual_seed(seed)\n",
    "torch.cuda.manual_seed(seed)\n",
    "torch.backends.cudnn.deterministic = True\n",
    "random.seed(seed)\n",
    "np.random.seed(seed)\n",
    "os.environ['PYTHONHASHSEED'] = str(seed)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "04049ac1-9fb6-4f40-8e43-e624c0b6a749",
   "metadata": {},
   "source": [
    "# Dataset Creation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "3d5af2db-610d-442d-a968-f80c58445690",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Statistics\n",
    "TOTAL = 40_000\n",
    "s1 = 0.9\n",
    "s2 = 0.75\n",
    "c1_m_red = c2_fm_green = math.ceil(TOTAL * s1 * s2 * 0.5)\n",
    "c1_m_green = c2_fm_red = math.ceil(TOTAL * s2 * (1 - s1) * 0.5)\n",
    "c1_fm_red = c2_m_green = math.ceil(TOTAL * s1 * (1 - s2) * 0.5)\n",
    "c1_fm_green = c2_m_red = math.ceil(TOTAL * (1 - s1) * (1 - s2) * 0.5)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "c7218e6f-d24a-409f-a2ba-c5810d904891",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "13500\n",
      "1500\n",
      "4500\n",
      "500\n",
      "500\n",
      "4500\n",
      "1500\n",
      "13500\n"
     ]
    }
   ],
   "source": [
    "print(c1_m_red,\n",
    "      c1_m_green,\n",
    "      c1_fm_red,\n",
    "      c1_fm_green,\n",
    "      c2_m_red,\n",
    "      c2_m_green,\n",
    "      c2_fm_red,\n",
    "      c2_fm_green,\n",
    "      sep='\\n')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "32b702ba-bf2e-4ae8-a264-547c22b57eb5",
   "metadata": {},
   "outputs": [],
   "source": [
    "def color_grayscale_arr(arr, red=True):\n",
    "  \"\"\"Converts grayscale image to either red or green\"\"\"\n",
    "  assert arr.ndim == 2\n",
    "  dtype = arr.dtype\n",
    "  h, w = arr.shape\n",
    "  arr = np.reshape(arr, [1, h, w])\n",
    "  if red:\n",
    "    arr = np.concatenate([arr,\n",
    "                          np.zeros((2, h, w), dtype=dtype)], axis=0)\n",
    "  else:\n",
    "    arr = np.concatenate([np.zeros((1, h, w), dtype=dtype),\n",
    "                          arr,\n",
    "                          np.zeros((1, h, w), dtype=dtype)], axis=0)\n",
    "  return arr"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "843e28ce-0aee-4766-a15c-979130130a6f",
   "metadata": {},
   "outputs": [],
   "source": [
    "def keep_only_lbls(dataset, lbls):\n",
    "  lbls = {lbl: i for i, lbl in enumerate(lbls)}\n",
    "  final_X, final_Y = [], []\n",
    "  for x, y in dataset:\n",
    "    if y in lbls:\n",
    "      final_X.append(x)\n",
    "      final_Y.append(lbls[y])\n",
    "  X = torch.stack(final_X)\n",
    "  Y = torch.tensor(final_Y).float().view(-1,1)\n",
    "  return X, Y"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "d455f42a-2d21-461e-92ce-4abc55a1f019",
   "metadata": {},
   "outputs": [],
   "source": [
    "mnist_transform = torchvision.transforms.Compose([\n",
    "          torchvision.transforms.ToTensor(), \n",
    "          torchvision.transforms.Normalize(mean=0.485,\n",
    "                                           std=0.229),\n",
    "        ])\n",
    "mnist_train = torchvision.datasets.MNIST('./data/mnist/', train=True, download=True, transform=mnist_transform)\n",
    "X_mnist, _ = keep_only_lbls(mnist_train, lbls=[0, 1, 2, 3])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "28f9b7d5-e317-4e47-8cc1-e970ba89940b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([24754, 1, 28, 28])"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "X_mnist.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "496f1f70-63b0-4e61-98c0-ca00e91d9c6f",
   "metadata": {},
   "outputs": [],
   "source": [
    "fmnist_train = torchvision.datasets.FashionMNIST('./data/fashion_mnist/', train=True, download=True, transform=mnist_transform)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "db190aa4-6442-44cb-b245-9b9f22e3bfd0",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([24000, 1, 28, 28])"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "X_fmnist, _ = keep_only_lbls(fmnist_train, lbls=[0, 3, 4, 6])\n",
    "X_fmnist.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "6ba37fb5-b9f7-4775-b683-20940db1cd41",
   "metadata": {},
   "outputs": [],
   "source": [
    "red_mnist, red_fmnist, green_mnist, green_fmnist = [], [], [], []\n",
    "\n",
    "for idx, img in enumerate(X_mnist):\n",
    "    img = color_grayscale_arr(img.squeeze().numpy(), red=True)\n",
    "    img = np.pad(img, ((0, 0), (2, 2), (2, 2)), constant_values=0)\n",
    "    red_mnist.append(torch.tensor(img))\n",
    "red_mnist = torch.stack(red_mnist)\n",
    "for idx, img in enumerate(X_mnist):\n",
    "    img = color_grayscale_arr(img.squeeze().numpy(), red=False)\n",
    "    img = np.pad(img, ((0, 0), (2, 2), (2, 2)), constant_values=0)\n",
    "    green_mnist.append(torch.tensor(img))\n",
    "green_mnist = torch.stack(green_mnist)\n",
    "\n",
    "for idx, img in enumerate(X_fmnist):\n",
    "    img = color_grayscale_arr(img.squeeze().numpy(), red=True)\n",
    "    img = np.pad(img, ((0, 0), (2, 2), (2, 2)), constant_values=0)\n",
    "    red_fmnist.append(torch.tensor(img))\n",
    "red_fmnist = torch.stack(red_fmnist)\n",
    "for idx, img in enumerate(X_fmnist):\n",
    "    img = color_grayscale_arr(img.squeeze().numpy(), red=False)\n",
    "    img = np.pad(img, ((0, 0), (2, 2), (2, 2)), constant_values=0)\n",
    "    green_fmnist.append(torch.tensor(img))\n",
    "green_fmnist = torch.stack(green_fmnist)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "dc1a1b55-f9f9-4572-b99c-0acb7155eb21",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(torch.Size([24754, 3, 32, 32]),\n",
       " torch.Size([24754, 3, 32, 32]),\n",
       " torch.Size([24000, 3, 32, 32]),\n",
       " torch.Size([24000, 3, 32, 32]))"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "red_mnist.shape, green_mnist.shape, red_fmnist.shape, green_fmnist.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "ed7b717d-ef0f-4f53-8adc-6f1a8adcb9e0",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(torch.Size([13500, 3, 32, 32]), torch.Size([500, 3, 32, 32]))"
      ]
     },
     "execution_count": 14,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "idx = torch.randperm(red_mnist.shape[0])\n",
    "red_mnist_0_idx = idx[:13500]\n",
    "red_mnist_4_idx = idx[13500:13500+500]\n",
    "red_mnist_0 = red_mnist[red_mnist_0_idx]\n",
    "red_mnist_4 = red_mnist[red_mnist_4_idx]\n",
    "red_mnist_0.shape, red_mnist_4.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "50a2ffa7-a41b-4cbe-833f-d62175bb8b94",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(torch.Size([1500, 3, 32, 32]), torch.Size([4500, 3, 32, 32]))"
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "idx = torch.randperm(green_mnist.shape[0])\n",
    "green_mnist_1_idx = idx[:1500]\n",
    "green_mnist_5_idx = idx[1500:1500+4500]\n",
    "green_mnist_1 = green_mnist[green_mnist_1_idx]\n",
    "green_mnist_5 = green_mnist[green_mnist_5_idx]\n",
    "green_mnist_1.shape, green_mnist_5.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "e369e2c2-5d8c-4ed2-a9d9-3abeef593dd0",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(torch.Size([4500, 3, 32, 32]), torch.Size([1500, 3, 32, 32]))"
      ]
     },
     "execution_count": 16,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "idx = torch.randperm(red_fmnist.shape[0])\n",
    "red_fmnist_2_idx = idx[:4500]\n",
    "red_fmnist_6_idx = idx[4500:4500+1500]\n",
    "red_fmnist_2 = red_fmnist[red_fmnist_2_idx]\n",
    "red_fmnist_6 = red_fmnist[red_fmnist_6_idx]\n",
    "red_fmnist_2.shape, red_fmnist_6.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "e2fbd487-ad89-4132-89a0-6e1fc7c8ea85",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(torch.Size([500, 3, 32, 32]), torch.Size([13500, 3, 32, 32]))"
      ]
     },
     "execution_count": 17,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "idx = torch.randperm(green_fmnist.shape[0])\n",
    "green_fmnist_3_idx = idx[:500]\n",
    "green_fmnist_7_idx = idx[500:500+13500]\n",
    "green_fmnist_3 = green_fmnist[green_fmnist_3_idx]\n",
    "green_fmnist_7 = green_fmnist[green_fmnist_7_idx]\n",
    "green_fmnist_3.shape, green_fmnist_7.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "919e9b30-2866-47e4-bf95-5446b55822e8",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "(torch.Size([20000, 3, 32, 32]), torch.Size([20000, 3, 32, 32]))"
      ]
     },
     "execution_count": 18,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "cifar_transform = torchvision.transforms.Compose([\n",
    "          torchvision.transforms.ToTensor(), \n",
    "          torchvision.transforms.Normalize(mean=[0.485, 0.456, 0.406],\n",
    "                                           std=[0.229, 0.224, 0.225]),\n",
    "        ])\n",
    "cifar_train = torchvision.datasets.CIFAR10('./data/cifar10/', train=True, download=True, transform=cifar_transform)\n",
    "X_c_cars, _ = keep_only_lbls(cifar_train, lbls=[0, 1, 8, 9])\n",
    "X_c_animals, _ = keep_only_lbls(cifar_train, lbls=[3, 4, 5, 7])\n",
    "X_c_cars.shape, X_c_animals.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "f84cb04d-1a47-4f84-aadd-d0e3a03d97e4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(torch.Size([13500, 3, 32, 32]),\n",
       " torch.Size([1500, 3, 32, 32]),\n",
       " torch.Size([4500, 3, 32, 32]),\n",
       " torch.Size([500, 3, 32, 32]))"
      ]
     },
     "execution_count": 19,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "X_c_cars_0 = X_c_cars[:13500]\n",
    "X_c_cars_1 = X_c_cars[13500:13500+1500]\n",
    "X_c_cars_2 = X_c_cars[13500+1500:13500+1500+4500]\n",
    "X_c_cars_3 = X_c_cars[13500+1500+4500:]\n",
    "X_c_cars_0.shape, X_c_cars_1.shape, X_c_cars_2.shape, X_c_cars_3.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "1b77e629-33e4-4457-b814-6fe5e20e1b07",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(torch.Size([13500, 3, 32, 32]),\n",
       " torch.Size([1500, 3, 32, 32]),\n",
       " torch.Size([4500, 3, 32, 32]),\n",
       " torch.Size([500, 3, 32, 32]))"
      ]
     },
     "execution_count": 20,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "X_c_animals_7 = X_c_animals[:13500]\n",
    "X_c_animals_6 = X_c_animals[13500:13500+1500]\n",
    "X_c_animals_5 = X_c_animals[13500+1500:13500+1500+4500]\n",
    "X_c_animals_4 = X_c_animals[13500+1500+4500:]\n",
    "X_c_animals_7.shape, X_c_animals_6.shape, X_c_animals_5.shape, X_c_animals_4.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "6ca938b3-b750-4b78-8af9-5383d5fa1388",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([13500, 3, 64, 32])"
      ]
     },
     "execution_count": 21,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "group_0_X = torch.cat((X_c_cars_0, red_mnist_0), dim=2)\n",
    "group_0_Y = torch.zeros(len(group_0_X))\n",
    "group_0_G = torch.tensor([0] * len(group_0_X))\n",
    "group_0_X.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "ff7a3077-0de7-4377-8acd-913935fce030",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([1500, 3, 64, 32])"
      ]
     },
     "execution_count": 22,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "group_1_X = torch.cat((X_c_cars_1, green_mnist_1), dim=2)\n",
    "group_1_Y = torch.zeros(len(group_1_X))\n",
    "group_1_G = torch.tensor([1] * len(group_1_X))\n",
    "group_1_X.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "4921b503-0c25-42de-aa6a-49b4d849b479",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([4500, 3, 64, 32])"
      ]
     },
     "execution_count": 23,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "group_2_X = torch.cat((X_c_cars_2, red_fmnist_2), dim=2)\n",
    "group_2_Y = torch.zeros(len(group_2_X))\n",
    "group_2_G = torch.tensor([2] * len(group_2_X))\n",
    "group_2_X.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "6b5c4019-e443-4a68-8cff-04630c469e0b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([500, 3, 64, 32])"
      ]
     },
     "execution_count": 24,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "group_3_X = torch.cat((X_c_cars_3, green_fmnist_3), dim=2)\n",
    "group_3_Y = torch.zeros(len(group_3_X))\n",
    "group_3_G = torch.tensor([3] * len(group_3_X))\n",
    "group_3_X.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "3a240a8a-c286-422e-8a93-8ea713dfafbd",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([500, 3, 64, 32])"
      ]
     },
     "execution_count": 25,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "group_4_X = torch.cat((X_c_animals_4, red_mnist_4), dim=2)\n",
    "group_4_Y = torch.ones(len(group_4_X))\n",
    "group_4_G = torch.tensor([4] * len(group_4_X))\n",
    "group_4_X.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "fca75c98-8591-4560-87e4-a7c5d695bbd7",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([4500, 3, 64, 32])"
      ]
     },
     "execution_count": 26,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "group_5_X = torch.cat((X_c_animals_5, green_mnist_5), dim=2)\n",
    "group_5_Y = torch.ones(len(group_5_X))\n",
    "group_5_G = torch.tensor([5] * len(group_5_X))\n",
    "group_5_X.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "d48bd971-115f-4bc9-a104-fc4b685e1d83",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([1500, 3, 64, 32])"
      ]
     },
     "execution_count": 27,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "group_6_X = torch.cat((X_c_animals_6, red_fmnist_6), dim=2)\n",
    "group_6_Y = torch.ones(len(group_6_X))\n",
    "group_6_G = torch.tensor([6] * len(group_6_X))\n",
    "group_6_X.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "0f45a391-483d-44a7-8d90-c5c4df3f33e3",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([13500, 3, 64, 32])"
      ]
     },
     "execution_count": 28,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "group_7_X = torch.cat((X_c_animals_7, green_fmnist_7), dim=2)\n",
    "group_7_Y = torch.ones(len(group_7_X))\n",
    "group_7_G = torch.tensor([7] * len(group_7_X))\n",
    "group_7_X.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "22841724-0602-429c-aa19-d2bf51592d9b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(torch.Size([40000, 3, 64, 32]), torch.Size([40000]), torch.Size([40000]))"
      ]
     },
     "execution_count": 29,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "X = torch.cat((group_0_X, group_1_X, group_2_X, group_3_X, group_4_X, group_5_X, group_6_X, group_7_X))\n",
    "Y = torch.cat((group_0_Y, group_1_Y, group_2_Y, group_3_Y, group_4_Y, group_5_Y, group_6_Y, group_7_Y))\n",
    "G = torch.cat((group_0_G, group_1_G, group_2_G, group_3_G, group_4_G, group_5_G, group_6_G, group_7_G))\n",
    "X.shape, Y.shape, G.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "62700fbc-f634-4cfc-ba20-99acaba470b9",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "((tensor([0., 1.]), tensor([20000, 20000])),\n",
       " (tensor([0, 1, 2, 3, 4, 5, 6, 7]),\n",
       "  tensor([13500,  1500,  4500,   500,   500,  4500,  1500, 13500])))"
      ]
     },
     "execution_count": 30,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "Y.unique(return_counts=True), G.unique(return_counts=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "c01f3198-d6ce-4bd4-8cbf-e7e7ff649ceb",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.save(X, 'cifar_cmnist_fmnist_dominoe_X_07095.pt')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "f636d377-af1b-4b31-9efc-38a90a6035a4",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.save(Y, 'cifar_cmnist_fmnist_dominoe_Y_07095.pt')\n",
    "torch.save(G, 'cifar_cmnist_fmnist_dominoe_G_07095.pt')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 66,
   "id": "db12e290-370f-43d6-952f-9c24fe64a3d5",
   "metadata": {},
   "outputs": [],
   "source": [
    "X = torch.load('cifar_cmnist_fmnist_dominoe_X_07095.pt')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 67,
   "id": "0f6aad9c-8314-4186-8092-17cfa78b372c",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Clipping input data to the valid range for imshow with RGB data ([0..1] for floats or [0..255] for integers).\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Clipping input data to the valid range for imshow with RGB data ([0..1] for floats or [0..255] for integers).\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Clipping input data to the valid range for imshow with RGB data ([0..1] for floats or [0..255] for integers).\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Clipping input data to the valid range for imshow with RGB data ([0..1] for floats or [0..255] for integers).\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAMwAAAGFCAYAAACxAhziAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/H5lhTAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAbNElEQVR4nO3dWaxd91XH8bXPPNzrO3oe48xNmqQDLWUIHaSmVKh9AKS+IhASlXhBSEiAxEtfixCvCPGGKpUHJNRCpdKilhJom8RJnKm1E8exHdvXvr7zGffePCBFpP/fSvdSncS+/n4el//aZ59h3a29vP5rZ2VZlgagktr7fQLA7YSEAQJIGCCAhAECSBgggIQBAkgYIICEAQIaVRd+9Q8flfE9va6Mt+rpobd3BnJtr53J+GKvJePtel3GlbLQ/y+bm44X7p+Q9BzHhV45HI5kfDrV6/MiPVDPeYvNpn7RZk2/n0ZDf8Xj8bjSeZiZZeK7NDMrxGdiZpbnuYxbmR6/luk3muf62LWajrfq6fufOueRZfoYv/3V52T8ba//c1cAeAsJAwSQMEAACQMEkDBAQOUq2dpQVxbaLV2d6babSWxmJo2ZmWXFRMaLUudzlqXVs9FIV6YGAx33/lTUnIpds5Gee7/ZkWs7NV3d2x7r9zkU5z6cOmtzXWrrdJxKY9OpNLbTc9/c2pRrx1NdbXKKapZ71TPxfdZMH6TrVAkbNec3kae/w9pU/zZLvxT6c3GFAQJIGCCAhAECSBgggIQBAipXyaZOz8/Kjq7alK208tPKdEWk7VTDtqdORaQtYi1dDRoNdYUndxu7nApPMUxi/U7aj2Vm1nTOpeH0ZM2LnrkycyqKNX3eE6dktTnS6xvio/WqW3Xnu5+K3jCzd+hJE0Wr0vlNTJ1ev6bzk83E+6ll+rt3imeVcIUBAkgYIICEAQJIGCCg8k1/Y1bcaZtZOdU3vtc209aO+jS9cTYzm23qm8o93Rl9LrX0RrHM9XnkTuuFbjwxy2r6XBrN9KMaO0cZOzfahVNo6In7+/6sPo+uWmxmzYb+fgbb+nMZb6XxgV5qI6c1ZjzR78fbnNeopZ9h5rS6OPv+rO4UTjJRgCgz5+etqg8VcYUBAkgYIICEAQJIGCCAhAECKlfJ2ssLMj6X6dJKfZhWUAZruvVix9nkNRxt6ZOZS6tnMx2nHUW/pE1yPfJp5FS+skY/PXZdj5gaF077ilPJm07S1xxPdZVsa0NXrPpdfWyx783MzDI5rsgZyeRsWpuUzigkUVE00+Oa6k6VbOpUXwfebCvxmKPMafWp/KMXuMIAASQMEEDCAAEkDBBAwgABlQsGb050dWKn3pPx5T1phaLT02OJRje2ZXzrxoaMj9fTcUC9oS4H9dq6x6rV0BWuhumKXVP0gZXeJqdSf6yls/krH6cVsRtOP5qJjWxmZt2urp41nbFRmdi0V4pRRWZmpVNtMncovDdmKT1+4Ww2azo9fS2n7DeZpFU1NXDdzGxnQi8Z8J4gYYAAEgYIIGGAABIGCKhcJas5VYsVp5qzLlqy2l1dPZnbOyvje/boHZfT7bRSNFrXfWfjbR3vzByV8VG2V8ZXm4eSWNY9ItfOtXUlq1+7pNcXaV9bbbgq127eWJHxjcmOjLd3dNWvXqYVpLpXgZJRs6k5Q8qdaltDPCqvnHo7YvUx6s5uSbVptznjVGWd+fRVcIUBAkgYIICEAQJIGCCg8k3/IwfukfHXV/VN6MokveGqlfqGepzr02i29ROx8jy9a9twRgQNct0Cc2NyQMZHNb1RblqmG8jKIo2ZmY2GuljRrn1Cxg8vped4dL++M52/X7cR1Se6uDG5cU7GB1dfTGL51nm5Nh9c0XFnY5k5o61MjD2qO8WkrvPL3HSLGGkxoFH/BYYoO7jCAAEkDBBAwgABJAwQQMIAAZWrZHcf+5iMX9ZFG7s6+nAS6zT1ZrNxQ1czVsaXZfzaetpicvHKGbm2WdPtG/v36o1lM11dmWs10/aVMrsq115887qM//ScrvBcOvqZJPb84km5dn8/bdExM7vvhP4qT979IRmfP/4bSWxj7VW5du3C0zK+ffF5GZ+snpVxK9aSUKPutOM0dXWz5VTg2p30+6xN9Oc9GurWpSq4wgABJAwQQMIAASQMEEDCAAGVq2RnV3TV4ps/viHjV8bXklhnXld4es4g8azUG7Smg7SHafXCN+XaA0fnZLwY75HxlTXdk9XupesbzmPyrp/TVaJ5Z3D7wX5aPew2dMXq2g3dj/cfl/Vn++zCkozfeyj9XE7u+yW5dvmRh2R8/sQbMr52/hkZX73w4yQ22NTvczzSI7aaTptaz9LfRM95FGSeORPaK+AKAwSQMEAACQMEkDBAAAkDBFSukv3wtO7rOnP6WRkfbn4tic0f+qxcuzGjd3NOTO9c7PTSit24/ohcm3f1zspp46KMtzNdnRntpP1hG1t6ANG2eASfmdl9j/2yXr+e7lotx7p6tK+rv4fhWlqBMjN748JBGb94/uEk9vx+PXrqyD69s/Te/cdl/OAH7pbxA/d/OoltX35Frj3/4ndkfLj2nIxno7UkVneGwnuPCayCKwwQQMIAASQMEEDCAAEkDBBQuUp2dSXtDTMzaxe6anPPvWm/zrmL35JrV2/onXGl8zjAVjet/DRnHtDHXtNVombjPhlv2Esynk/TOV4bV07LtUv7dWWuv6D7wNavrSWxq5u6Wrda17sF88G6jGer+hy3V/8riU02dC/ZlcuPyvi5pRMyflyPdrOToq9v34Ffk2tni8MyXi/073B6NX2f9Su60pYP39QnWAFXGCCAhAECSBgggIQBAirf9A8HemPVXffqm+rPP5HezH332/8i1/77D/W4nnr3fhkfF+lsp8lQjzya7OibxM0NveFqYflBGe/3RPyALjSM2vrpYa++8pqM13bSTXi1uvNkrrEeGzUu9N++wnlK2OJs2r5Tr52Sa3eu6Q1hK+t6Y9nm3sdl/PzGchJbntXntzyrv5/FGWdYfD8tMl2dvCzX9of6SXBVcIUBAkgYIICEAQJIGCCAhAECKlfJrNTPgT9xLK18mJktzKfjfQ7u12OT2vY/Ml5r6Y1l1kg3kI3G+rFy4w09Bsq29KMGy/VZGV+bT99nf04P+m7152W8ML1prZikm6KKga4c5iPdilSWur1oPNWjnZb2pFWoSaHHXc109LEHmz+U8eHrumK51fn9JNYodZWsPdYbE/cU+nu78PL3ktj6Nb32RDuT8Sq4wgABJAwQQMIAASQMEEDCAAGVq2S9pn6sXquucy6rpRWXekNvCKuVum+qcHp+aq20n6jd0D1GhTNqZ5yvyXi+rePZKN2gtbOqN3ltru6X8dl5Z2PZ/CeTWHPxo3JtY1sPOh+u6n6v0VTH8820lyxrpQO9zcwy5zvui0qomdnouq7kFZM0vnx0Xq493tGfbX2i49NBWpnLnbXD6rXhBFcYIICEAQJIGCCAhAECSBggoHK94OGH9Fii0y/qsUTnX389ib1xPo2ZmZW57idqdvRuzvFY9IeN012YZmbN9qKMdzr6kX3DQn8kUzHGKBvoPq3JUPev7dzQVb96Nz3H+f36vBcWPyLj3eMfk/H2/jMyvraRDi8vN/Rg8H6mP9t2Rz8Sr1HX32ejkVZDO3P68Xkbb+rP8MJZZ/fnpXRnbVO8npnZoKErvlVwhQECSBgggIQBAkgYIKDyTf9DD+qb/m5PP6v+2aefSmIvv6Q3BU3E06PMzCbr6THMzFoz6cijMtM3mpOB3lhWOG0T7Y6ef1xrph9VUTg3/aOBjJdjPRe5Pkrj69t6/u/GFR3vzOsCyfz+EzI+cyR9Alk2fkOu9dpuJlsv6PWjCzLeb6dPlGuW+rN67SW9Oe3qii4cjUUBptvSv82tOjf9wHuChAECSBgggIQBAkgYIKBylazV1iN47rlLP6t9cS5tPdl/WFegTp/Wz14/c1Zvlrq2LlpMsnvl2kZHj3YqnY1lww1d4cka6fp6e59c22zrjXJTp6o2nojKV+60neR65NHWjq42Da7oKmF3Id38NbM8L9fO7P2sjLcO6aHj2fpPZHy5m/59LgZ6FJJqrTIza4nNg2Zmjdl0cP3U2VQ4raVPk6uKKwwQQMIAASQMEEDCAAEkDBBQuUrWaOqNPoXeo2ML82mF4lc//nG59v57dYXr9Gm9Oe2pZ36UxF77qe532ljVm6Lqsx+W8VpLV9UmebqhaTw4L9e263rzV6OrxxJlllbEJmM9/L21pJ9f31nU8dGaftTi5tW0Gri5qh9v2L6gK4ezR47K+NG9J2T8I1f/KYld3fNJuXZ0+E9lPMs7Mt6aTYfIb17Xj+zb16JKBrwnSBgggIQBAkgYIICEAQIqV8nW1nXFodu7Sx+4mT5WL/ceH7ege34++lFdyTp8KK3OPPecrpKdfkHHL1x6UsaHQz0wvNH/QBLLGroaNi10hWuyrfvA6llagWyY7g2bvvkNGbehrpLNnXxCxmv1tNdv8/p1fehSf2/DV/UIp+ZPXpXxi0X6WL3Lw0fk2tw+oeN9vVN2/970b3820N/9woIeul4FVxgggIQBAkgYIICEAQJIGCCg+o7LwTkZ39jRlZX+Ujr3qtWZl2uLiZ7XNdvX/Wvdu9L5VgtLc3LtwWN6XtepZ56W8bOv6MrPjevfSWJF36kQzj4k45brOVmTXOyK3NYzvx64R1fmZrp6h+KZl/9Oxpce/XISG27oytxkuCnjNtHfT39W7/K83k/7vU6dS4e8m5l1l/Uw8j/6kv4Mi1H6m/j6dT1/rNSbhyvhCgMEkDBAAAkDBJAwQEDlm/4jh/TGqisr6fPRzcy2VtPB4/nsCbm229LPtc8yPWqotPRmbk5sWDMze/jhD8r43mX9mscO/lTGX3jhVBJ7/Q1dINh0nlPf7D0q4zadpOfhPL/+N7/wBRkfOQPQ7T+/K8Nnz3w7ic3O6s/q2o4uKGQTfVM9X9cb68at9Oc2FW1BZmadrt6ZuLGmb/qfejFtmRmXemPi5nbatlUVVxgggIQBAkgYIICEAQJIGCCgcpWs1tQDtg/sPSbjW5tpy8zVbT3yaDvXz4HvtPUYH6unlZJaQ1dPejX9N+HYET2uZ3lOj0I6eCgdPH7K2bT28iunZXzl+vdlfLKdVoruefwzcu3iXr3BzXsc4H0Ppo83NDN7+V/TcywXdEtPp6nbjhqZfs2jly/K+J6NtDXmtQ/pYfZFX1cx//HbenPejmjrmW/qEVPNGWc2WAVcYYAAEgYIIGGAABIGCCBhgIDKVbJmTVfJRqYrXHNidFK/rV/uwjU97HpnrDcudWbSjVv1bF6uzeu6H63l/K1oL+jdRR/spxvi9h7UFbV9B/SjCZ91Nq29ejYdS7S0qCtTM710PJKZWdnW/VF7lw/JeLv2X0nsxtYbcm2jkW7OMjOzrq7YTX/9r2T8no+n53LqnD7G+hu6wlUO9JilfjNPYgfbui+w4YzBqoIrDBBAwgABJAwQQMIAASQMEFC9StbS1aOak3PTPK1ytDu68nPioO4DW1lflfG17XQwetnU45RaDV2FsZp+zSzT76eWpe//wEH9mr2erigeXNbn8mT/B0ls5cqbcu0414O0h86g88EorR6ZmeXTdH1W1xWovKYrjdNcVzdf2tL9hY1JOtB99azuL7SR7lNb7qW7U83MHjyc/lYe2qe/47WVu/VrVsAVBgggYYAAEgYIIGGAABIGCKhcJdsZ6f6bXjPdRWdm1hCVlfFUz86qN3TeHljWp9fZSqs5l9bOybU7pis/nZ7ezZk1dK9W1kyrZLVC79ybm9PVwEce04+n27dvOYl9/wdp5czM7NRTT8n48ZO68nPmrO6n2hmlw74ze1GurTf0rk3L9a7VC6/peW1bG+mu1b7pxwEeWNB9bY/epYff338ijfVb+jGGh/fq76cKrjBAAAkDBJAwQAAJAwRUvum/du2ajM/M6JaMmf58Emu19Eak3BmzVJb6pnphJr0xz+RKs2tOe81kqG82y5becNVopzespdNGUzT0x1oz/X6OHUtbST4/o4spTz+vn0z25Pf+TcbffP2sjD9wIm3T6e/RbTf1TLfprKzrQeLDjv6e986kY6kO9y7JtSf26QLRsTldaKjl6W9i6BQlmnVu+oH3BAkDBJAwQAAJAwSQMEBA5SrZcEe3xgyGupoxmKQbfZZn0xYQM7NeW7ejTJxH9hV5ei57ZvQj+5qipcXMbGNbj3Ba29FVpaFosWm2nUHa3oPg67qWVxc1vsWltCpnZvb44/p9XnY2nK2t6gHjNk4/w8IZazUe6s1c46muqk2cjWW9dlppne/pz6rb11XCsbOZrVakjw+sm7N5LtfVvSq4wgABJAwQQMIAASQMEEDCAAGVq2SZ85SzsaiGmZndWLmaHmOsD1Iu6OHd3a6uoBSiJ6tW08fuOE1mDSfecx5Dd3k7fQxdkev33up51TM9fsmydBxQ2UirPmZmtVKf+JEjJ2T8wAG9UW48SCtiO1srcu3Ott6ENxnpKlmW6++iIXrvanX9E2y2dCWr3nAqp7X0c/F+s27jYQVcYYAAEgYIIGGAABIGCCBhgIDqY5YGupes6zwqrhBDs2+s6irMZKz70eYXFmW810t39DWdUU2Nuo43vUf51fUAazUKamNnXa4thukj+P7vRfXw8lE97bHLC6dnKtMlnqLUVTVPq5P279Vb+r13Z/V3Px3p3baTsd7Nmol+L5dTycqcf8jE3/6ycD4r059tFVxhgAASBgggYYAAEgYIqHzTP3aeZGWlvvHrdtMbyDzXN4PXnWLAwCk0LCymN8lLi0tyrbeBzL39dHpp6kVaxJgr9c3jeKI/k9H0iozXxFimSUOPAhqMvL9xzmgnpweomKr2Iv1ZNVtO4aTVl/GuU4CYTtLfUO60FxWFjudO200mvtHCmX3txavgCgMEkDBAAAkDBJAwQAAJAwRUrpI5c8FtONJtLWr3TqfjVKxautq0tbUl45NJWrGaTnUFbmlJV89abT2ouu600nR6aRVm7LVpOOOU2qJdyMysLNIq4aimK02drlM904e28cTZiCZabEqn7aYs9c+kdKphmdjMZWbWFO+pVuhj57n+rWRTXT2r5aKKm+nKbuHEq+AKAwSQMEAACQMEkDBAAAkDBGSlV+r42YVOBQXYLaqkAlcYIICEAQJIGCCAhAECSBgggIQBAkgYIICEAQJIGCCAhAECSBgggIQBAkgYIICEAQJIGCCAhAECKo9ZuuP9s4h98V18PW+/XuxBYzfnNfEWrjBAAAkDBJAwQAAJAwSQMEAAVbKf9W5Wod4Pp534/SLmvffPOfFvxU/ndscVBgggYYAAEgYIIGGAABIGCKBKVtWGiO1x1l504odv0rlEXnPWib8iYvppeGafdeJUyQC8ExIGCCBhgAASBgggYYAAqmRVqU/K67065MRfd+LH46eTeNKJf8KJexUx5VPBc9nFuMIAASQMEEDCAAEkDBBw5970fz24fihiveAxFgJr9weP/WknftWJN0Vs4Ky9O3guuxhXGCCAhAECSBgggIQBAkgYIODOrZL9TnC9qiCpTWXeWjOzFSeuvoXLztozTvySE/c2kJ0XsWPO2o4TvwNxhQECSBgggIQBAkgYIICEAQLu3CpZlBqR9Iyz9oYTP+vEVY+ZV7HyNop5I5+8TWtqdJLXv+Yd4ytO/C+d+C7AFQYIIGGAABIGCCBhgAASBgigSvaL6DrxqRP3dlwuiphXaVM7Jd/p2F5fW2RHp9dL9hdOnCoZADMSBgghYYAAEgYIIGGAgN1fJfuTm3ScvxaxP3DWbjpxr5dMDQb3Km3esb1qmDdI/Bsi5j2az9sp6sV3Ma4wQAAJAwSQMEAACQMEZGVZes/RevvCLHu3z+Xd4Y0r8lpPHnDi6u17n9yOE/+mE1ejkLwB4KcCxzAze8KJq/fjbUJbd+KqcGBm9ltO/BZXJRW4wgABJAwQQMIAASQMEEDCAAG7vzXG2yj1307cq5Ipp534w058nxNXj9VTw8LN/E1rXstMxPHgem/8kvc4xN8NHv8WxBUGCCBhgAASBgggYYAAEgYI2P1VMo/atBV10YmfdOJelUxtLPPGKXkjj7zqWcSfB9d7lbwv/6IncuviCgMEkDBAAAkDBJAwQAAJAwTsniqZN07Jq4Z5Y4kix3/MWetVj7zXVCOVvG9m6MQPOfEIrzfMs3oTXvM2wxUGCCBhgAASBgggYYCA3XPT/1UnvuHEvdnFkeP/rbPWm1EcfU3Fez/3O/HvOfEvidifBc/F27TmveYuwBUGCCBhgAASBgggYYAAEgYI2D1VMo83dNwbkeT5mogtOGu9Fpib0UritcZ4G8i8938znHLi3vs/LGLeJrxbFFcYIICEAQJIGCCAhAECSBggYPdXydSgb7ObM3jbG1zujUjyKlaq2uZVmrwNcXuduEed4zPO2g85ce8xgd7n8vci9jln7S2KKwwQQMIAASQMEEDCAAEkDBCw+6tkXrUpOpZIDQx/zFnrDQz3KlyqSuZV96K9ZF4lS8W9/rpolew7Tvz3nPhthCsMEEDCAAEkDBBAwgABJAwQsPurZN4sMK/fyaNmcHmP4Iuey6KIqaqcmV/1y5y4Vw1UFbtLzlrPHifuneOPgse/BXGFAQJIGCCAhAECSBggYPff9HtjfLzh3R61+cu76fc2inmfduRb8FpgyuCx1QYyr+3G432G3nGOBY9/C+IKAwSQMEAACQMEkDBAAAkDBNyeVbLIRiSvfUMNxn4nqt3Dq1h5rSHe+CXVMuOtjfI2eanB6J8KHttb/0TwOLcRrjBAAAkDBJAwQAAJAwSQMEDA7Vkl+4eKMTO/ovbF4GuqapPa+GWmN5u9E7XeG8nk8TaQ7Xfi6ty96p7nNhskfjNwhQECSBgggIQBAkgYIICEAQJuzypZhFc98+LezkVvF6XiPT7Pq56pYeQ3q5fMowamR3dc3oG4wgABJAwQQMIAASQMEEDCAAG7v0p2s0Q+qZ4TV7sczfS8Lu/1vIHmUaqXzHvUIN7CFQYIIGGAABIGCCBhgABu+n+W1xrz+ZtwbO+GXcW9EU7RjWUeNUjde028hSsMEEDCAAEkDBBAwgABJAwQcOdWyX7FiXsbxW7Ghq5LTvxuEfOqYSs34Ty8c1Eb2fA2XGGAABIGCCBhgAASBgggYYCAO7dK9jdO3BswripZUV9w4uoRd95mM+/8ov5YxLwqHt7CFQYIIGGAABIGCCBhgAASBgi4c6tkH3u/T+D/UY/Vu/Ken4XvK+/3Cdw6uMIAASQMEEDCAAEkDBCQlWXpDRZ6+8LMexA8sDtUSQWuMEAACQMEkDBAAAkDBJAwQAAJAwSQMEAACQMEkDBAAAkDBJAwQEDlDWQVW86AXY0rDBBAwgABJAwQQMIAASQMEEDCAAEkDBBAwgABJAwQ8L9BYn95aAomwwAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Clipping input data to the valid range for imshow with RGB data ([0..1] for floats or [0..255] for integers).\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Clipping input data to the valid range for imshow with RGB data ([0..1] for floats or [0..255] for integers).\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Clipping input data to the valid range for imshow with RGB data ([0..1] for floats or [0..255] for integers).\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Clipping input data to the valid range for imshow with RGB data ([0..1] for floats or [0..255] for integers).\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "for i in [0, 13500, 15000, 19500, 20000, 20500, 25000, 26500]:\n",
    "    X[i][:, :32, :] = (X[i][:, :32, :]-X[i][:, :32, :].min())/(X[i][:, :32, :].max()-X[i][:, :32, :].min())\n",
    "    plt.imshow(X[i].permute(1,2,0), cmap='gray')\n",
    "    plt.axis('off')\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "1d1b05cb-98ff-41c1-b42f-a9a5f5ca85ff",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(torch.Size([40000, 3, 64, 32]), torch.Size([40000]))"
      ]
     },
     "execution_count": 34,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "X.shape, Y.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "3d7b97f1-e45d-45be-84c0-89e54b22727c",
   "metadata": {},
   "outputs": [],
   "source": [
    "dataset = TensorDataset(X, Y.long(), G.long())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "id": "f3089dca-aecf-4cd0-898f-88705bc3e1ae",
   "metadata": {},
   "outputs": [],
   "source": [
    "train_dataset, lastlayer_dataset, val_dataset, test_dataset = random_split(dataset, [0.25, 0.25, 0.25, 0.25], generator=torch.Generator().manual_seed(42))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "id": "4863b952-2e6f-455f-a46b-0a3fb7b97d7c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(10000, 10000, 10000, 10000)"
      ]
     },
     "execution_count": 37,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(train_dataset), len(lastlayer_dataset), len(val_dataset), len(test_dataset)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "id": "ba9c4d6d-73ac-40fb-bc02-b160c3680e17",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pickle\n",
    "with open('train_indices_07095.pkl', 'wb') as f:\n",
    "    pickle.dump(train_dataset.indices, f)\n",
    "with open('lastlayer_indices_07095.pkl', 'wb') as f:\n",
    "    pickle.dump(lastlayer_dataset.indices, f)\n",
    "with open('val_indices_07095.pkl', 'wb') as f:\n",
    "    pickle.dump(val_dataset.indices, f)\n",
    "with open('test_indices_07095.pkl', 'wb') as f:\n",
    "    pickle.dump(test_dataset.indices, f)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3f3b9384-a21b-4c43-bd32-044cf379e93a",
   "metadata": {},
   "source": [
    "# Loading dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 115,
   "id": "d464ca87-19cd-42bf-ad1d-5bdcdd84ea5e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([10000, 3, 64, 32]) torch.Size([10000]) torch.Size([10000])\n"
     ]
    }
   ],
   "source": [
    "import pickle\n",
    "with open('train_indices_07095.pkl', 'rb') as f:\n",
    "    tr = pickle.load(f)\n",
    "with open('lastlayer_indices_07095.pkl', 'rb') as f:\n",
    "    ll = pickle.load(f)\n",
    "with open('val_indices_07095.pkl', 'rb') as f:\n",
    "    val = pickle.load(f)\n",
    "with open('test_indices_07095.pkl', 'rb') as f:\n",
    "    te = pickle.load(f)\n",
    "X = torch.load('cifar_cmnist_fmnist_dominoe_X_07095.pt')\n",
    "Y = torch.load('cifar_cmnist_fmnist_dominoe_Y_07095.pt')\n",
    "G = torch.load('cifar_cmnist_fmnist_dominoe_G_07095.pt')\n",
    "X_tr, Y_tr, G_tr = X[tr], Y[tr], G[tr]\n",
    "X_ll, Y_ll, G_ll = X[ll], Y[ll], G[ll]\n",
    "X_val, Y_val, G_val = X[val], Y[val], G[val]\n",
    "X_te, Y_te, G_te = X[te], Y[te], G[te]\n",
    "train_dataset = TensorDataset(X_tr, Y_tr.long(), G_tr.long())\n",
    "lastlayer_dataset = TensorDataset(X_ll, Y_ll.long(), G_ll.long())\n",
    "val_dataset = TensorDataset(X_val, Y_val.long(), G_val.long())\n",
    "test_dataset = TensorDataset(X_te, Y_te.long(), G_te.long())\n",
    "print(X_tr.shape, Y_tr.shape, G_tr.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 118,
   "id": "2ac51c51-cadd-4a4a-b61c-e6092ee0ce5e",
   "metadata": {},
   "outputs": [],
   "source": [
    "batch_size = 128\n",
    "train_loader = DataLoader(train_dataset, batch_size=batch_size, shuffle=True)\n",
    "lastlayer_loader = DataLoader(lastlayer_dataset, batch_size=batch_size, shuffle=False)\n",
    "val_loader = DataLoader(val_dataset, batch_size=batch_size, shuffle=False)\n",
    "test_loader = DataLoader(test_dataset, batch_size=batch_size, shuffle=False)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "550581cf-bb1c-43f3-803c-9e5a513dea6c",
   "metadata": {
    "jp-MarkdownHeadingCollapsed": true
   },
   "source": [
    "# ERM Training"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "a2fe811c-c1ca-4488-b0cf-f8703f3375e2",
   "metadata": {},
   "outputs": [],
   "source": [
    "device = torch.device(\"cuda:0\" if torch.cuda.is_available() else \"cpu\")\n",
    "\n",
    "model = models.resnet18(weights=ResNet18_Weights.DEFAULT)\n",
    "\n",
    "for param in model.parameters():\n",
    "    param.requires_grad = True\n",
    "\n",
    "num_ftrs = model.fc.in_features\n",
    "model.fc = nn.Linear(num_ftrs, 2)\n",
    "\n",
    "model.to(device);"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "23e74295-15d2-4e26-a751-3f4ea7460d68",
   "metadata": {},
   "outputs": [],
   "source": [
    "criterion = nn.CrossEntropyLoss()\n",
    "optimizer = optim.SGD(model.parameters(), lr=1e-3, momentum=0.9, weight_decay=1e-4)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "265ae6cf-1cd3-43c4-86fc-b167f1a371c9",
   "metadata": {},
   "outputs": [],
   "source": [
    "epochs = 20\n",
    "batch_size = 128\n",
    "train_loader = DataLoader(train_dataset, batch_size=batch_size, shuffle=True)\n",
    "val_loader = DataLoader(val_dataset, batch_size=batch_size, shuffle=False)\n",
    "test_loader = DataLoader(test_dataset, batch_size=batch_size, shuffle=False)\n",
    "lastlayer_loader = DataLoader(lastlayer_dataset, batch_size=batch_size, shuffle=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "baac0ac4-8b24-43a6-a2eb-1cb033f5e0b4",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "790b0746db7e43eeb0a0343e0ece0aa1",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/20 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "c8574186583e4569a4cda4fd79814522",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train loss = 0.2734653055667877\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "4abfb98326f74ace9c5ac2b3f67ee5d5",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 1, Train Acc: \n",
      "{0: 0.9970648664514236,\n",
      " 1: 0.8689839572192514,\n",
      " 2: 0.9768683274021353,\n",
      " 3: 0.7304964539007093,\n",
      " 4: 0.5128205128205128,\n",
      " 5: 0.9502183406113537,\n",
      " 6: 0.7576601671309192,\n",
      " 7: 0.9867986798679867}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "5187caabb6c44db1b3b6d786b1da2f7b",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 1, Val Acc: \n",
      "{0: 0.9955103262496259,\n",
      " 1: 0.8217821782178217,\n",
      " 2: 0.9498239436619719,\n",
      " 3: 0.5793650793650794,\n",
      " 4: 0.3793103448275862,\n",
      " 5: 0.9097345132743363,\n",
      " 6: 0.6683804627249358,\n",
      " 7: 0.9864824271553019}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "03d4b51bf2a04ccb951ea87eaa9b6d67",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train loss = 0.10355518758296967\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "e4a91dc74bdf4f2eb549944dd3256c29",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 2, Train Acc: \n",
      "{0: 0.9976518931611388,\n",
      " 1: 0.9090909090909091,\n",
      " 2: 0.9875444839857651,\n",
      " 3: 0.8085106382978723,\n",
      " 4: 0.7948717948717948,\n",
      " 5: 0.9912663755458515,\n",
      " 6: 0.9526462395543176,\n",
      " 7: 0.9978997899789979}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "67d0b60814bc479283664666beb348d5",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 2, Val Acc: \n",
      "{0: 0.9910206524992518,\n",
      " 1: 0.7821782178217822,\n",
      " 2: 0.9357394366197183,\n",
      " 3: 0.5555555555555556,\n",
      " 4: 0.5586206896551724,\n",
      " 5: 0.9530973451327434,\n",
      " 6: 0.794344473007712,\n",
      " 7: 0.9915890657855212}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "deebaa05ad4249d380400d14f04054cc",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train loss = 0.06594350188970566\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "c0314a124e064be1a0a2e72377e95f40",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 3, Train Acc: \n",
      "{0: 0.9994129732902847,\n",
      " 1: 0.9411764705882353,\n",
      " 2: 0.9973309608540926,\n",
      " 3: 0.8936170212765957,\n",
      " 4: 0.8974358974358975,\n",
      " 5: 0.9973799126637555,\n",
      " 6: 0.9749303621169917,\n",
      " 7: 0.9996999699969997}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "10900ee9953d4056bce771ff0f3976bf",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 3, Val Acc: \n",
      "{0: 0.9922178988326849,\n",
      " 1: 0.7920792079207921,\n",
      " 2: 0.9427816901408451,\n",
      " 3: 0.5396825396825397,\n",
      " 4: 0.5172413793103449,\n",
      " 5: 0.9575221238938053,\n",
      " 6: 0.8020565552699229,\n",
      " 7: 0.9954941423851006}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6c8c0b1e4b544030aa0ee19d3bfe2fd1",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train loss = 0.0395054966211319\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "40583807ce7a48ccbcca9342cc42a670",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 4, Train Acc: \n",
      "{0: 0.9997064866451424,\n",
      " 1: 0.9866310160427807,\n",
      " 2: 0.9991103202846975,\n",
      " 3: 0.9432624113475178,\n",
      " 4: 0.9658119658119658,\n",
      " 5: 1.0,\n",
      " 6: 0.9972144846796658,\n",
      " 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "310ddd0ba71941eb80b1e9592920ac96",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 4, Val Acc: \n",
      "{0: 0.9907213409158935,\n",
      " 1: 0.8044554455445545,\n",
      " 2: 0.9427816901408451,\n",
      " 3: 0.5476190476190477,\n",
      " 4: 0.593103448275862,\n",
      " 5: 0.95929203539823,\n",
      " 6: 0.8123393316195373,\n",
      " 7: 0.993091018323821}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "d4371810b62e44e6a1b557cbf5b4b3c9",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train loss = 0.0280020609498024\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "9ec5fa99635b464ca1e1027955f93249",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 5, Train Acc: \n",
      "{0: 1.0,\n",
      " 1: 0.9973262032085561,\n",
      " 2: 1.0,\n",
      " 3: 0.9716312056737588,\n",
      " 4: 0.9829059829059829,\n",
      " 5: 1.0,\n",
      " 6: 1.0,\n",
      " 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6e24dfd194544e7a8dddd885780378ca",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 5, Val Acc: \n",
      "{0: 0.9943130799161928,\n",
      " 1: 0.844059405940594,\n",
      " 2: 0.9621478873239436,\n",
      " 3: 0.6349206349206349,\n",
      " 4: 0.5241379310344828,\n",
      " 5: 0.9504424778761061,\n",
      " 6: 0.7660668380462725,\n",
      " 7: 0.993091018323821}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "7b9b64d100924b8f80404636df802d68",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train loss = 0.020310064777731895\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "bf5be8fa426649d28e9a1044432acb65",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 6, Train Acc: \n",
      "{0: 1.0, 1: 1.0, 2: 1.0, 3: 0.9716312056737588, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "3f5711b56a96401ea8ca90c2bf0c06c4",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 6, Val Acc: \n",
      "{0: 0.988027536665669,\n",
      " 1: 0.8044554455445545,\n",
      " 2: 0.9366197183098591,\n",
      " 3: 0.5555555555555556,\n",
      " 4: 0.6206896551724138,\n",
      " 5: 0.968141592920354,\n",
      " 6: 0.8303341902313625,\n",
      " 7: 0.9954941423851006}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "e9b82ae1ef4649b895c434457451253b",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train loss = 0.014955111779272556\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "1dfaa7e2c1a848b48343e8cfa3e9dbeb",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 7, Train Acc: \n",
      "{0: 1.0, 1: 1.0, 2: 1.0, 3: 0.9716312056737588, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "787fe21a7c6145fd90874fb7f8c11e42",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 7, Val Acc: \n",
      "{0: 0.99131996408261,\n",
      " 1: 0.8168316831683168,\n",
      " 2: 0.9498239436619719,\n",
      " 3: 0.5873015873015873,\n",
      " 4: 0.5862068965517241,\n",
      " 5: 0.9619469026548673,\n",
      " 6: 0.8303341902313625,\n",
      " 7: 0.9939921898468008}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "cba86d7bafb642ec9cd14e6979e4ae14",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train loss = 0.017731163650751114\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "f71cc7949d5c4f03b69af70b07f7b2c8",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 8, Train Acc: \n",
      "{0: 1.0, 1: 1.0, 2: 1.0, 3: 0.9787234042553191, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "7b307bdce955446d9e0b431e70ecafa6",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 8, Val Acc: \n",
      "{0: 0.9937144567494762,\n",
      " 1: 0.844059405940594,\n",
      " 2: 0.9621478873239436,\n",
      " 3: 0.7063492063492064,\n",
      " 4: 0.5586206896551724,\n",
      " 5: 0.9442477876106194,\n",
      " 6: 0.7634961439588689,\n",
      " 7: 0.9906878942625413}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "19b471dc294b492cacf8d20383b985ef",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train loss = 0.016594471409916878\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "73a2f50d7463448bb11826f586a7662c",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 9, Train Acc: \n",
      "{0: 1.0, 1: 1.0, 2: 1.0, 3: 0.9858156028368794, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "030e21a29bb5420c80e335b0af82fc6a",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 9, Val Acc: \n",
      "{0: 0.9937144567494762,\n",
      " 1: 0.8589108910891089,\n",
      " 2: 0.965669014084507,\n",
      " 3: 0.6666666666666666,\n",
      " 4: 0.503448275862069,\n",
      " 5: 0.9398230088495575,\n",
      " 6: 0.7557840616966581,\n",
      " 7: 0.992490237308501}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "5e6485d72c364cfcbdac00dcae5f6c92",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train loss = 0.012218784540891647\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "87738bdf6b834111ac4003773cf146db",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 10, Train Acc: \n",
      "{0: 1.0, 1: 1.0, 2: 1.0, 3: 0.9929078014184397, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "459301979d8d41c3a732a1762b3366b9",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 10, Val Acc: \n",
      "{0: 0.9877282250823107,\n",
      " 1: 0.8118811881188119,\n",
      " 2: 0.9463028169014085,\n",
      " 3: 0.5793650793650794,\n",
      " 4: 0.6482758620689655,\n",
      " 5: 0.9663716814159292,\n",
      " 6: 0.8149100257069408,\n",
      " 7: 0.9951937518774406}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "d21b7d51a61b4ba5b2a72e8b7d6a7a5c",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train loss = 0.010895946063101292\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "49dc3fbf881744e9a1d90bbbd8af1e85",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 11, Train Acc: \n",
      "{0: 1.0, 1: 1.0, 2: 1.0, 3: 0.9929078014184397, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6eb6c1c2184c4cec89f72f2df801c3ae",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 11, Val Acc: \n",
      "{0: 0.9856330439988028,\n",
      " 1: 0.7747524752475248,\n",
      " 2: 0.9322183098591549,\n",
      " 3: 0.5396825396825397,\n",
      " 4: 0.6275862068965518,\n",
      " 5: 0.9761061946902655,\n",
      " 6: 0.8586118251928021,\n",
      " 7: 0.9960949234004205}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "0d057dee930d4ce2a82f7c7a828c93f3",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train loss = 0.006740607786923647\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "59b2cfc7bdf44ae4959737396d8b03ed",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 12, Train Acc: \n",
      "{0: 1.0, 1: 1.0, 2: 1.0, 3: 0.9929078014184397, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "249e5b09ad824c809e91d7fa9f240295",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 12, Val Acc: \n",
      "{0: 0.9895240945824604,\n",
      " 1: 0.806930693069307,\n",
      " 2: 0.954225352112676,\n",
      " 3: 0.6666666666666666,\n",
      " 4: 0.593103448275862,\n",
      " 5: 0.9654867256637168,\n",
      " 6: 0.8020565552699229,\n",
      " 7: 0.993091018323821}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "2d0708aa2bea46799b24f3c6067fc5cf",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train loss = 0.0038992613554000854\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "82af661457064fce8b1df8b40085e9ca",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 13, Train Acc: \n",
      "{0: 1.0, 1: 1.0, 2: 1.0, 3: 0.9929078014184397, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "67725e5a20634134bdc68113c0621e1e",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 13, Val Acc: \n",
      "{0: 0.99131996408261,\n",
      " 1: 0.8267326732673267,\n",
      " 2: 0.960387323943662,\n",
      " 3: 0.6825396825396826,\n",
      " 4: 0.5379310344827586,\n",
      " 5: 0.95929203539823,\n",
      " 6: 0.7917737789203085,\n",
      " 7: 0.992490237308501}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "b3985530e3ae41f5b89edd1d25126269",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train loss = 0.003980405628681183\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "e92216991c4348e6b1ba892b2905336e",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 14, Train Acc: \n",
      "{0: 1.0, 1: 1.0, 2: 1.0, 3: 1.0, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "0c662eeb4f2f4a84a618e10fd1973493",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 14, Val Acc: \n",
      "{0: 0.9901227177491769,\n",
      " 1: 0.8118811881188119,\n",
      " 2: 0.9524647887323944,\n",
      " 3: 0.6111111111111112,\n",
      " 4: 0.6137931034482759,\n",
      " 5: 0.9663716814159292,\n",
      " 6: 0.8149100257069408,\n",
      " 7: 0.9945929708621207}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "e87eb050ecea425e8bd5e3138744d500",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train loss = 0.003820809070020914\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "ffb3e1936cb34f0993e42be2366b1a3b",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 15, Train Acc: \n",
      "{0: 1.0, 1: 1.0, 2: 1.0, 3: 1.0, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "af3aeb5461f3450c80cc5d4d0857a51a",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 15, Val Acc: \n",
      "{0: 0.9922178988326849,\n",
      " 1: 0.849009900990099,\n",
      " 2: 0.9639084507042254,\n",
      " 3: 0.7063492063492064,\n",
      " 4: 0.5862068965517241,\n",
      " 5: 0.9504424778761061,\n",
      " 6: 0.7840616966580977,\n",
      " 7: 0.9918894562931811}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "ea499c218e064e33adc541e5e18a5d1e",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train loss = 0.0059946151450276375\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "846bb4e21c6e443385488b2e24d283a3",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 16, Train Acc: \n",
      "{0: 1.0, 1: 1.0, 2: 1.0, 3: 1.0, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "1567b9657e54435695708b77b24ff3ad",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 16, Val Acc: \n",
      "{0: 0.9928165219994014,\n",
      " 1: 0.849009900990099,\n",
      " 2: 0.965669014084507,\n",
      " 3: 0.6984126984126984,\n",
      " 4: 0.5379310344827586,\n",
      " 5: 0.952212389380531,\n",
      " 6: 0.7763496143958869,\n",
      " 7: 0.9918894562931811}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "7c4be1fd082c4572a147f10c861b2eb1",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train loss = 0.004343008156865835\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "b4d4f609d4d444aa908ffa362ec8509a",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 17, Train Acc: \n",
      "{0: 1.0, 1: 1.0, 2: 1.0, 3: 1.0, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "cf9fda9b22da4c74ab0684378432e4f7",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 17, Val Acc: \n",
      "{0: 0.99131996408261,\n",
      " 1: 0.849009900990099,\n",
      " 2: 0.9647887323943662,\n",
      " 3: 0.6746031746031746,\n",
      " 4: 0.5862068965517241,\n",
      " 5: 0.9513274336283186,\n",
      " 6: 0.7737789203084833,\n",
      " 7: 0.9918894562931811}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "a06217605e5f45e98404d9322bc22826",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train loss = 0.0021464312449097633\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "b0376404df824c5180d3952a18436e0d",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 18, Train Acc: \n",
      "{0: 1.0, 1: 1.0, 2: 1.0, 3: 1.0, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "108f46945c384202bda0966c04f05f19",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 18, Val Acc: \n",
      "{0: 0.9919185872493266,\n",
      " 1: 0.849009900990099,\n",
      " 2: 0.9647887323943662,\n",
      " 3: 0.6666666666666666,\n",
      " 4: 0.6,\n",
      " 5: 0.9548672566371681,\n",
      " 6: 0.7840616966580977,\n",
      " 7: 0.9915890657855212}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "2ec85ef0752e4a389647e6bfe36e20d8",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train loss = 0.0015021211002022028\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "cce2e2df2802450589fe0212489ba4f3",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 19, Train Acc: \n",
      "{0: 1.0, 1: 1.0, 2: 1.0, 3: 1.0, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "2ed08ed056c14577bedcde7e3adca675",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 19, Val Acc: \n",
      "{0: 0.99131996408261,\n",
      " 1: 0.8465346534653465,\n",
      " 2: 0.960387323943662,\n",
      " 3: 0.6746031746031746,\n",
      " 4: 0.6068965517241379,\n",
      " 5: 0.9584070796460177,\n",
      " 6: 0.781491002570694,\n",
      " 7: 0.9912886752778612}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "01a29f14f76d4892842c13d836d395d5",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train loss = 0.001937918714247644\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "52cd1021cc9a4943b166d4bbb402301f",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 20, Train Acc: \n",
      "{0: 1.0, 1: 1.0, 2: 1.0, 3: 1.0, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "7b46fd0ebff1485c839fdf1ad7fc69a2",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 20, Val Acc: \n",
      "{0: 0.9919185872493266,\n",
      " 1: 0.849009900990099,\n",
      " 2: 0.965669014084507,\n",
      " 3: 0.7142857142857143,\n",
      " 4: 0.5793103448275863,\n",
      " 5: 0.9530973451327434,\n",
      " 6: 0.7866323907455013,\n",
      " 7: 0.9900871132472214}\n"
     ]
    }
   ],
   "source": [
    "for epoch in tqdm(range(epochs)):\n",
    "    model.train()\n",
    "    loss_total, count = 0, 0\n",
    "    for batch in tqdm(train_loader):\n",
    "        count += 1\n",
    "        X_batch, Y_batch, G_batch = batch\n",
    "        X_batch, Y_batch = X_batch.to(device), Y_batch.to(device)\n",
    "        optimizer.zero_grad()\n",
    "        outputs = model(X_batch)\n",
    "        loss = criterion(outputs, Y_batch)\n",
    "        loss_total += loss\n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "    print(f'Train loss = {loss_total/count}')\n",
    "    \n",
    "    model.eval()\n",
    "    correct_counts = {}\n",
    "    total_counts = {}\n",
    "    with torch.no_grad():\n",
    "        for batch in tqdm(train_loader):\n",
    "            X_batch, Y_batch, G_batch = batch\n",
    "            X_batch, Y_batch = X_batch.to(device), Y_batch.to(device)\n",
    "            outputs = model(X_batch)\n",
    "            _, predicted = torch.max(outputs, 1)\n",
    "            for g in torch.unique(G_batch):\n",
    "                g_idx = (G_batch == g).nonzero().squeeze()\n",
    "                if len(g_idx.shape) == 0: \n",
    "                    g_idx = g_idx.unsqueeze(0)\n",
    "                total = len(g_idx)\n",
    "                if total == 0:\n",
    "                    continue\n",
    "                correct = (predicted[g_idx] == Y_batch[g_idx]).sum().item()\n",
    "                if g.item() not in correct_counts:\n",
    "                    correct_counts[g.item()] = 0\n",
    "                    total_counts[g.item()] = 0\n",
    "                correct_counts[g.item()] += correct\n",
    "                total_counts[g.item()] += total\n",
    "    train_acc = {g: correct_counts[g] / total_counts[g] for g in correct_counts}\n",
    "    print(f\"Epoch {epoch+1}, Train Acc: \")\n",
    "    pprint(train_acc)\n",
    "    correct_counts = {}\n",
    "    total_counts = {}\n",
    "    with torch.no_grad():\n",
    "        for batch in tqdm(val_loader):\n",
    "            X_batch, Y_batch, G_batch = batch\n",
    "            X_batch, Y_batch = X_batch.to(device), Y_batch.to(device)\n",
    "            outputs = model(X_batch)\n",
    "            _, predicted = torch.max(outputs, 1)\n",
    "            for g in torch.unique(G_batch):\n",
    "                g_idx = (G_batch == g).nonzero().squeeze()\n",
    "                if len(g_idx.shape) == 0: \n",
    "                    g_idx = g_idx.unsqueeze(0)\n",
    "                total = len(g_idx)\n",
    "                if total == 0:\n",
    "                    continue\n",
    "                correct = (predicted[g_idx] == Y_batch[g_idx]).sum().item()\n",
    "                if g.item() not in correct_counts:\n",
    "                    correct_counts[g.item()] = 0\n",
    "                    total_counts[g.item()] = 0\n",
    "                correct_counts[g.item()] += correct\n",
    "                total_counts[g.item()] += total\n",
    "\n",
    "    val_acc = {g: correct_counts[g] / total_counts[g] for g in correct_counts}\n",
    "    print(f\"Epoch {epoch+1}, Val Acc: \")\n",
    "    pprint(val_acc)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "7e30d655-ba30-4d69-83f4-67de3c4e653a",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.save(model.state_dict(), 'dominoes_cmnist_fmnist_cifar_07095_seed42.model')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2058db27-abef-4835-aa1f-aa9e6022d545",
   "metadata": {},
   "source": [
    "# Evaluate ERM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "ece4f5e7-e34d-43a7-80d5-a9402d4c7678",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "ebae2851bdcb436c8ab684b21e63526d",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Train Acc: \n",
      "{0: 1.0, 1: 1.0, 2: 1.0, 3: 1.0, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    }
   ],
   "source": [
    "device = torch.device(\"cuda:0\" if torch.cuda.is_available() else \"cpu\")\n",
    "\n",
    "model = models.resnet18(weights=ResNet18_Weights.DEFAULT)\n",
    "num_ftrs = model.fc.in_features\n",
    "model.fc = nn.Linear(num_ftrs, 2)\n",
    "model.load_state_dict(torch.load('dominoes_cmnist_fmnist_cifar_07095_seed3.model', map_location=device))\n",
    "model.cuda()\n",
    "model.eval()\n",
    "correct_counts = {}\n",
    "total_counts = {}\n",
    "with torch.no_grad():\n",
    "    for batch in tqdm(train_loader):\n",
    "        X_batch, Y_batch, G_batch = batch\n",
    "        X_batch, Y_batch = X_batch.to(device), Y_batch.to(device)\n",
    "        outputs = model(X_batch)\n",
    "        _, predicted = torch.max(outputs, 1)\n",
    "        for g in torch.unique(G_batch):\n",
    "            g_idx = (G_batch == g).nonzero().squeeze()\n",
    "            if len(g_idx.shape) == 0: \n",
    "                g_idx = g_idx.unsqueeze(0)\n",
    "            total = len(g_idx)\n",
    "            if total == 0:\n",
    "                continue\n",
    "            correct = (predicted[g_idx] == Y_batch[g_idx]).sum().item()\n",
    "            if g.item() not in correct_counts:\n",
    "                correct_counts[g.item()] = 0\n",
    "                total_counts[g.item()] = 0\n",
    "            correct_counts[g.item()] += correct\n",
    "            total_counts[g.item()] += total\n",
    "\n",
    "train_acc = {g: correct_counts[g] / total_counts[g] for g in correct_counts}\n",
    "print(f\"Train Acc: \")\n",
    "pprint(train_acc)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 139,
   "id": "44e3fe29-02b5-4792-8d83-6621ef31f59c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "508cc263ae0946bfb3ac1041e59936f4",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test Acc: \n",
      "{0: 0.9904449089280383,\n",
      " 1: 0.8216216216216217,\n",
      " 2: 0.9467028003613369,\n",
      " 3: 0.5128205128205128,\n",
      " 4: 0.6271186440677966,\n",
      " 5: 0.9672566371681416,\n",
      " 6: 0.868421052631579,\n",
      " 7: 0.9941673957421989}\n"
     ]
    }
   ],
   "source": [
    "device = torch.device(\"cuda:0\" if torch.cuda.is_available() else \"cpu\")\n",
    "\n",
    "model = models.resnet18(weights=ResNet18_Weights.DEFAULT)\n",
    "num_ftrs = model.fc.in_features\n",
    "model.fc = nn.Linear(num_ftrs, 2)\n",
    "model.load_state_dict(torch.load('dominoes_cmnist_fmnist_cifar_07095.model', map_location=device))\n",
    "model.cuda()\n",
    "model.eval()\n",
    "correct_counts = {}\n",
    "total_counts = {}\n",
    "with torch.no_grad():\n",
    "    for batch in tqdm(test_loader):\n",
    "        X_batch, Y_batch, G_batch = batch\n",
    "        X_batch, Y_batch = X_batch.to(device), Y_batch.to(device)\n",
    "        outputs = model(X_batch)\n",
    "        _, predicted = torch.max(outputs, 1)\n",
    "        for g in torch.unique(G_batch):\n",
    "            g_idx = (G_batch == g).nonzero().squeeze()\n",
    "            if len(g_idx.shape) == 0: \n",
    "                g_idx = g_idx.unsqueeze(0)\n",
    "            total = len(g_idx)\n",
    "            if total == 0:\n",
    "                continue\n",
    "            correct = (predicted[g_idx] == Y_batch[g_idx]).sum().item()\n",
    "            if g.item() not in correct_counts:\n",
    "                correct_counts[g.item()] = 0\n",
    "                total_counts[g.item()] = 0\n",
    "            correct_counts[g.item()] += correct\n",
    "            total_counts[g.item()] += total\n",
    "\n",
    "test_acc = {g: correct_counts[g] / total_counts[g] for g in correct_counts}\n",
    "print(f\"Test Acc: \")\n",
    "pprint(test_acc)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "44833c94-a801-4297-b9f7-0a8fb072378e",
   "metadata": {},
   "source": [
    "# DFR"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 119,
   "id": "271a6649-29d8-4284-9472-b30cf6320ce0",
   "metadata": {},
   "outputs": [],
   "source": [
    "device = torch.device(\"cuda:0\" if torch.cuda.is_available() else \"cpu\")\n",
    "\n",
    "model = models.resnet18(weights=ResNet18_Weights.DEFAULT)\n",
    "num_ftrs = model.fc.in_features\n",
    "model.fc = nn.Linear(num_ftrs, 2)\n",
    "model.load_state_dict(torch.load('dominoes_cmnist_fmnist_cifar_07095_seed3.model', map_location=device))\n",
    "model.cuda()\n",
    "model.eval()\n",
    "classifier = model.fc\n",
    "model.fc = torch.nn.Identity()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 120,
   "id": "ab14828d-2db7-415c-98c6-6f87bae64f80",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "5517eae59ff74105879488b4944724de",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "41db74309bf142d5b518805d4c7d41ef",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6a7a7f1e0a9d4ea3bdb75913fc85724b",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "de4aeabc0f974ff183a4d4900a8c488d",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "EIIL = True\n",
    "if not EIIL:\n",
    "    all_embeddings = {}\n",
    "    all_y, all_g = {}, {}\n",
    "for name, loader in [(\"lastlayer\", lastlayer_loader), (\"val\", val_loader), (\"train\", train_loader)]:\n",
    "    all_embeddings[name] = []\n",
    "    all_y[name] = []\n",
    "    if name != 'val':\n",
    "        all_g[name] = []\n",
    "    for x, y, g in tqdm(loader):\n",
    "        with torch.no_grad():\n",
    "            all_embeddings[name].append(model(x.cuda()).detach().cpu().numpy())\n",
    "            all_y[name].append(y.detach().cpu().numpy())\n",
    "            if name != 'val':\n",
    "                all_g[name].append(g.detach().cpu().numpy()//2)\n",
    "    all_embeddings[name] = np.vstack(all_embeddings[name])\n",
    "    all_y[name] = np.concatenate(all_y[name])\n",
    "    if name != 'val':\n",
    "        all_g[name] = np.concatenate(all_g[name])\n",
    "# Test split\n",
    "for name, loader in [(\"test\", test_loader)]:\n",
    "    all_embeddings[name] = []\n",
    "    all_y[name], all_g[name] = [], []\n",
    "    for x, y, g in tqdm(loader):\n",
    "        with torch.no_grad():\n",
    "            all_embeddings[name].append(model(x.cuda()).detach().cpu().numpy())\n",
    "            all_y[name].append(y.detach().cpu().numpy())\n",
    "            all_g[name].append(g.detach().cpu().numpy())\n",
    "    all_embeddings[name] = np.vstack(all_embeddings[name])\n",
    "    all_y[name] = np.concatenate(all_y[name])\n",
    "    all_g[name] = np.concatenate(all_g[name])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 121,
   "id": "eff8d9e2-713f-4ebc-9676-f76d55fd77cf",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "((10000, 512), (10000, 512), (10000, 512))"
      ]
     },
     "execution_count": 121,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "all_embeddings['lastlayer'].shape, all_embeddings['val'].shape, all_embeddings['train'].shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "08d07896-9309-421d-9a1c-fa081c1533a8",
   "metadata": {},
   "outputs": [],
   "source": [
    "def select_high_low_loss_samples(embedding_tensor, classifier, labels_tensor, k, groups_tensor):\n",
    "    embedding_tensor = torch.tensor(embedding_tensor)\n",
    "    labels_tensor = torch.tensor(labels_tensor)\n",
    "    groups_tensor = torch.tensor(groups_tensor)\n",
    "    logits = classifier(embedding_tensor.cuda()).detach().cpu()\n",
    "    loss = torch.nn.functional.cross_entropy(logits, labels_tensor, reduction='none')\n",
    "    unique_labels = torch.unique(labels_tensor)\n",
    "    high_low_loss_samples = []\n",
    "    y, g = [], []\n",
    "    for label in unique_labels:\n",
    "        label_mask = labels_tensor == label\n",
    "        label_loss = loss[label_mask]\n",
    "        label_embedding = embedding_tensor[label_mask]\n",
    "        label_y = labels_tensor[label_mask]\n",
    "        label_g = groups_tensor[label_mask]\n",
    "        _, high_loss_indices = torch.topk(label_loss, k)\n",
    "        _, low_loss_indices = torch.topk(-label_loss, k)\n",
    "        high_loss_samples = label_embedding[high_loss_indices]\n",
    "        low_loss_samples = label_embedding[low_loss_indices]\n",
    "        high_loss_y = label_y[high_loss_indices]\n",
    "        low_loss_y = label_y[low_loss_indices]\n",
    "        high_loss_g = label_g[high_loss_indices]\n",
    "        low_loss_g = label_g[low_loss_indices]\n",
    "        high_low_loss_samples.append(torch.cat((high_loss_samples, low_loss_samples), dim=0))\n",
    "        y.append(torch.cat((high_loss_y, low_loss_y), dim=0))\n",
    "        g.append(torch.cat((high_loss_g, low_loss_g), dim=0))\n",
    "    high_low_loss_samples = torch.cat(high_low_loss_samples, dim=0).numpy()\n",
    "    high_low_loss_y = torch.cat(y, dim=0).numpy()\n",
    "    high_low_loss_g = torch.cat(g, dim=0).numpy()\n",
    "    return high_low_loss_samples, high_low_loss_y, high_low_loss_g"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "b7ad23fd-09e7-4db2-b433-b89dce9b431f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(40, 512) (40,) (40,)\n"
     ]
    }
   ],
   "source": [
    "k = 10\n",
    "x, y, g = select_high_low_loss_samples(all_embeddings['lastlayer'], classifier, all_y['lastlayer'], k, all_g['lastlayer'])\n",
    "print(x.shape, y.shape, g.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 123,
   "id": "ef384dda-b5dd-42d4-a4ba-368873e2d4df",
   "metadata": {},
   "outputs": [],
   "source": [
    "from sklearn.linear_model import LogisticRegression\n",
    "from sklearn.preprocessing import StandardScaler\n",
    "\n",
    "C_OPTIONS = [3., 1., 0.7, 0.5, 0.3, 0.2, 0.1, 0.07, 0.05, 0.03, 0.02, 0.01, 0.003, 0.001]\n",
    "CLASS_WEIGHT_OPTIONS = [1., 2., 3., 10., 100., 300., 500., 1000.]\n",
    "REG = \"l1\"\n",
    "CLASS_WEIGHT_OPTIONS = [{0: 1, 1: w} for w in CLASS_WEIGHT_OPTIONS] + \\\n",
    "                       [{0: w, 1: 1} for w in CLASS_WEIGHT_OPTIONS]\n",
    "\n",
    "def dfr_on_validation_tune(\n",
    "        all_embeddings, all_y, all_g, preprocess=False,\n",
    "        balance_val=True, add_train=False, num_retrains=1, loss_based=False):\n",
    "    global classifer, k\n",
    "    print(k, classifier)\n",
    "    worst_accs = {}\n",
    "    for i in range(num_retrains):\n",
    "        x_val = all_embeddings[\"val\"]\n",
    "        y_val = all_y[\"val\"]\n",
    "        g_val = all_g[\"val\"]\n",
    "        n_groups = np.max(g_val) + 1\n",
    "\n",
    "        n_val = len(all_embeddings[\"val\"])\n",
    "        \n",
    "        x_valtrain = all_embeddings[\"lastlayer\"]\n",
    "        y_valtrain = all_y[\"lastlayer\"]\n",
    "        g_valtrain = all_g[\"lastlayer\"]\n",
    "        if loss_based:\n",
    "            x_valtrain, y_valtrain, g_valtrain = select_high_low_loss_samples(x_valtrain, classifier, y_valtrain, k, g_valtrain)\n",
    "        n_groups = np.max(g_valtrain) + 1\n",
    "        g_idx = [np.where(g_valtrain == g)[0] for g in range(n_groups)]\n",
    "        min_g = np.min([len(g) for g in g_idx])\n",
    "        for g in g_idx:\n",
    "            np.random.shuffle(g)\n",
    "        if balance_val:\n",
    "            x_valtrain = np.concatenate([x_valtrain[g[:min_g]] for g in g_idx])\n",
    "            y_valtrain = np.concatenate([y_valtrain[g[:min_g]] for g in g_idx])\n",
    "            g_valtrain = np.concatenate([g_valtrain[g[:min_g]] for g in g_idx])\n",
    "\n",
    "        # x_val = x_val[idx[:n_val]]\n",
    "        # y_val = y_val[idx[:n_val]]\n",
    "        # g_val = g_val[idx[:n_val]]\n",
    "\n",
    "        n_train = len(x_valtrain) if add_train else 0\n",
    "\n",
    "        x_train = np.concatenate([all_embeddings[\"train\"][:n_train], x_valtrain])\n",
    "        y_train = np.concatenate([all_y[\"train\"][:n_train], y_valtrain])\n",
    "        g_train = np.concatenate([all_g[\"train\"][:n_train], g_valtrain])\n",
    "        # print(np.bincount(g_train))\n",
    "        if preprocess:\n",
    "            scaler = StandardScaler()\n",
    "            x_train = scaler.fit_transform(x_train)\n",
    "            x_val = scaler.transform(x_val)\n",
    "\n",
    "\n",
    "        if balance_val and not add_train:\n",
    "            cls_w_options = [{0: 1., 1: 1.}]\n",
    "        else:\n",
    "            cls_w_options = CLASS_WEIGHT_OPTIONS\n",
    "        for c in C_OPTIONS:\n",
    "            for class_weight in cls_w_options:\n",
    "                logreg = LogisticRegression(penalty=REG, C=c, solver=\"liblinear\",\n",
    "                                            class_weight=class_weight)\n",
    "                logreg.fit(x_train, y_train)\n",
    "                preds_val = logreg.predict(x_val)\n",
    "                group_accs = np.array(\n",
    "                    [(preds_val == y_val)[g_val == g].mean()\n",
    "                     for g in range(n_groups)])\n",
    "                worst_acc = np.min(group_accs)\n",
    "                if i == 0:\n",
    "                    worst_accs[c, class_weight[0], class_weight[1]] = worst_acc\n",
    "                else:\n",
    "                    worst_accs[c, class_weight[0], class_weight[1]] += worst_acc\n",
    "    ks, vs = list(worst_accs.keys()), list(worst_accs.values())\n",
    "    best_hypers = ks[np.argmax(vs)]\n",
    "    print(worst_accs[best_hypers]/num_retrains)\n",
    "    return best_hypers\n",
    "\n",
    "def dfr_on_validation_eval(\n",
    "        c, w1, w2, all_embeddings, all_y, all_g, num_retrains=3,\n",
    "        preprocess=False, balance_val=True, add_train=False, loss_based=False):\n",
    "    coefs, intercepts = [], []\n",
    "    if preprocess:\n",
    "        scaler = StandardScaler()\n",
    "        scaler.fit(all_embeddings[\"train\"])\n",
    "\n",
    "    for i in range(num_retrains):\n",
    "        x_val = all_embeddings[\"lastlayer\"]\n",
    "        y_val = all_y[\"lastlayer\"]\n",
    "        g_val = all_g[\"lastlayer\"]\n",
    "        if loss_based:\n",
    "            x_val, y_val, g_val = select_high_low_loss_samples(x_val, classifier, y_val, k, g_val)\n",
    "        n_groups = np.max(g_val) + 1\n",
    "        g_idx = [np.where(g_val == g)[0] for g in range(n_groups)]\n",
    "        min_g = np.min([len(g) for g in g_idx])\n",
    "        for g in g_idx:\n",
    "            np.random.shuffle(g)\n",
    "        if balance_val:\n",
    "            x_val = np.concatenate([x_val[g[:min_g]] for g in g_idx])\n",
    "            y_val = np.concatenate([y_val[g[:min_g]] for g in g_idx])\n",
    "            g_val = np.concatenate([g_val[g[:min_g]] for g in g_idx])\n",
    "\n",
    "        n_train = len(x_val) if add_train else 0\n",
    "        train_idx = np.arange(len(all_embeddings[\"train\"]))\n",
    "        np.random.shuffle(train_idx)\n",
    "        train_idx = train_idx[:n_train]\n",
    "\n",
    "        x_train = np.concatenate(\n",
    "            [all_embeddings[\"train\"][train_idx], x_val])\n",
    "        y_train = np.concatenate([all_y[\"train\"][train_idx], y_val])\n",
    "        g_train = np.concatenate([all_g[\"train\"][train_idx], g_val])\n",
    "        # print(np.bincount(g_train))\n",
    "        if preprocess:\n",
    "            x_train = scaler.transform(x_train)\n",
    "\n",
    "        logreg = LogisticRegression(penalty=REG, C=c, solver=\"liblinear\",\n",
    "                                    class_weight={0: w1, 1: w2})\n",
    "        logreg.fit(x_train, y_train)\n",
    "        coefs.append(logreg.coef_)\n",
    "        intercepts.append(logreg.intercept_)\n",
    "\n",
    "    x_test = all_embeddings[\"test\"]\n",
    "    y_test = all_y[\"test\"]\n",
    "    g_test = all_g[\"test\"]\n",
    "    # print(np.bincount(g_test))\n",
    "\n",
    "    if preprocess:\n",
    "        x_test = scaler.transform(x_test)\n",
    "    logreg = LogisticRegression(penalty=REG, C=c, solver=\"liblinear\",\n",
    "                                class_weight={0: w1, 1: w2})\n",
    "    n_classes = np.max(y_train) + 1\n",
    "    # the fit is only needed to set up logreg\n",
    "    logreg.fit(x_train[:n_classes], np.arange(n_classes))\n",
    "    logreg.coef_ = np.mean(coefs, axis=0)\n",
    "    logreg.intercept_ = np.mean(intercepts, axis=0)\n",
    "    preds_test = logreg.predict(x_test)\n",
    "    preds_train = logreg.predict(x_train)\n",
    "    n_groups = np.max(g_train) + 1\n",
    "    train_accs = [(preds_train == y_train)[g_train == g].mean()\n",
    "                  for g in range(n_groups)]\n",
    "    n_groups = np.max(g_test) + 1\n",
    "    test_accs = [(preds_test == y_test)[g_test == g].mean()\n",
    "                 for g in range(n_groups)]\n",
    "    test_mean_acc = (preds_test == y_test).mean()\n",
    "    return test_accs, test_mean_acc, train_accs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 124,
   "id": "c8866347-754d-442e-a932-8240b603fc92",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0 Linear(in_features=512, out_features=2, bias=True)\n",
      "0.8272017837235228\n",
      "Hypers: (1.0, 1.0, 1.0)\n",
      "{'best_hypers': (1.0, 1.0, 1.0), 'test_accs': [0.9581964765601673, 0.7378378378378379, 0.9629629629629629, 0.7264957264957265, 0.5847457627118644, 0.9504424778761061, 0.7894736842105263, 0.973753280839895], 'train_accs': [0.9720164609053498, 0.9827160493827161, 0.9843621399176955, 0.9695473251028807], 'test_worst_acc': 0.5847457627118644, 'test_mean_acc': 0.9415}\n",
      "\n"
     ]
    }
   ],
   "source": [
    "k = 0\n",
    "dfr_lastlayer_results = {}\n",
    "c, w1, w2 = dfr_on_validation_tune(\n",
    "    all_embeddings, all_y, all_g,\n",
    "    balance_val=True, add_train=False)\n",
    "dfr_lastlayer_results[\"best_hypers\"] = (c, w1, w2)\n",
    "print(\"Hypers:\", (c, w1, w2))\n",
    "test_accs, test_mean_acc, train_accs = dfr_on_validation_eval(\n",
    "        c, w1, w2, all_embeddings, all_y, all_g,\n",
    "    balance_val=True, add_train=False)\n",
    "dfr_lastlayer_results[\"test_accs\"] = test_accs\n",
    "dfr_lastlayer_results[\"train_accs\"] = train_accs\n",
    "dfr_lastlayer_results[\"test_worst_acc\"] = np.min(test_accs)\n",
    "dfr_lastlayer_results[\"test_mean_acc\"] = test_mean_acc\n",
    "print(dfr_lastlayer_results)\n",
    "print()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "decd45ee-3cad-4542-8c5f-3db6b3551e4c",
   "metadata": {
    "jp-MarkdownHeadingCollapsed": true
   },
   "source": [
    "# EVaLS-GL"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "bc4584fc-259c-4aa8-8773-36b93094d838",
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "for k in range(5, 51):\n",
    "    dfr_lastlayer_results = {}\n",
    "    c, w1, w2 = dfr_on_validation_tune(\n",
    "        all_embeddings, all_y, all_g,\n",
    "        balance_val=False, add_train=False, loss_based=True)\n",
    "    dfr_lastlayer_results[\"best_hypers\"] = (c, w1, w2)\n",
    "    print(\"Hypers:\", (c, w1, w2))\n",
    "    test_accs, test_mean_acc, train_accs = dfr_on_validation_eval(\n",
    "            c, w1, w2, all_embeddings, all_y, all_g,\n",
    "        balance_val=False, add_train=False, loss_based=True)\n",
    "    dfr_lastlayer_results[\"test_accs\"] = test_accs\n",
    "    dfr_lastlayer_results[\"train_accs\"] = train_accs\n",
    "    dfr_lastlayer_results[\"test_worst_acc\"] = np.min(test_accs)\n",
    "    dfr_lastlayer_results[\"test_mean_acc\"] = test_mean_acc\n",
    "    print(f'\\n##############\\nfor k={k}:', dfr_lastlayer_results['test_worst_acc'], '\\n##############\\n')\n",
    "    pprint(dfr_lastlayer_results)\n",
    "    print('-'*60)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "35192da0-2aed-4813-b05d-70252256af37",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "petra",
   "language": "python",
   "name": "petra"
  },
  "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.11.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
