{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "47681c43-fd88-49af-ab0a-51931254ba5b",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.optim as optim\n",
    "from torchvision import models\n",
    "from torch.utils.data import TensorDataset, DataLoader\n",
    "import numpy as np\n",
    "from torchvision.models import ResNet18_Weights\n",
    "from tqdm.notebook import tqdm"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "12ad3103-438b-4539-bb06-0e916168fe99",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torchvision\n",
    "from torch.utils.data import DataLoader, Dataset, random_split\n",
    "from matplotlib import pyplot as plt\n",
    "import random\n",
    "import os\n",
    "import math\n",
    "from pprint import pprint"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "40baca94-3599-4867-b6ac-c13d78fc4e61",
   "metadata": {},
   "outputs": [],
   "source": [
    "seed = 0\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": {
    "jp-MarkdownHeadingCollapsed": true
   },
   "source": [
    "# Dataset Creation"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c186ca23-ed1f-4c49-85c2-58a052d96efd",
   "metadata": {},
   "source": [
    "spurious correlation:\n",
    "- shape: 75%, color: 85%\n",
    "- shape: 75%, color: 90%\n",
    "- shape: 75%, color: 95%"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "3d5af2db-610d-442d-a968-f80c58445690",
   "metadata": {},
   "outputs": [],
   "source": [
    "TOTAL = 40_000\n",
    "s1 = 0.95\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 = int(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 = int(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": [
      "14250\n",
      "750\n",
      "4750\n",
      "250\n",
      "250\n",
      "4750\n",
      "750\n",
      "14250\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": "ee9aa8e4-02b2-4de9-a196-319fa9a4051b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(tensor(-2.1179), tensor(-1.5367), tensor(2.2489))"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "X_mnist.min(), X_mnist.mean(), X_mnist.max()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "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": 12,
   "id": "db190aa4-6442-44cb-b245-9b9f22e3bfd0",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([24000, 1, 28, 28])"
      ]
     },
     "execution_count": 12,
     "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": 13,
   "id": "5b0196b4-8519-4631-ac69-6d6ddece2922",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(tensor(-2.1179), tensor(-0.6969), tensor(2.2489))"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "X_fmnist.min(), X_fmnist.mean(), X_fmnist.max()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "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": 15,
   "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": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "red_mnist.shape, green_mnist.shape, red_fmnist.shape, green_fmnist.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "ed7b717d-ef0f-4f53-8adc-6f1a8adcb9e0",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(torch.Size([14250, 3, 32, 32]), torch.Size([250, 3, 32, 32]))"
      ]
     },
     "execution_count": 16,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "idx = torch.randperm(red_mnist.shape[0])\n",
    "red_mnist_0_idx = idx[:c1_m_red]\n",
    "red_mnist_4_idx = idx[c1_m_red:c1_m_red+c2_m_red]\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": 17,
   "id": "50a2ffa7-a41b-4cbe-833f-d62175bb8b94",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(torch.Size([750, 3, 32, 32]), torch.Size([4750, 3, 32, 32]))"
      ]
     },
     "execution_count": 17,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "idx = torch.randperm(green_mnist.shape[0])\n",
    "green_mnist_1_idx = idx[:c1_m_green]\n",
    "green_mnist_5_idx = idx[c1_m_green:c1_m_green+c2_m_green]\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": 18,
   "id": "e369e2c2-5d8c-4ed2-a9d9-3abeef593dd0",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(torch.Size([4750, 3, 32, 32]), torch.Size([750, 3, 32, 32]))"
      ]
     },
     "execution_count": 18,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "idx = torch.randperm(red_fmnist.shape[0])\n",
    "red_fmnist_2_idx = idx[:c1_fm_red]\n",
    "red_fmnist_6_idx = idx[c1_fm_red:c1_fm_red+c2_fm_red]\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": 19,
   "id": "e2fbd487-ad89-4132-89a0-6e1fc7c8ea85",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(torch.Size([250, 3, 32, 32]), torch.Size([14250, 3, 32, 32]))"
      ]
     },
     "execution_count": 19,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "idx = torch.randperm(green_fmnist.shape[0])\n",
    "green_fmnist_3_idx = idx[:c1_fm_green]\n",
    "green_fmnist_7_idx = idx[c1_fm_green:c1_fm_green+c2_fm_green]\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": 20,
   "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": 20,
     "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": 21,
   "id": "ef4e1b89-89e9-4c7b-ad47-a5645080973d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(tensor(-2.1179), tensor(0.2564), tensor(2.6400))"
      ]
     },
     "execution_count": 21,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "X_c_cars.min(), X_c_cars.mean(), X_c_cars.max()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "dd3bec7e-22d9-4d4a-a614-564c5e484fe9",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(tensor(-2.1179), tensor(0.0274), tensor(2.6400))"
      ]
     },
     "execution_count": 22,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "X_c_animals.min(), X_c_animals.mean(), X_c_animals.max()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "f84cb04d-1a47-4f84-aadd-d0e3a03d97e4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(torch.Size([14250, 3, 32, 32]),\n",
       " torch.Size([750, 3, 32, 32]),\n",
       " torch.Size([4750, 3, 32, 32]),\n",
       " torch.Size([250, 3, 32, 32]))"
      ]
     },
     "execution_count": 23,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "X_c_cars_0 = X_c_cars[:c1_m_red]\n",
    "X_c_cars_1 = X_c_cars[c1_m_red:c1_m_red+c1_m_green]\n",
    "X_c_cars_2 = X_c_cars[c1_m_red+c1_m_green:c1_m_red+c1_m_green+c1_fm_red]\n",
    "X_c_cars_3 = X_c_cars[c1_m_red+c1_m_green+c1_fm_red:]\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": 24,
   "id": "1b77e629-33e4-4457-b814-6fe5e20e1b07",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(torch.Size([14250, 3, 32, 32]),\n",
       " torch.Size([750, 3, 32, 32]),\n",
       " torch.Size([4750, 3, 32, 32]),\n",
       " torch.Size([250, 3, 32, 32]))"
      ]
     },
     "execution_count": 24,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "X_c_animals_7 = X_c_animals[:c2_fm_green]\n",
    "X_c_animals_6 = X_c_animals[c2_fm_green:c2_fm_green+c2_fm_red]\n",
    "X_c_animals_5 = X_c_animals[c2_fm_green+c2_fm_red:c2_fm_green+c2_fm_red+c2_m_green]\n",
    "X_c_animals_4 = X_c_animals[c2_fm_green+c2_fm_red+c2_m_green:]\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": 25,
   "id": "6ca938b3-b750-4b78-8af9-5383d5fa1388",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([14250, 3, 64, 32])"
      ]
     },
     "execution_count": 25,
     "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": 26,
   "id": "ff7a3077-0de7-4377-8acd-913935fce030",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([750, 3, 64, 32])"
      ]
     },
     "execution_count": 26,
     "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": 27,
   "id": "4921b503-0c25-42de-aa6a-49b4d849b479",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([4750, 3, 64, 32])"
      ]
     },
     "execution_count": 27,
     "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": 28,
   "id": "6b5c4019-e443-4a68-8cff-04630c469e0b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([250, 3, 64, 32])"
      ]
     },
     "execution_count": 28,
     "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": 29,
   "id": "3a240a8a-c286-422e-8a93-8ea713dfafbd",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([250, 3, 64, 32])"
      ]
     },
     "execution_count": 29,
     "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": 30,
   "id": "fca75c98-8591-4560-87e4-a7c5d695bbd7",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([4750, 3, 64, 32])"
      ]
     },
     "execution_count": 30,
     "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": 31,
   "id": "d48bd971-115f-4bc9-a104-fc4b685e1d83",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([750, 3, 64, 32])"
      ]
     },
     "execution_count": 31,
     "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": 32,
   "id": "0f45a391-483d-44a7-8d90-c5c4df3f33e3",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([14250, 3, 64, 32])"
      ]
     },
     "execution_count": 32,
     "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": 33,
   "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": 33,
     "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": 34,
   "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([14250,   750,  4750,   250,   250,  4750,   750, 14250])))"
      ]
     },
     "execution_count": 34,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "Y.unique(return_counts=True), G.unique(return_counts=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "f636d377-af1b-4b31-9efc-38a90a6035a4",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.save(X, 'cifar_cmnist_fmnist_dominoe_X_9575.pt')\n",
    "torch.save(Y, 'cifar_cmnist_fmnist_dominoe_Y_9575.pt')\n",
    "torch.save(G, 'cifar_cmnist_fmnist_dominoe_G_9575.pt')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0f6aad9c-8314-4186-8092-17cfa78b372c",
   "metadata": {},
   "outputs": [],
   "source": [
    "invTrans = torchvision.transforms.Compose([ torchvision.transforms.Normalize(mean = [ 0., 0., 0. ],\n",
    "                                                     std = [ 1/0.229, 1/0.224, 1/0.225 ]),\n",
    "                                            torchvision.transforms.Normalize(mean = [ -0.485, -0.456, -0.406 ],\n",
    "                                                     std = [ 1., 1., 1. ]),\n",
    "                               ])\n",
    "\n",
    "for i in [0, 13500, 15000, 19500, 20000, 20500, 25000, 26500]:\n",
    "    # plt.imshow(invTrans(X[i]).permute(1,2,0))\n",
    "    X[i][:, :32, :] = (X[i][:, :32, :]-X[i][:, :32, :].min())/(X[i][:, :32, :].max()-X[i][:, :32, :].min())\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": 37,
   "id": "1d1b05cb-98ff-41c1-b42f-a9a5f5ca85ff",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(torch.Size([40000, 3, 64, 32]), torch.Size([40000]))"
      ]
     },
     "execution_count": 37,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "X.shape, Y.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "id": "3d7b97f1-e45d-45be-84c0-89e54b22727c",
   "metadata": {},
   "outputs": [],
   "source": [
    "dataset = TensorDataset(X, Y.long(), G.long())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "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": 40,
   "id": "4863b952-2e6f-455f-a46b-0a3fb7b97d7c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(10000, 10000, 10000, 10000)"
      ]
     },
     "execution_count": 40,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(train_dataset), len(lastlayer_dataset), len(val_dataset), len(test_dataset)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "id": "ba9c4d6d-73ac-40fb-bc02-b160c3680e17",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pickle\n",
    "with open('train_indices_9575.pkl', 'wb') as f:\n",
    "    pickle.dump(train_dataset.indices, f)\n",
    "with open('lastlayer_indices_9575.pkl', 'wb') as f:\n",
    "    pickle.dump(lastlayer_dataset.indices, f)\n",
    "with open('val_indices_9575.pkl', 'wb') as f:\n",
    "    pickle.dump(val_dataset.indices, f)\n",
    "with open('test_indices_9575.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": 89,
   "id": "300bad43-8463-401e-8554-22532826d48c",
   "metadata": {},
   "outputs": [],
   "source": [
    "EIIL = False"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 90,
   "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",
    "if EIIL:\n",
    "    G_val = torch.load('domino_val_envs.pt').argmax(1)\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": 44,
   "id": "8465ab3d-1a81-4ccc-b509-c89d838e337a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([3533,  212, 1194,   68,   77, 1198,  190, 3528])"
      ]
     },
     "execution_count": 44,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.bincount(G_val.numpy())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 45,
   "id": "5870748e-0aac-47d7-878c-11366b1cf03f",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(None, None, None)"
      ]
     },
     "execution_count": 45,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# run to save the data\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",
    "torch.save(X_tr, 'train_images_9575.pt'), torch.save(Y_tr, 'train_labels_9575.pt'), torch.save(G_tr, 'train_envs_9575.pt')\n",
    "torch.save(X_ll, 'lastlayer_images_9575.pt'), torch.save(Y_ll, 'lastlayer_labels_9575.pt'), torch.save(G_ll, 'lastlayer_envs_9575.pt')\n",
    "torch.save(X_val, 'val_images_9575.pt'), torch.save(Y_val, 'val_labels_9575.pt'), torch.save(G_val, 'val_envs_9575.pt')\n",
    "torch.save(X_te, 'test_images_9575.pt'), torch.save(Y_te, 'test_labels_9575.pt'), torch.save(G_te, 'test_envs_9575.pt')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d27d89d4-471d-4ff0-959c-98f87ea94e6f",
   "metadata": {},
   "source": [
    "## Feature load instead of images"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "id": "7ac25e91-04e4-480e-9cc8-0718d24de54b",
   "metadata": {},
   "outputs": [],
   "source": [
    "# for loading images\n",
    "X_tr, Y_tr, G_tr = torch.load('train_images_8575.pt'), torch.load('train_labels_8575.pt'), torch.load('train_envs_8575.pt')\n",
    "X_ll, Y_ll, G_ll = torch.load('lastlayer_images_8575.pt'), torch.load('lastlayer_labels_8575.pt'), torch.load('lastlayer_envs_8575.pt')\n",
    "X_val, Y_val, G_val = torch.load('val_images_8575.pt'), torch.load('val_labels_8575.pt'), torch.load('val_envs_8575.pt')\n",
    "X_te, Y_te, G_te = torch.load('test_images_8575.pt'), torch.load('test_labels_8575.pt'), torch.load('test_envs_8575.pt')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 138,
   "id": "6294ebd8-13af-45f7-a7e7-74c8c29141cc",
   "metadata": {},
   "outputs": [],
   "source": [
    "# for loading features\n",
    "X_tr, Y_tr, G_tr = torch.load('train_features_8575.pt'), torch.load('train_labels_8575.pt'), torch.load('train_envs_8575.pt')\n",
    "X_ll, Y_ll, G_ll = torch.load('lastlayer_features_8575.pt'), torch.load('lastlayer_labels_8575.pt'), torch.load('lastlayer_envs_8575.pt')\n",
    "X_val, Y_val, G_val = torch.load('val_features_8575.pt'), torch.load('val_labels_8575.pt'), torch.load('val_envs_8575.pt')\n",
    "X_te, Y_te, G_te = torch.load('test_features_8575.pt'), torch.load('test_labels_8575.pt'), torch.load('test_envs_8575.pt')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 127,
   "id": "faddb1f0-f264-40a6-9b50-703980c14289",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([10000, 512]) torch.Size([10000]) torch.Size([10000])\n",
      "torch.Size([10000, 512]) torch.Size([10000]) torch.Size([10000])\n",
      "torch.Size([10000, 512]) torch.Size([10000]) torch.Size([10000])\n",
      "torch.Size([10000, 512]) torch.Size([10000]) torch.Size([10000])\n"
     ]
    }
   ],
   "source": [
    "print(X_tr.shape, Y_tr.shape, G_tr.shape)\n",
    "print(X_ll.shape, Y_ll.shape, G_ll.shape)\n",
    "print(X_val.shape, Y_val.shape, G_val.shape)\n",
    "print(X_te.shape, Y_te.shape, G_te.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "d41af2f3-58d3-4f3a-8d51-5f65f81d690f",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Y_tr, G_tr = Y_tr.argmax(1), G_tr.argmax(1)\n",
    "# Y_ll, G_ll = Y_ll.argmax(1), G_ll.argmax(1)\n",
    "# Y_val, G_val = Y_val.argmax(1), G_val.argmax(1)\n",
    "# Y_te, G_te =  Y_te.argmax(1), G_te.argmax(1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 86,
   "id": "58eb350a-6d5e-4554-9fe7-46291975728b",
   "metadata": {},
   "outputs": [],
   "source": [
    "train_dataset = TensorDataset(X_tr, Y_tr, G_tr)\n",
    "lastlayer_dataset = TensorDataset(X_ll, Y_ll, G_ll)\n",
    "val_dataset = TensorDataset(X_val, Y_val, G_val)\n",
    "test_dataset = TensorDataset(X_te, Y_te, G_te)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "9c6b06ec-dd44-4d42-88d4-a852679ca58d",
   "metadata": {},
   "outputs": [],
   "source": [
    "if EIIL:\n",
    "    all_embeddings, all_y, all_g = {}, {}, {}\n",
    "    # all_embeddings['val'] = X_val.numpy()\n",
    "    # all_y['val'] = Y_val.numpy()\n",
    "    all_g['val'] = G_val.numpy()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "44cdfe6d-ba76-457d-9584-a39715a84427",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(array([0, 1, 2, 3]), array([4159,  848, 4135,  858]))"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.unique(all_g['val'], return_counts=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "64c6985f-b12a-412f-8813-c0c47a430fef",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(tensor([0, 1, 2, 3, 4, 5, 6, 7]),\n",
       " tensor([3341,  404, 1136,  126,  145, 1130,  389, 3329]))"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "G_val.unique(return_counts=True) # results for EIIl"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "b1c43925-8b8b-4f42-ab1f-7dacca1d39f2",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(tensor([0, 1, 2, 3, 4, 5, 6, 7]),\n",
       " tensor([3163,  582, 1068,  194,  201, 1074,  567, 3151]))"
      ]
     },
     "execution_count": 26,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "G_val.unique(return_counts=True) # results for ground-truth"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 82,
   "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": {},
   "source": [
    "# ERM Training"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "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": 47,
   "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": 48,
   "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": 49,
   "id": "baac0ac4-8b24-43a6-a2eb-1cb033f5e0b4",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "3f94fcde5d3b426bba2c027e149d0f89",
       "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": "4b6c328d9c19450aaa73730f37815070",
       "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.22544778883457184\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "a4c81b65f5394d0ca0715d97a174d7a2",
       "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.998614574674425,\n",
      " 1: 0.75,\n",
      " 2: 0.9900166389351082,\n",
      " 3: 0.49206349206349204,\n",
      " 4: 0.3793103448275862,\n",
      " 5: 0.9867109634551495,\n",
      " 6: 0.7802197802197802,\n",
      " 7: 0.9988603988603989}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "fc1c976377734738a7a811af7d71e28f",
       "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.9954712708746108,\n",
      " 1: 0.5754716981132075,\n",
      " 2: 0.958961474036851,\n",
      " 3: 0.3235294117647059,\n",
      " 4: 0.2077922077922078,\n",
      " 5: 0.9682804674457429,\n",
      " 6: 0.5736842105263158,\n",
      " 7: 0.9937641723356009}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "8068bd6d13c54b109fe54e09c75c1e81",
       "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.06603167206048965\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "b26866baad834867b20a019f67b3c036",
       "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.999168744804655,\n",
      " 1: 0.8604651162790697,\n",
      " 2: 0.9983361064891847,\n",
      " 3: 0.7936507936507936,\n",
      " 4: 0.7241379310344828,\n",
      " 5: 0.9958471760797342,\n",
      " 6: 0.9065934065934066,\n",
      " 7: 0.9985754985754985}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "9ee4d68a1ff04e3c88ac10f343a29d09",
       "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.996886498726295,\n",
      " 1: 0.6320754716981132,\n",
      " 2: 0.9698492462311558,\n",
      " 3: 0.4264705882352941,\n",
      " 4: 0.2597402597402597,\n",
      " 5: 0.9749582637729549,\n",
      " 6: 0.6052631578947368,\n",
      " 7: 0.9903628117913832}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "5c7117b41ef742fab446967b0f483b3c",
       "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.038588885217905045\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "a6afafcc931646608f3c2948b724b12f",
       "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.999722914934885,\n",
      " 1: 0.9651162790697675,\n",
      " 2: 1.0,\n",
      " 3: 0.8888888888888888,\n",
      " 4: 0.9310344827586207,\n",
      " 5: 0.9991694352159468,\n",
      " 6: 0.989010989010989,\n",
      " 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "8e4cc8643e3e4a169861568e46dba3aa",
       "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.9966034531559581,\n",
      " 1: 0.6650943396226415,\n",
      " 2: 0.9681742043551089,\n",
      " 3: 0.4264705882352941,\n",
      " 4: 0.36363636363636365,\n",
      " 5: 0.9749582637729549,\n",
      " 6: 0.6842105263157895,\n",
      " 7: 0.9926303854875284}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "2fa8786c8f3e48a2b7524bd8bc6e4373",
       "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.02328432723879814\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "89c4510489bd4785917387a591e3bc6c",
       "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.999722914934885,\n",
      " 1: 0.9709302325581395,\n",
      " 2: 1.0,\n",
      " 3: 0.9206349206349206,\n",
      " 4: 0.9827586206896551,\n",
      " 5: 1.0,\n",
      " 6: 1.0,\n",
      " 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "b08a54a3323b44ada5bc794628ccde08",
       "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.9974525898669686,\n",
      " 1: 0.6509433962264151,\n",
      " 2: 0.9731993299832495,\n",
      " 3: 0.39705882352941174,\n",
      " 4: 0.35064935064935066,\n",
      " 5: 0.9799666110183639,\n",
      " 6: 0.6578947368421053,\n",
      " 7: 0.9934807256235828}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "c40299d32e734ccabaebd5270c1eb466",
       "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.014482310973107815\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "0dddaecd43de4523aee9a2461e425120",
       "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.9883720930232558,\n",
      " 2: 1.0,\n",
      " 3: 0.9523809523809523,\n",
      " 4: 1.0,\n",
      " 5: 1.0,\n",
      " 6: 1.0,\n",
      " 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "afe8d0ec24924427bc6f86584db9b947",
       "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.9983017265779791,\n",
      " 1: 0.660377358490566,\n",
      " 2: 0.9748743718592965,\n",
      " 3: 0.4411764705882353,\n",
      " 4: 0.35064935064935066,\n",
      " 5: 0.9816360601001669,\n",
      " 6: 0.6421052631578947,\n",
      " 7: 0.9929138321995464}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6354952cc869414d8f64a44b79fbbeff",
       "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.0104364650323987\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "e95865e576024ae6bec8236147fb39d1",
       "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,\n",
      " 1: 0.9941860465116279,\n",
      " 2: 1.0,\n",
      " 3: 0.9682539682539683,\n",
      " 4: 1.0,\n",
      " 5: 1.0,\n",
      " 6: 1.0,\n",
      " 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "7f46468aafa04fb7affad8ee2e1aa2cf",
       "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.9983017265779791,\n",
      " 1: 0.6509433962264151,\n",
      " 2: 0.9731993299832495,\n",
      " 3: 0.4264705882352941,\n",
      " 4: 0.35064935064935066,\n",
      " 5: 0.9816360601001669,\n",
      " 6: 0.6684210526315789,\n",
      " 7: 0.9943310657596371}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "a45f9f850fea49c2a29fe70c207dc64f",
       "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.007714112754911184\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "177c0cbd29a541a2bea2226ac5b29bec",
       "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,\n",
      " 1: 0.9941860465116279,\n",
      " 2: 1.0,\n",
      " 3: 0.9682539682539683,\n",
      " 4: 1.0,\n",
      " 5: 1.0,\n",
      " 6: 1.0,\n",
      " 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "0dbaf9dff80a4fabb08e796bb09fa3b7",
       "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.9980186810076422,\n",
      " 1: 0.6650943396226415,\n",
      " 2: 0.9748743718592965,\n",
      " 3: 0.4852941176470588,\n",
      " 4: 0.33766233766233766,\n",
      " 5: 0.9824707846410684,\n",
      " 6: 0.6421052631578947,\n",
      " 7: 0.9946145124716553}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "bead9cb303ac444dab4bbfaa89ecd61a",
       "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.005826366599649191\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "801266d712af451694f53c09c0e7362f",
       "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,\n",
      " 1: 0.9941860465116279,\n",
      " 2: 1.0,\n",
      " 3: 0.9682539682539683,\n",
      " 4: 1.0,\n",
      " 5: 1.0,\n",
      " 6: 1.0,\n",
      " 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "539b4c38cbf94c31a2f31dc362860845",
       "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.9966034531559581,\n",
      " 1: 0.6367924528301887,\n",
      " 2: 0.9723618090452262,\n",
      " 3: 0.4117647058823529,\n",
      " 4: 0.44155844155844154,\n",
      " 5: 0.9858096828046744,\n",
      " 6: 0.7,\n",
      " 7: 0.9951814058956916}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "2ac726bfd0a14426a3b4ca116c76b9d6",
       "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.004400551784783602\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "ab7ec328b64e49db93ad1013371e015c",
       "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.9841269841269841, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "186d538be1794c72b3a593aec0920fda",
       "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.9985847721483159,\n",
      " 1: 0.7028301886792453,\n",
      " 2: 0.9807370184254607,\n",
      " 3: 0.5147058823529411,\n",
      " 4: 0.3246753246753247,\n",
      " 5: 0.9791318864774624,\n",
      " 6: 0.6157894736842106,\n",
      " 7: 0.9926303854875284}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "d5060cf7df244b3586b59f8fd7e6b305",
       "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.0034765321761369705\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "e8d497347bac4068b586c0205cfc8e52",
       "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.9841269841269841, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "2c50989f47ff49aba82b52ff88345b7d",
       "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.9988678177186527,\n",
      " 1: 0.6933962264150944,\n",
      " 2: 0.9782244556113903,\n",
      " 3: 0.4852941176470588,\n",
      " 4: 0.2857142857142857,\n",
      " 5: 0.9774624373956594,\n",
      " 6: 0.6263157894736842,\n",
      " 7: 0.9946145124716553}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6896e28efdf64c69a8eb56e60746273f",
       "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.0032778859604150057\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "c3bebe3a988b4e4b9f1eb6253f8d957e",
       "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.9841269841269841, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "29317d634b384571a704966cad4a042d",
       "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.9977356354373054,\n",
      " 1: 0.6367924528301887,\n",
      " 2: 0.97571189279732,\n",
      " 3: 0.4117647058823529,\n",
      " 4: 0.36363636363636365,\n",
      " 5: 0.9841402337228714,\n",
      " 6: 0.6842105263157895,\n",
      " 7: 0.9957482993197279}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "b28e935eed4b41d0bc9055bc51bf67d2",
       "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.00664189038798213\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "5c20da26352a4a20bebd74c606d26e4a",
       "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.9841269841269841, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "854f100e9fb54f8a8d4d6266fbf578e0",
       "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.9985847721483159,\n",
      " 1: 0.6745283018867925,\n",
      " 2: 0.9798994974874372,\n",
      " 3: 0.4852941176470588,\n",
      " 4: 0.35064935064935066,\n",
      " 5: 0.9774624373956594,\n",
      " 6: 0.631578947368421,\n",
      " 7: 0.9926303854875284}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "44246230cc1e41b2b48890b444f485ba",
       "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.004654097370803356\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "3ef30c1ddb0749da9d1002689535fbff",
       "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.9841269841269841, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "56e598707ae94749b496d702ed362da5",
       "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.9983017265779791,\n",
      " 1: 0.6933962264150944,\n",
      " 2: 0.9773869346733668,\n",
      " 3: 0.4411764705882353,\n",
      " 4: 0.36363636363636365,\n",
      " 5: 0.9774624373956594,\n",
      " 6: 0.6736842105263158,\n",
      " 7: 0.9940476190476191}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "2edb93f35002484cbfba2ba500d27264",
       "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.002107365755364299\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "54ae834838444d4f93bd13eeb81d5d87",
       "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: 0.9841269841269841, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "097ba1c244c14dc5bb5b7973fe4a9e3a",
       "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.9985847721483159,\n",
      " 1: 0.7028301886792453,\n",
      " 2: 0.9857621440536013,\n",
      " 3: 0.47058823529411764,\n",
      " 4: 0.3246753246753247,\n",
      " 5: 0.9766277128547579,\n",
      " 6: 0.6157894736842106,\n",
      " 7: 0.9934807256235828}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "30a07a54457946708991c1de87d989a2",
       "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.002173955552279949\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "f381a36bbfbc448c883e57d2924ec3ac",
       "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: 0.9841269841269841, 4: 1.0, 5: 1.0, 6: 1.0, 7: 1.0}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "cf12113a5e7d40deb5da2f04044ae906",
       "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.9977356354373054,\n",
      " 1: 0.6037735849056604,\n",
      " 2: 0.9706867671691792,\n",
      " 3: 0.35294117647058826,\n",
      " 4: 0.4155844155844156,\n",
      " 5: 0.986644407345576,\n",
      " 6: 0.7052631578947368,\n",
      " 7: 0.9957482993197279}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "016dadf41f384d52861d987f62e900d3",
       "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.001708241761662066\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "29a7afb52258490fbefede8dc036e9de",
       "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": "bcc45568cdbc46fda7f8541204ff1447",
       "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.996886498726295,\n",
      " 1: 0.6037735849056604,\n",
      " 2: 0.9698492462311558,\n",
      " 3: 0.3235294117647059,\n",
      " 4: 0.4155844155844156,\n",
      " 5: 0.988313856427379,\n",
      " 6: 0.7105263157894737,\n",
      " 7: 0.9957482993197279}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "52dc1b0a5525432184c4f96ae9bd0a28",
       "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.001634641783311963\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "04b8549d191141f5aa0f983afaf92b32",
       "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": "1e9af67c848c48b2906095477e862878",
       "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.9977356354373054,\n",
      " 1: 0.6415094339622641,\n",
      " 2: 0.9773869346733668,\n",
      " 3: 0.38235294117647056,\n",
      " 4: 0.36363636363636365,\n",
      " 5: 0.9849749582637729,\n",
      " 6: 0.6894736842105263,\n",
      " 7: 0.9948979591836735}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "116cf3a320324e3f81e6c99e16565e96",
       "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.0012947871582582593\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "ee3f421b6419404eaa82b57ec821810a",
       "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": "d4eb89d54c7349708ccdfa83f590461c",
       "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.9985847721483159,\n",
      " 1: 0.6650943396226415,\n",
      " 2: 0.981574539363484,\n",
      " 3: 0.45588235294117646,\n",
      " 4: 0.3246753246753247,\n",
      " 5: 0.9808013355592654,\n",
      " 6: 0.6263157894736842,\n",
      " 7: 0.9934807256235828}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "a2a370464b034d728031dba5e9b05808",
       "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.005401098635047674\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "e3799b89d5ed4f6485ed76f3a2eabe11",
       "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": "9b46f8ab78a44e06a55c5de6b2a48688",
       "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.9966034531559581,\n",
      " 1: 0.6084905660377359,\n",
      " 2: 0.9748743718592965,\n",
      " 3: 0.38235294117647056,\n",
      " 4: 0.4025974025974026,\n",
      " 5: 0.9874791318864775,\n",
      " 6: 0.7,\n",
      " 7: 0.9948979591836735}\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6200a05b369d41a6838dd2063fb4e1a9",
       "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.0065918597392737865\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "0fd2fa0bd79d48578c08852df2469b69",
       "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": "a96da57037784b33bc4dcdc7c477499e",
       "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.9980186810076422,\n",
      " 1: 0.6415094339622641,\n",
      " 2: 0.97571189279732,\n",
      " 3: 0.4264705882352941,\n",
      " 4: 0.33766233766233766,\n",
      " 5: 0.9799666110183639,\n",
      " 6: 0.631578947368421,\n",
      " 7: 0.9931972789115646}\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": 50,
   "id": "7e30d655-ba30-4d69-83f4-67de3c4e653a",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.save(model.state_dict(), 'dominoes_cmnist_fmnist_cifar_9575.model')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 95,
   "id": "66ee7ad1-dcbd-4ae9-a6bd-f506ee5d10cf",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([10000, 512]) torch.Size([10000, 512]) torch.Size([10000, 512]) torch.Size([10000, 512])\n"
     ]
    }
   ],
   "source": [
    "model.load_state_dict(torch.load('dominoes_cmnist_fmnist_cifar_9575.model'))\n",
    "model.fc = torch.nn.Identity()\n",
    "train_latent = model(X_tr.cuda())\n",
    "val_latent = model(X_val.cuda())\n",
    "ll_latent = model(X_ll.cuda())\n",
    "test_latent = model(X_te.cuda())\n",
    "print(train_latent.shape, val_latent.shape, ll_latent.shape, test_latent.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 52,
   "id": "de7356bc-9917-4094-a9ef-da7e18b75aa4",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.save(train_latent, 'train_features_9575.pt')\n",
    "torch.save(val_latent, 'val_features_9575.pt')\n",
    "torch.save(ll_latent, 'lastlayer_features_9575.pt')\n",
    "torch.save(test_latent, 'test_features_9575.pt')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2058db27-abef-4835-aa1f-aa9e6022d545",
   "metadata": {},
   "source": [
    "# Evaluate ERM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "id": "ece4f5e7-e34d-43a7-80d5-a9402d4c7678",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "f9ec63366a1a4c85b49dd9259a8c7601",
       "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_8575.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": 56,
   "id": "ebeb6a0a-a929-4a7c-9ae7-fcd4b39b4fa0",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "({0: 3609, 1: 172, 2: 1202, 4: 58, 5: 1204, 6: 182, 7: 3510, 3: 63},\n",
       " {0: 3609, 1: 172, 2: 1202, 4: 58, 5: 1204, 6: 182, 7: 3510, 3: 63})"
      ]
     },
     "execution_count": 56,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "correct_counts, total_counts"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "id": "44e3fe29-02b5-4792-8d83-6621ef31f59c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "ef0c1148b8934d629f6efea291215284",
       "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.9853829043533524,\n",
      " 1: 0.8933566433566433,\n",
      " 2: 0.9561487130600572,\n",
      " 3: 0.6857142857142857,\n",
      " 4: 0.6827956989247311,\n",
      " 5: 0.9397363465160076,\n",
      " 6: 0.8618881118881119,\n",
      " 7: 0.9836268149521161}\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_8575.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": "code",
   "execution_count": 40,
   "id": "bb990857-12e6-4aff-8993-240dee73ea0d",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{0: 3101, 1: 511, 2: 1003, 3: 120, 4: 127, 5: 998, 6: 493, 7: 3184}\n",
      "{0: 3147, 1: 572, 2: 1049, 3: 175, 4: 186, 5: 1062, 6: 572, 7: 3237}\n"
     ]
    }
   ],
   "source": [
    "print(correct_counts)\n",
    "print(total_counts)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c3656591-1d25-4e0a-bdc3-839a197bc99d",
   "metadata": {},
   "outputs": [],
   "source": [
    "# torch.save(model.state_dict(), 'dominoes_cmnist_fmnist_cifar.model')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "44833c94-a801-4297-b9f7-0a8fb072378e",
   "metadata": {},
   "source": [
    "# DFR"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "ab14828d-2db7-415c-98c6-6f87bae64f80",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "521a172cf147475fb2bc3b552de230aa",
       "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": "01c6f22730274fd0a744f9ac833c1e66",
       "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": "7a44483c11944a65bea5a86bee03d04a",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# if you have loaded image and want to extract features from the model, run this cell\n",
    "# if 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 (EIIL and name != 'val') or (not EIIL):\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 (EIIL and name != 'val') or (not EIIL):\n",
    "                # m = {0:0, 2:0, 1:1, 3:1, 4:2, 6:2, 5:3, 7:3} # Ignoring one spurious attribute\n",
    "                g = list(g.detach().cpu().numpy())\n",
    "                # all_g[name].append(np.array(list(map(lambda x: m[x], g))))\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 (EIIL and name != 'val') or (not EIIL):\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": 139,
   "id": "035143f9-9c8b-40b3-826f-7b3072191852",
   "metadata": {},
   "outputs": [],
   "source": [
    "all_embeddings, all_y, all_g = {}, {}, {}\n",
    "# if you have loaded the features, run this cell\n",
    "all_embeddings['train'], all_y['train'], all_g['train'] = X_tr.detach().cpu().numpy(), Y_tr.detach().cpu().numpy(), G_tr.detach().cpu().numpy()\n",
    "# all_embeddings['train'], all_y['train'], all_g['train'] = np.empty((1,512), dtype=np.int64), np.zeros((1,), dtype=np.int64), np.zeros((1,), dtype=np.int64)\n",
    "all_embeddings['lastlayer'], all_y['lastlayer'], all_g['lastlayer'] = X_ll.detach().cpu().numpy(), Y_ll.detach().cpu().numpy(), G_ll.detach().cpu().numpy()\n",
    "all_embeddings['val'], all_y['val'], all_g['val'] = X_val.detach().cpu().numpy(), Y_val.detach().cpu().numpy(), G_val.detach().cpu().numpy()\n",
    "all_embeddings['test'], all_y['test'], all_g['test'] = X_te.detach().cpu().numpy(), Y_te.detach().cpu().numpy(), G_te.detach().cpu().numpy()\n",
    "# all_g['val'] = all_g['val']//2\n",
    "# all_g['lastlayer'] = all_g['lastlayer']//2\n",
    "m = {0:0, 2:0, 1:1, 3:1, 4:2, 6:2, 5:3, 7:3} # Ignoring one spurious attribute\n",
    "all_g['val'] = np.array(list(map(lambda x: m[x], list(all_g['val']))))\n",
    "all_g['lastlayer'] = np.array(list(map(lambda x: m[x], list(all_g['lastlayer']))))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "0d8f66e6-36ec-42dd-9190-0941307e9deb",
   "metadata": {},
   "outputs": [],
   "source": [
    "all_g['val'] = torch.load('./val_eiil_envs_8575.pt').detach().cpu().numpy()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 73,
   "id": "eff8d9e2-713f-4ebc-9676-f76d55fd77cf",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "((10000, 512), (10000, 512), (10000, 512), (10000, 512))"
      ]
     },
     "execution_count": 73,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "all_embeddings['train'].shape, all_embeddings['lastlayer'].shape, all_embeddings['val'].shape, all_embeddings['train'].shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 140,
   "id": "177b5e69-4871-4b43-8ee8-f3a83538b769",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(array([4274,  730,  764, 4232]),\n",
       " array([4231,  776,  768, 4225]),\n",
       " array([3147,  572, 1049,  175,  186, 1062,  572, 3237]))"
      ]
     },
     "execution_count": 140,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.bincount(all_g['lastlayer']), np.bincount(all_g['val']), np.bincount(all_g['test'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "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.long(), 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": 25,
   "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": 130,
   "id": "5807a83a-3f6f-433d-a9a9-19b48f729015",
   "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]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 131,
   "id": "775b3d11-5b20-4fef-b7bf-0f37ef8db9ad",
   "metadata": {},
   "outputs": [],
   "source": [
    "def dfr_on_validation_tune(\n",
    "        all_embeddings, all_y, all_g, preprocess=False,\n",
    "        balance_val=True, add_train=False, num_retrains=3, 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_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_train = len(x_valtrain) if add_train else 0\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_val))\n",
    "        if preprocess:\n",
    "            scaler = StandardScaler()\n",
    "            x_train = scaler.fit_transform(x_train)\n",
    "            x_val = scaler.transform(x_val)\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",
    "    return best_hypers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 132,
   "id": "6d2fc426-f0a1-4246-9144-5bd41842c028",
   "metadata": {},
   "outputs": [],
   "source": [
    "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",
    "        # Added for the sake of fairness for DFR\n",
    "        # min_g = 50\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",
    "        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",
    "    x_val = all_embeddings[\"val\"]\n",
    "    y_val = all_y[\"val\"]\n",
    "    g_val = all_g[\"val\"]\n",
    "    # print(np.bincount(g_test))\n",
    "\n",
    "    if preprocess:\n",
    "        x_test = scaler.transform(x_test)\n",
    "        x_val = scaler.transform(x_val)\n",
    "    logreg = LogisticRegression(penalty=REG, C=c, solver=\"liblinear\",\n",
    "                                class_weight={0: w1, 1: w2})\n",
    "    n_classes = int(np.max(y_train) + 1)\n",
    "    # print(f'n_classes {n_classes}')\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_val = logreg.predict(x_val)\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",
    "    val_accs = [(preds_val == y_val)[g_val == g].mean()\n",
    "                 for g in range(n_groups)]\n",
    "    print('worst val acc:', np.min(val_accs))\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": 141,
   "id": "c8866347-754d-442e-a932-8240b603fc92",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Hypers: (0.1, 1.0, 1.0)\n",
      "worst val acc: 0.8815104166666666\n",
      "{'best_hypers': (0.1, 1.0, 1.0),\n",
      " 'test_accs': [0.969176993962504,\n",
      "               0.9493006993006993,\n",
      "               0.8894184938036225,\n",
      "               0.7542857142857143,\n",
      "               0.7526881720430108,\n",
      "               0.8615819209039548,\n",
      "               0.9265734265734266,\n",
      "               0.9721964782205746],\n",
      " 'test_mean_acc': 0.939,\n",
      " 'test_worst_acc': 0.7526881720430108,\n",
      " 'train_accs': [0.9616438356164384,\n",
      "                0.9273972602739726,\n",
      "                0.9424657534246575,\n",
      "                0.947945205479452]}\n",
      "\n"
     ]
    }
   ],
   "source": [
    "from pprint import pprint\n",
    "# classifier = None\n",
    "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",
    "pprint(dfr_lastlayer_results)\n",
    "print()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "decd45ee-3cad-4542-8c5f-3db6b3551e4c",
   "metadata": {},
   "source": [
    "# loss-based (EVaLS-GL)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 143,
   "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_8575.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": null,
   "id": "69d61cb8-a0a5-46ea-91ad-99d840b06d70",
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "best_worst = -1.\n",
    "for k in tqdm(range(10, 80)):\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) # , preprocess=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) # , preprocess=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",
    "    if best_worst < max(best_worst, dfr_lastlayer_results['test_worst_acc']):\n",
    "        print(f'\\n##############\\nfor k={k}:', dfr_lastlayer_results['test_worst_acc'], '\\n##############\\n')\n",
    "        best_worst = dfr_lastlayer_results['test_worst_acc']\n",
    "    # pprint(dfr_lastlayer_results)\n",
    "    print('-'*60)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "49bca678-e360-496f-af32-c7551be0ea7c",
   "metadata": {},
   "source": [
    "# EIIL"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 78,
   "id": "b3dd1669-9903-43b4-8248-82c79daeeef5",
   "metadata": {},
   "outputs": [],
   "source": [
    "all_g['val'] = torch.load('./val_eiil_envs_8575.pt').detach().cpu().numpy()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0a8149fa-11de-4220-aa35-9cdeb5494e6d",
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "best_worst = -1.\n",
    "for k in tqdm(range(10, 80)):\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) # , preprocess=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) # , preprocess=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",
    "    if best_worst < max(best_worst, dfr_lastlayer_results['test_worst_acc']):\n",
    "        print(f'\\n##############\\nfor k={k}:', dfr_lastlayer_results['test_worst_acc'], '\\n##############\\n')\n",
    "        best_worst = dfr_lastlayer_results['test_worst_acc']\n",
    "    # pprint(dfr_lastlayer_results)\n",
    "    print('-'*60)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 145,
   "id": "398904cb-98cb-4840-82c3-a04d4f06a212",
   "metadata": {},
   "outputs": [],
   "source": [
    "s = [85, 90, 95]\n",
    "erm = [68.2, 50.6, 36.84]\n",
    "dfr = [70.29, 60.2, 39.13]\n",
    "dfr_rev = [75.27, 71.5, 71.92]\n",
    "Evalsgl = [71.1111, 63.6, 47.37]\n",
    "Evalsgl_rev = [71.5, 65.5, 59.65]\n",
    "Evals = [72.2222, 67.1, 52.17]\n",
    "dfr_known = [79.43, 79.48, 73.68]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 149,
   "id": "35192da0-2aed-4813-b05d-70252256af37",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA1oAAAKmCAYAAABUhe9IAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/H5lhTAAAACXBIWXMAABcSAAAXEgFnn9JSAAEAAElEQVR4nOzdd3gTV9bA4Z8ky93GDRsbN3rvYGpCSSAFEkjv2fQlu+l9s6mbbL70Xjd1U0ll0ysphGrTewcbG4PBxsa4qsz3x1iDRsWWbbnJ530ePeBpupJmJB3de88xKIqiIIQQQgghhBDCb4xt3QAhhBBCCCGECDQSaAkhhBBCCCGEn0mgJYQQQgghhBB+JoGWEEIIIYQQQviZBFpCCCGEEEII4WcSaAkhhBBCCCGEn0mgJYQQQgghhBB+JoGWEEIIIYQQQviZBFpCCCGEEEII4WcSaAkhhBBCCCGEn0mgJYQQQgghhBB+JoGWEEIIIYQQQviZBFpCCCGEEEII4WcSaAkhhBBCCCGEn0mg1cHZbDaefvppRowYQUREBAaDAYPBwP/+97+2bppoxyorK7n33nsZMGAAYWFh2nmzZs2atm5aq/j999+1x+xvjuP+/vvvfj+2J5mZmRgMBt55551WuT8hmmL9+vWce+65JCcnExQUhMFgYPjw4W3dLL9p7nXfku9JgaCzvc/Jd7vAEdTWDWjPHnjgAR588EG35SEhISQkJDBy5EguvvhizjnnnDZ7c7zpppt48cUXAQgODiYpKQmA0NDQNmmP6BjOO+88vvnmGwDCwsK088ZsNrdls4QQAWj37t1MnDiR8vJyAOLi4jCbzSQkJLRxy1peaWkpzz77LKB+XsfExLRpe0TH0F6/2+3Zs0cLdh944IE2bUtHIYGWjxwnOUBZWRkFBQUUFBTw9ddf88477zB//nxCQkJatU3l5eW89tprADz++OPcdttt8muYaNCWLVu0IOvjjz/m3HPPbeMWiebo1asXoaGhdOnSpa2bIoRHr732GuXl5fTu3Zvff/+d7t27t3WTWk1paan2g+1ll13mNdAKDw+nX79+rdiyjqUzvc+15+92e/bs0c5nCbR8I4GWj/bv36/93263s3nzZm6++WZ+/vlnvv/+e+655x6eeOKJVm3Tli1bsFgsAFx77bXt5kIU7dv69esBiI+PlyArACxYsKCtmyBEvRzvObNnz+5UQVZjZGVlsWXLlrZuRrvVmd7n5LtdYJE5Wk1gNBoZNGgQX331Fb179wbUX+ysVmurtqOyslL7f2RkZKvet+i4HOeNnDNCiNYg7zlC+E6+2wUWCbSaITQ0lHPOOQdQu3pdf40qLy/n0UcfZfz48cTFxRESEkJaWhrnn38+S5cu9XjMPXv2aJMe9+zZw86dO7nmmmvo0aMHISEhZGZm8s4772AwGJgyZYq2n2Mf1+UOX3zxBbNmzSIpKUkb7ztr1izmz5/v9fFddtllGAwGLrvsMhRF4Y033mDSpEnEx8frJqVOmTIFg8HAAw88gNVq5ZlnnmHEiBFERkaSmJjInDlzWLt2rXbcyspKHn74YQYPHkxERATx8fGcd9557Ny502M77HY7CxYs4IYbbmDcuHGkpqYSHBxMfHw8kydP5tVXX9V+/Wno+Txw4AA33ngjPXr0IDQ0lKSkJM4///wGf0m02+188sknzJkzh+7duxMSEkLXrl0ZNWoUd955Jxs2bPC4X1POAV9VV1fz7LPPMmHCBGJjYwkNDSUjI4NLL73UY1KLBx54QHs9AXJzc3XnjWO5L5wnfhcXF3PLLbfQq1cvwsLCyMjI4LrrruPgwYPa9rm5uVx77bXa856ens6tt96qzdnwZvXq1Vx66aVkZGQQGhpKbGwsEyZM4Nlnn6Wmpqbefbds2cJFF11Et27dCA0NpWfPnlx//fUcOHDAp8fYEq/d4cOHue+++xg5ciTR0dEEBwfTrVs3hg4dyty5c5v0q219k8SdX6fy8nLuuece+vfvT1hYGPHx8cyaNYvly5c36bG4Xlvbt2/nsssuIzU1lZCQENLT05k7dy779u3zuL/r5P/Vq1dz0UUXkZqaitlsdnsfq62t5eWXX2bq1KkkJCRoz93s2bP5/vvvvbazqqqKJ598kvHjxxMbG4vZbKZr164MHDiQv/zlL3z++ede992wYQPXXHMNffr0ITw8nMjISIYOHco///lPDh065HEfx3XmaP+CBQuYOXMmXbt2JTQ0lAEDBvDggw9SXV1dz7MLxcXF/Otf/2Ls2LHExcURGhpKZmYmM2bM4JVXXqGsrMxvbfbVzp07ufbaa+nTpw9hYWFER0czcuRI/vWvf3HkyBG37R3npiNBxIMPPqh7z/E1cYTjMy8zMxOAP//8k9NOO43ExEQiIiIYMWIEb775pm6fb7/9lunTp9O1a1fCw8MZM2YMH3/8scfju57L3jQ2IcOUKVPo0aOH9nePHj28flb7IxlGfn4+N998M4MGDSIiIoKQkBBSUlIYNWoUN998Mzk5Obrtm3sNu74untT33Lru/9tvvzFnzhySk5MxmUy6z6SGnnubzcZbb73FtGnTSEhIICQkhO7du3POOefUe5758po6fxdyZbVa+c9//sOUKVNISEjAbDYTHx9Pv379OO+889zOy/o05btdU653i8XCV199xTXXXMPo0aNJTk4mODiYxMRETjrpJD766CMURXHbLzMzk6lTp3psm+vzU99z5vp4PZ0/vn7/dNizZw833XQTgwYNIjIykvDwcPr378+NN95IXl6e1zZs2bKFa665hr59+xIeHk5oaChpaWmMGzeOu+++2z+9zIrw6v7771cApb6n6aWXXtK2Wbx4sbZ89erVSmpqqrbOZDIpUVFR2t8Gg0F55JFH3I63e/dubZsPPvhAiYyMVAAlPDxciYiIUDIyMpR58+YpSUlJSmxsrLZtUlKSdjvjjDO049XU1CjnnXeetp3RaFRiY2MVo9GoLbvggguU2tpat7b85S9/UQDl0ksvVc466yy3/d9++21FURRl8uTJCqDcfffdygknnKAASnBwsBIREaHdR2RkpJKTk6McOnRIGTFihAIooaGhSlhYmLZNYmKikpubW+9z4jhWly5ddMuOO+44pbKyst59v/nmGyUxMVF7PkNCQrR10dHRypo1azy+xgcPHlSOP/543f3FxMRorw2gzJ49222/pp4DvsjPz1cGDx6sHctsNuueE6PRqDz//PO6fZ544gklKSlJiY6O1rZxPm9uuOEGn+/fcT///e9/tccYERGhBAcHa+sGDBigHD58WMnOzlbi4+O15zkoKEjbZuLEiYrVavV4H08//bRiMBi0bbt06aKYzWbt76FDhyr79u3zuO/333+ve30jIyOV0NBQBVCSk5OVt956q95ruzmvnWOb3377Tbd87969Snp6utu1aDKZtGWTJ0/2+TVwyMjIUADtevTUlg8//FDp3bu3dt2Fh4dr64KDg5Uff/yx0ffrfG3NmzdPe34iIyN113VcXJyycuVKt/1/++03bZvPPvtMe22jo6OV0NBQ3XOxZ88eZdCgQbrn3/U9YO7cuW73ceTIEWXYsGG6/WJiYnTnYEZGhsfH99hjj+neJ8PDw3Xnd3JysrJq1Sq3/RyfG5MnT1Yef/xxxWAwaPfrfD5PnTrV67n/448/6t7fg4KClPj4eN35P3/+fL+12Rcff/yx7pqKiorS/Z2WlqZs2rRJt8/o0aOVpKQkrd0RERG69xznz8z6vP3229pr9frrrytGo9HjOXDXXXcpiqIo9913n3aNuW7zyiuvuB3f+VzevXu313b4cq05X/dnnHGGkpCQoK1LSEjw+lntfD00xZo1a3TnjMlkUmJjY3Xn3F/+8hevj7sp17Dz6+JNfc+t8/7PPvus1lbHe71ze+t77ktLS5UpU6boHrvr9Xbbbbd5bF99x3VwfBdyff6sVqsyffp03fnVpUsX3XXRmNezMd/tFKXp17vzueZ4z3X+fAOUc845R7HZbLr9Ro8e7bVtrt8hvD1nzuo7f3z9/qkoivL+++/rnvOQkBDd+RsVFeXxM+6nn37S7Wc2m5WYmBjd83D//fd7bb+vJNCqhy+B1u23365ts3nzZkVRFGXfvn3aF/ozzzxTWbFihRbIHDhwQLn33nu1D3rXD0vnN6XIyEhl7NixSk5OjrZ+69at2v99eWO+9dZbtS8Y9957r3L48GFFURSlpKREufvuu7X977zzTrd9HSd6ZGSkEhQUpDz55JNKWVmZoiiKUl5ern3JdQRaMTExSnx8vPLpp58qtbW1it1uV7Kzs5WePXsqgDJhwgTljDPOUDIzM5Uff/xRsdlsis1mU3755Rela9euCqBcdNFFbu3Yu3evctFFFylfffWVUlxcrC0vLy9X3n77bSUlJUUBlJtvvtltX+fnMzY2Vpk4caL2fFosFuXnn39WkpOTFVCDNVcWi0WZOHGidvE+9thjSlFRkba+oKBAee2115R//OMfuv2acw40xGq1KmPHjtXe1N9//32lpqZGURRF2blzpzJr1iztNf/uu+/c9vflw7Ehjuc0JiZGGT58uLJs2TJFURSltrZW+eijj7Qv8tddd52SkZGhTJs2TdmwYYOiKIpSVVWlvPDCC1qA8frrr7sd/+uvv9buY/bs2cquXbsURVF/OHj33Xe1D4UJEya4fVndu3evFkwOHTpUWb58uaIoimKz2ZTvv/9eSU1N1b2Zumrua+fpC5eiKMqVV16pAEpmZqbyyy+/aO22Wq3Knj17lFdeecXjddgQX778xcbGKgMHDlR+/fVXxWazaddmv379tHPB9UO1Ic7XVpcuXXTPtd1uV3788UctsExPT1eOHDmi29/5/SsyMlI59dRTtfdQRVGUbdu2KYqiKEePHlX69++vAMqUKVOU33//XamurlYURf2C9fTTT2s/ejz77LO6+3jooYe0L4qff/65tp/NZlMKCgqUd999V7n66qvdHtsbb7yhtevf//63UlhYqCiK+lqtWLFCmTZtmgIoqampSnl5uW5fx+dGTEyMYjQalX/84x/KwYMHFUVRlLKyMi0IAJQ333zT7b5XrVql/SgwaNAg5bvvvtPOP8f933rrrcovv/zitzY3ZOXKlVqwNHHiRGXdunXa8/jVV19p76G9evXyeGzHZ0RTv7Q43rMcXyRvuOEG7X24uLhY+6wyGo3KY489pphMJuXhhx9WSktLFUVRr+mTTz5ZATXYcyx3aKlAqzHHbm6g5fiRc+TIkcrSpUsVu92uKIr6nrlt2zblySefVB5//HGvbWvKNeyvQCs0NFQxmUzKZZddpuTl5SmKop63O3bs0Lat77l3fBEPDg5Wnn/+eaWiokJRFEUpLCxUrrjiCu3+PQXZzQm03nvvPa39b7zxhnbu2+125cCBA8oXX3yhnH322V6P640v50Jzrvfly5crf/3rX5Wff/5Z+06nKOq19Nxzz2mfn88991yT2qYo/gu0Gvr++dNPPylGo1EJCgpS7rjjDmX37t2K3W5X7Ha7smXLFuWcc87RgknXH/J79eqlAMqMGTOU9evXa8urqqqUDRs2KA8++GC954WvJNCqR0OBVllZmfYlPy4uTvui4riwL7zwQq/HfvrppxVAGTZsmG6585tSRkZGvR+IDZ3w+fn52hdC10DA4ZZbbtEiedfeAceJDrj1jjhzfIgCyp9//um2fsGCBdr6sLAwZfv27W7bvPnmm9p6T71r9cnJydE+QKuqqnTrnJ/P/v37e+z1+uqrr7Rt9u7dq1vneDMzGAzKt99+63ObmnMONGTevHlaez39SmOxWLRAbPDgwW7r/RloJSUlKYcOHXJbf++992rbDBo0SPuC6+ySSy5RAOWEE05wWzdgwAAt+PX0q7/za/bpp5/q1l177bUKoMTHxysHDhxw23f9+vW6ngFXzX3tvH3hcjymDz/80Otxm8KXL39du3b1+FysW7dO22bRokWNul/na8vb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ydTTd6ry8nIefvhhr8fz9Xx2PK85OTkeg63NmzdrPwA01dSpU7Us1jfffHODPU/Onw8NbevLNeOrjveJ3gE88sgjpKSkcOjQIcaPH897771HeXm5tv7gwYN8/vnnnHHGGS36odS9e3dtEu2jjz7K/fffr10YpaWl3HvvvTzxxBOAmjo5OTm5xdrSVJGRkdqvFrfccgu//vqr9oGyYcMGTj31VFasWNGoieONcckllzBp0iQUReGss87iiSee0H0Y7du3j2eeecYtFXhLngNnnXWWllXt3HPP5cMPP9TGgO/evZuzzjqLpUuXAuoE6Y7I8cPAn3/+ydlnn83u3bsBdSjpBx98oD1nEyZM0IaAOfzjH/8gKiqKQ4cOMX36dO2XLkVR+OmnnzjllFPqzcTVUq9dRkYG//jHP1i2bJnuA3rHjh1cdNFFVFZWasMYOhqLxcL06dO1LJ2OnrmTTjqJmpoa0tLSmDt3bpOOfeuttzJkyBCqq6uZOnUqL774ojYsFtT3su+//55LL72U4447Trfv2LFjueGGG/j99991Q/H27dvH9ddfz44dOwB0pRtiYmJ49tlnATWd9cyZM1m+fLn2vmO329m8eTNPPfUUgwYN4ptvvmnS4/LmueeeIzQ0lO3btzNx4kR++OEH7fq22Wzk5OQwd+5c3a/OLd3mhx9+GLPZzI4dOzjppJO0X3jtdjvfffcdp556KlarlV69evlUCqQ9clzLb731Fi+//LI2cX7v3r1cddVVfPzxxz5n8HMWExOj9cq+/fbbbglQ/CE/P58+ffrw8MMPs3r1at19rFu3josvvhhQA0pvmVybcg2npqZqJV1uueUWfvnlF+2HsZUrV3LiiSe2aDImUDPfOn4kvv7663nxxRe1HwP279/P1Vdfzaeffgqoad5DQ0N1+zte982bN3PNNddo7y1HjhzhmWeeYe7cuV6Hws6ZM4crrriC77//Xhd0lJSU8PDDD2s9wI506f7SnOt9wIABWk/+FVdcoSuHsXTpUqZMmVJvVs6+fftqP1C98cYbXoO80047jcjISCwWC+eeey5bt24F1PPsyy+/5MQTT2z297agoCBeffVVgoKCWLRoEccffzwLFizQzYnbtWsXr776KmPGjOHll1/Wli9ZsoShQ4fyzDPPsHnzZu25UxSFJUuWcO211wLqOT506NBmtVMKFtejoYLF9dm0aZPSt29fbX+j0ajExcVpBRMdN9eClb4WN1QU3wrH1dTUKOeee66uHbGxsYrRaNSWXXDBBR4LgfpScE5RfCtG6UtRQEd7XAs+rlixQve8hYSEKFFRUQqgBAUFKe+++67X4/ujEOXBgweV4447TjuOwWBQYmJilMjISG3Z7Nmz3fZr6jngi/z8fGXQoEHaMYKDg7UCho778lTVXVH8W7DY9bVy8OV5b+j8ffrppxWDwaBtExMToxXOBJQhQ4YoBQUFHvf95ptvlJCQEG3bqKgoraBncnKy8tZbb9V738157bw9N877Oa7D0NBQ3XnV2MKSitK0IqrOmlpM1vk1njdvnnZNRkZGKuHh4brXLScnx23/xhRoLSgoUMaNG+d2DUZHR+ue1969e+v2czw3zvu4voY333yzx/t85ZVXdOdbSEiIEh8fryt2DSjvv/++bj/H58bkyZO9Pp6GHvuPP/6odOnSRdvGbDa73ff8+fP91mZfzJs3T3fs6Oho3fmblpambNq0yeO+/ipYXN97li/Pe32faeXl5crAgQN116jjPdVsNisfffRRk6+1hx56SPeapKWlKRkZGcp5552nbdOcgsXO1yKgmEwmJS4uTvd6BQcHuxV3b+41rCiKsnr1at11GBoaql1jSUlJyrfffuv1s6Axn0X1PfelpaXaOeb4XhAbG6v7/Ljtttu8HvuSSy7RPX8xMTHad6Trr7/e63njfJ+Oa8L1Penss89WbDZbg4/Pma/nQlOv96+//lorMA4o4eHh2usdERGh/PLLL/Wez1deeaVu3/T0dCUjI0O59dZbddu98cYbutcgKipKa++4ceOUF1980evr7+v3T0VRlPnz52vnrvP7pfN3ANAXMnd+jp33cX5eoqOjlYULFzZ4/w2RHq0WMmDAANatW8drr73GjBkzSEhI4MiRIyiKQu/evTnnnHP4z3/+wyeffNKi7QgODubjjz/ms88+45RTTiE+Pp7y8nLi4+M55ZRT+OKLL/jwww/b9Zj6UaNGkZ2dzbnnnktCQgJ2u52oqCjOPfdclixZwiWXXNKi95+QkMDvv//O+++/zymnnELXrl2pqKggPDycUaNGcdddd/HII4+47deS50D37t1ZsWIFTz/9NOPGjSMsLIzKykrS0tK45JJLWLlyJTfccIM/Hn6bufnmm1mxYgUXX3wxaWlpVFZWEhYWxrhx43jmmWfIycnxWjtj5syZrFq1ivPPP5/ExERqa2tJSkriuuuuY/Xq1VqRXm9a4rX76aef+Mc//sFxxx1HWlqa9ot57969ufzyy8nJyfFYaLUjGDt2LCtWrODSSy+lS5cuWK1WunfvztVXX8369eu1eRRNlZKSwqJFi/joo484/fTTSU5OprKyktraWjIzMznttNN49tlnWbhwoW6/efPm8eCDD3LCCSfQo0cPamtrsVgsZGRkcN5557FgwQKefvppj/c5d+5ctm7dym233cawYcMICQmhtLSUyMhIRo8ezfXXX8/PP//cIqMSZsyYwfbt2/nnP//JiBEjCAsLo6Kigu7du3PSSSfx2muveax71JJtPu+889i4cSN//etf6dWrFzU1NQQFBTF8+HAefPBBNmzYwIABA/zx8NtEZGQkixYt4pZbbqFHjx4EBQVpQ+qXLl2qDVFrirvvvpvnnnuO0aNHYzabyc/PJzc312/FYLt3785XX33FzTffzLhx40hOTubo0aMEBQUxcOBA/v73v7Nhwwavhdqh6dfw8OHDWb58ufZea7fbSUhI4O9//ztr1qzRhoG3pC5durBgwQLefPNNpkyZQlRUFEePHqVbt26cddZZ/Pbbb9roHU/eeecdnnvuOYYPH05YWBh2u52JEyfyySef8Pzzz3vd74UXXuCxxx7j1FNPpU+fPiiKQlVVFSkpKZx++ul8/vnnfPrppy02HLyp1/usWbNYuHAhM2fOJCYmBqvVSkJCApdffjkrV67Uin9789JLL/HAAw9o9Tnz8vLIzc11G3p65ZVX8u233zJt2jSio6OxWq307duXRx99lD/++MNvI5HmzJnDjh07uP/++8nKyiIyMpLS0lJCQkIYNmwYV111FfPnz+f222/X9hkzZgyffPIJ1157LaNGjdI+40NDQxk+fDh33HEHmzdvdhsl0RQGRWnj3I1CCCE6nD179mgB6+7du8nMzGzbBgkhGkWuYSFanvRoCSGEEEIIIYSfSaAlhBBCCCGEEH4mgZYQQgghhBBC+JkEWkIIIYQQQgjhZ5IMQwghhBBCCCH8THq0hBBCCCGEEMLPJNASQgghhBBCCD+TQEsIIYQQQggh/EwCLSGEEEIIIYTwMwm0hBBCCCGEEMLPgtq6AYGmW7duVFRUkJ6e3tZNEUIIIYQQotPLy8sjIiKC/fv3t+r9So+Wn1VUVGCxWNq6GUIIIYQQQgjAYrFQUVHR6vcrPVp+5ujJ2rhxYxu3RAghhBBCCDFo0KA2uV/p0RJCCCGEEEIIP5NASwghhBBCCCH8TAItIYQQQgghhPAzCbSEEEIIIYQQws8k0BJCCCGEEEIIP5NASwghhBBCCCH8TAItIYQQQgghhPAzCbSEEEIIIYQQws8k0BJCCCGEEEIIP5NASwghhBBCCCH8rEMHWjk5OZx77rmkpKRgNpuJiYnhuOOO4+2330ZRFLftbTYbzzzzDEOGDCEsLIyuXbty7rnnsnnz5jZovRBCCCGEECJQBbV1A5rq888/57zzzsNmszFy5EiOO+44Dh48yJ9//smiRYv45Zdf+OCDD7Tt7XY755xzDvPnzycmJoaZM2dy6NAhPvvsM7799lt+++03srKy2vARCSGEEEIIIQJFh+zRslqt/O1vf8Nms/HBBx+wcuVKPv74Y3799VfWrVtHXFwcH374Ib/99pu2z1tvvcX8+fPp06cPW7Zs4bPPPuP333/n008/pbKykosuugir1dqGj0oIIYQQQggRKDpkoLVlyxaKioro168fF154oW7dgAEDuPjiiwF1aKHD008/DcDjjz9OUlKStvyss87i9NNPZ8eOHXz55Zet0HohhBBCCCFEoOuQgVZISIhP28XHxwOwe/duNm/eTFhYGDNnznTb7uyzzwbg66+/9l8jhRBCCCGEEJ1Whwy0evbsSa9evdi6dSsffvihbt3mzZt5//33iY2N5YwzzgBg7dq1AAwePBiz2ex2vJEjRwKwbt26Fm65EEIIIYQQojPokMkwTCYT//3vf5k1axYXXXQRTz31FH369KGoqIg///yTgQMH8s477xAXFwdAXl4eAKmpqR6P51iem5vrcxsGDRrkcfnOnTvp1atXYx6OEEIIIYQQIsB0yEALYOLEifzxxx+cccYZrFq1ilWrVgEQHBzM9OnT6dmzp7bt0aNHAQgPD/d4rIiICADKy8tbuNVCCCGEEEKIzqDDBlofffQRl19+OePGjeOjjz5i0KBB7Nu3jyeffJKnnnqK3377jSVLlvg8n6uxNm7c6HG5t54uIYQQQgghROfRIedobd++nb/85S8kJCTwzTffkJWVRUREBH369OG1115j1qxZrFq1irfeeguAyMhIACorKz0er6KiAoCo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lrpdu3bRsWNHypYtS968eQkJCeHkyZNcvnwZJycnfv311yTveyXyMho1ahQQe8EWERGR9EA9WiIpZMiQIZjN5hhJFkQusd25c2cePnzIpk2b2LhxIxEREXTq1ImDBw9Ss2bNVIpYJG2ZMGECQUFBzz3vT0REJDWoR0vkBStUqFCiN0BOa7Zv357aIYiIiIikC+rREhERERERSWZKtERERERERJKZEi0REREREZFkpkQrFTy5opzZbE7FSEREXi4RERGW/9bqnSIikpqUaKUCk8mEnZ0dAAEBAakcjYjIy+PRo0cA2NvbK9ESEZFUpVUHU4mrqyv379+3bFzr7OyMlZXyXhGRpDIMg5CQEB4/fsyDBw8A8PDwSOWoRETkVadEK5VkypSJgIAAgoODuXHjRmqHIyLy0nB3d8fNzS21wxARkVecEq1UYm1tTZ48ebh//z6PHz8mNDQ0tUMSEUm3rK2tcXZ2xtXVFVdXVw0bFBGRVKdEKxVZW1vj6emJp6cnhmFgGEZqhyQiku6YTCYlViIikuYo0Uoj9KAgIiIiIvLy0OoLIiIiIiIiyUyJloiIiIiISDJToiUiIiIiIpLMlGiJiIiIiIgkMyVaIiIiIiIiyUyJloiIiIiISDIzGdq8KVm5uroSFhZGgQIFUjsUEREREZFX3sWLF7G1teXx48cv9Lrq0Upmzs7O2NrapnYYKerixYtcvHgxtcMQkVeIPndE5EXT587Lw9bWFmdn5xd+XfVoSZIVL14cgFOnTqVyJCLyqtDnjoi8aPrckeelHi0REREREZFkpkRLREREREQkmSnREhERERERSWZKtERERERERJKZEi0REREREZFkplUHRUREREREkpl6tERERERERJKZEi0REREREZFkpkRLREREREQkmSnREhERERERSWZKtERERERERJKZEi0REREREZFkpkRLREREREQkmSnRkhgOHjxI69atyZEjB7a2tri7u1OzZk1mzZrFf7dc8/HxwWQyxfs1ZcqUVLoLEUlPzpw5Q4cOHciePTv29vbky5ePjz76iHv37sXbZs2aNdSuXZsMGTKQIUMGvL29Wbdu3QuMWkTSs6R87syePTvB5522bdumwh1IemCT2gFI2rF8+XLatGlDREQE5cqVo2bNmty9e5ddu3axe/duNm/ezIIFC2K1a9iwIdmyZYtVXqRIkRcR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",
      "text/plain": [
       "<Figure size 960x720 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from matplotlib import pyplot as plt\n",
    "# plt.figure(figsize=(8, 9))\n",
    "plt.rcParams['figure.dpi'] = 150\n",
    "plt.plot(s, Evals, label='EVaLS', color=plt.cm.jet(0.99))\n",
    "plt.plot(s, Evalsgl, label='EVaLS-GL', color=plt.cm.jet(0.77))\n",
    "plt.plot(s, Evalsgl_rev, label='EVaLS-GL (rev)', color=plt.cm.jet(0.63))\n",
    "plt.plot(s, dfr, label='DFR', color=plt.cm.jet(0.5))\n",
    "plt.plot(s, dfr_rev, label='DFR (rev)', color=plt.cm.jet(0.25))\n",
    "plt.plot(s, erm, label='ERM', color=plt.cm.jet(0.1))\n",
    "plt.plot(s, dfr_known, '--', color='gray', label='DFR (group labels known)')\n",
    "plt.xticks(s)\n",
    "plt.xlim([84, 97])\n",
    "plt.xlabel('Spuriousity')\n",
    "plt.ylabel('Worst Group Acc.')\n",
    "plt.title('Performance of models in presence of multi-spurious features')\n",
    "plt.annotate('', xy=(94.85, 39.2), xytext=(94.85, 73.88),\n",
    "                 arrowprops=dict(arrowstyle='->', color='red'))\n",
    "plt.text(94.85, (39.13 + 73.68)/2 + 7.5, f'-{73.68-39.13:.2f}%', ha='right', va='center', color='red')\n",
    "plt.annotate('', xy=(95, 52.27), xytext=(95, 36.84),\n",
    "                 arrowprops=dict(arrowstyle='->', color='green'))\n",
    "plt.text(95, (36.84 + 52.17)/2, f'+{52.17-36.84:.2f}%', ha='left', va='center', color='green')\n",
    "plt.legend()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6ca8f4b2-4e9f-4be8-822e-73a7cb02f198",
   "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
}
