{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Dogs-vs-not-dogs evaluation in Imagenet\n",
    "\n",
    "\n",
    "The two subsets of the imagenet dataset are from this [Dropbox link](https://www.dropbox.com/sh/yrfmp7hwa2w9gxz/AAATMrfWNLctPq1vnRa3mtZPa?dl=0). We use the `dogs50A-train.tar.gz` and `non-dogs-val.tar.gz` subsets. Extract these into the respective folders and change the paths accordingly int his script."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import torch"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from PIL import Image\n",
    "class ImagePathDataset(torch.utils.data.Dataset):\n",
    "    def __init__(self, files, transforms=None):\n",
    "        self.files = files\n",
    "        self.transforms = transforms\n",
    "\n",
    "    def __len__(self):\n",
    "        return len(self.files)\n",
    "\n",
    "    def __getitem__(self, i):\n",
    "        path = self.files[i]\n",
    "        img = Image.open(path).convert('RGB')\n",
    "        if self.transforms is not None:\n",
    "            img = self.transforms(img)\n",
    "        return img\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import glob\n",
    "\n",
    "def list_file_types(directory):\n",
    "    # ensure the directory path ends with a separator\n",
    "    if not directory.endswith(os.path.sep):\n",
    "        directory += os.path.sep\n",
    "    \n",
    "    # recursive glob pattern to match all files\n",
    "    glob_pattern = directory + '**/*.*'\n",
    "    \n",
    "    # set to store unique file extensions\n",
    "    file_types = set()\n",
    "    \n",
    "    # recursively search for all files with an extension\n",
    "    for file in glob.glob(glob_pattern, recursive=True):\n",
    "        # extract file extension and add to the set\n",
    "        _, ext = os.path.splitext(file)\n",
    "        if ext:  # check if extension is not empty\n",
    "            file_types.add(ext.lower())  # add extension in lowercase to ensure uniqueness\n",
    "    \n",
    "    return file_types\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from torchvision import transforms\n",
    "\n",
    "dog_folder = '/mnt/dogs50A-train'\n",
    "dog_files = []\n",
    "for root, dirs, files in os.walk(dog_folder):\n",
    "    for file in files:\n",
    "        if file.endswith('.png') or file.endswith('.jpg') or file.endswith('.jpeg'):\n",
    "            dog_files.append(os.path.join(root, file))\n",
    "print(len(dog_files))\n",
    "\n",
    "nondog_folder = '/mnt/non-dogs-val'\n",
    "nondog_files = []\n",
    "for root, dirs, files in os.walk(nondog_folder):\n",
    "    for file in files:\n",
    "        if file.endswith('.png') or file.endswith('.jpg') or file.endswith('.jpeg'):\n",
    "            nondog_files.append(os.path.join(root, file))\n",
    "print(len(nondog_files))\n",
    "\n",
    "transform_inception_with_crop = transforms.Compose([\n",
    "    # transform,\n",
    "    transforms.Resize(64),  # Inception models expect a 299x299 input size\n",
    "    transforms.CenterCrop(64),\n",
    "    transforms.ToTensor(),\n",
    "    transforms.Normalize(mean=[0.485, 0.456, 0.406], std=[0.229, 0.224, 0.225]),\n",
    "])\n",
    "\n",
    "dog_dataset = ImagePathDataset(dog_files, transforms=transform_inception_with_crop)\n",
    "nondog_dataset = ImagePathDataset(nondog_files, transforms=transform_inception_with_crop)\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.imshow(dog_dataset[0].permute(1, 2, 0))\n",
    "plt.show()\n",
    "plt.imshow(nondog_dataset[0].permute(1, 2, 0))\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from labproject.embeddings import FIDEmbeddingNet\n",
    "net = FIDEmbeddingNet(device='cuda')\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "dataloader = torch.utils.data.DataLoader(dog_dataset, batch_size=100, shuffle=False, num_workers=4)\n",
    "dog_embeddings = net.get_embeddings(dataloader)\n",
    "\n",
    "dataloader = torch.utils.data.DataLoader(nondog_dataset, batch_size=100, shuffle=False, num_workers=4)\n",
    "nondog_embeddings = net.get_embeddings(dataloader)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "dog_embeddings.shape, nondog_embeddings.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.save(dog_embeddings, '/mnt/dog_embeddings.pt')\n",
    "torch.save(nondog_embeddings, '/mnt/nondog_embeddings.pt')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "dog_embeddings = torch.load('/mnt/dog_embeddings.pt')\n",
    "nondog_embeddings = torch.load('/mnt/nondog_embeddings.pt')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from labproject.metrics.gaussian_squared_wasserstein import gaussian_squared_w2_distance\n",
    "\n",
    "print('dog nondog', gaussian_squared_w2_distance(dog_embeddings, nondog_embeddings))\n",
    "print('dog dog', gaussian_squared_w2_distance(dog_embeddings[:3004], dog_embeddings[3004:]))\n",
    "print('nondog nondog', gaussian_squared_w2_distance(nondog_embeddings[:2405], nondog_embeddings[2405:]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from labproject.metrics.MMD_torch import compute_rbf_mmd\n",
    "print('dog nondog', compute_rbf_mmd(dog_embeddings, nondog_embeddings, 64))\n",
    "print('dog dog', compute_rbf_mmd(dog_embeddings[:3004], dog_embeddings[3004:], 64))\n",
    "print('nondog nondog', compute_rbf_mmd(nondog_embeddings[:2405], nondog_embeddings[2405:], 64))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from labproject.metrics.c2st import c2st_nn\n",
    "print('dog nondog', c2st_nn(dog_embeddings, nondog_embeddings))\n",
    "dog_embeddings_shuffled = dog_embeddings[torch.randperm(dog_embeddings.size(0))]\n",
    "print('dog dog', c2st_nn(dog_embeddings_shuffled[:3004], dog_embeddings_shuffled[3004:]))\n",
    "nondog_embeddings_shuffled = nondog_embeddings[torch.randperm(nondog_embeddings.size(0))]\n",
    "print('nondog nondog', c2st_nn(nondog_embeddings_shuffled[:2405], nondog_embeddings_shuffled[2405:]))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from labproject.metrics.sliced_wasserstein import sliced_wasserstein_distance\n",
    "print('dog nondog', sliced_wasserstein_distance(dog_embeddings[:4809], nondog_embeddings, 5000))\n",
    "print('dog dog', sliced_wasserstein_distance(dog_embeddings[:3004], dog_embeddings[3004:], 5000))\n",
    "print('nondog nondog', sliced_wasserstein_distance(nondog_embeddings[:2404], nondog_embeddings[2405:], 5000))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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zJgx+te//qUTJ04oPDxczz77rN555x05Of33R661a9dqwoQJxvFrr71W6nVat26tESNG6IcffpAkjRw5UsuWLSvwtALpWngwbNgwZWVlSZKefPJJdezYsSxfGoBqrI6Pj+rWrWvvNlAOCAgAAACsxMXFRZ988okGDBig7OxszZkzR+Hh4Ro6dKg8PDwUGRmp8PBwY3+CJ598UoMGDbphnrCwMI0dO9Y4vtl+Bh9++KH27t2r48ePKyEhQd27d9fAgQPVtWtX5eTkaOvWrVq7dq1xbkhIiN577z0bfeUAgKqAgAAAAMCK+vTpo++//16jR49WcnKyjh07prfffvuGuqeeekrz5s0r8zp+fn767bff9PDDD2vTpk3Ky8vTr7/+ql9//fWG2lGjRmn+/Plyd3cv83oAgKqPgAAAAMDKhgwZoqioKM2dO1fLly9XdHS00tPTVa9ePfXo0UMTJ05U7969LV6nUaNG+v3337Vs2TJ9/fXXioyM1Pnz51WjRg01bNhQd9xxhx5//HFj80QAAIpCQAAAAGAD9erV06xZszRr1qxSnztmzBiNGTOmRLUmk0n33Xef7rvvvlKvAwBAfjzmEAAAAAAAEBAAAAAAAAACAgAAAAAAIAICAAAAAAAgAgIAAAAAACACAgAAAAAAIAICAAAAAAAgAgIAAAAAACACAgAAAAAAIAICAAAAAAAgAgIAAAAAACACAgAAAAAAIAICAAAAAAAgAgIAAAAAACACAgAAAAAAIAICAAAAAAAgAgIAAAAAACACAquKi4vTyy+/rA4dOsjT01Nubm5q0aKFxo4dq4iICKuudezYMT3zzDNq06aN3N3d5enpqVatWmnSpEk6cOBAqeb66aefNGTIEDVs2FDOzs6qV6+eevTooX//+9+6evWqVfsGAAAAAFRMTvZuoKr46aefNHr0aF25cqXA68ePH9fx48cVFhamp59+WnPmzJGjo6NFa82fP1+TJ09WZmZmgdejoqIUFRWl+fPna/r06ZoyZUqR86SkpOihhx7SihUrCryekJCghIQEbd26Vf/+97/17bffqlu3bhb1DAAAAACo2AgIrGD9+vUaMWKEcnNzJUnBwcG655575OLiom3btmnVqlUym82aN2+eJOnDDz8s81oLFy7UX/7yF+O4R48e6tOnjyTpt99+0+bNm5WTk6OpU6eqZs2aevHF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HnvsMb3zzjuSpNWrVxe5ZwEAAACAqomAACijpKQkq53v6OioI0eOWNpSmd1+++3G+OTJk3brAwAAAID9EBAAZZSTk2OMvby8NGXKlFKd36JFC2u3VGa1a9c2xleuXLFjJwAAAADshYAAKKP8v1SbzWa98MILduzm5tLS0uTi4lLk5omSdPXqVWPs7e1t464AAAAAVERsUgiUUdOmTeXkdC1ju3Llis6ePVvsOVevXlVWVpatW1OXLl3k4eEhd3d3bdu2rdj6/fv3G+M2bdrYsjUAAAAAFRQBAVBGbm5u6tSpk3G8ZMmSYs95+OGH5erqqiZNmujzzz+3WW+1atUyNlAMDw8vstZsNmvx4sXGcd++fW3WFwAAAICKi4AAsMDo0aON8VtvvaWLFy8WWrtx40b9+uuvysvL09mzZ9W1a1eb9TVixAhjPG/evCL7mjt3rrFBYr169QqcCwAAAKD6ICAALDBmzBg1a9ZMkhQXF6e+ffvqxIkTN9Rt3bpVDz74oHE8YsQItW3b1mZ9jR492nj0YVJSku69917Fx8cXqDGbzfroo4/03HPPGa+9++67qlmzps36AgAAAFBxsUkhYAFXV1d9//336tWrl65evaq9e/eqTZs2uueee9S+fXtlZGRo165dWrNmjXFOQECA5s2bZ9O+3Nzc9PXXX6tv375KT09XZGSkWrRooeHDh+uWW27RpUuXFB4eroMHDxrnPPfccxo1apRN+wIAAABQcREQABbq2LGjIiIi9OCDD+qPP/5QVlaWli5dqqVLl95Q2717d3333XeqW7euzfvq2bOn1qxZo4cfflhnzpxRSkqK/vOf/9xQV7NmTU2bNk2vvPKKzXsCAAAAUHEREABW0K5dOx04cEDfffedlixZoh07dighIUF5eXmqV6+eunTpolGjRmno0KHFPnLQmnr27KmoqCiFhYXpp59+0r59+5SUlKRatWqpadOmGjhwoMaNG2fcJgEAAACg+jKZzWazvZsAAACortauXavPP/9cW7du1blz5+Tq6qrGjRurf//+Gj16tIKDg0s0z4YNG/Txxx8rIiJC58+fl6enp/z9/XX//ffrqaeekp+fn42/EgBAZUdAAAAAYAepqal6/PHHi3xMrqOjoyZPnqy33npLTk43v/AzJydHEydO1GeffVboPHXr1tUXX3yhe++91+K+AQBVFwEB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nVp3tWK3pWc2mxUSEqKIiAh98cUXGjNmTJH1X375pR5//PESzz969GiFhYVZ1iQAoEoiICijByavtHcLAABUS/uO7pIkudZ01eA7hlk83z09h2rjrnXKzs3WgeP77B4QzJ49WxERESWu37Nnjw27AQBUJwQEAACgUolPjJNMUoumrVWzRk2L5/Nw81SLpq10+OQBxSactkKHZffFF19oypQppTonf0Dw1ltvydHRscj6du3alak3AEDVR0AAAAAqlSupyZKkenXqW23OBr6NdPjkASVduWi1OUvDbDbr9ddf1/Tp00u9v9HevXslSQEBAfrHP/5hg+4AANUFAQEAAKhUsrKzJEluLu5Wm9PD1VOSlJmZYbU5Syo+Pl7jxo1TeHh4qc89efKkLl++LEnq1KmTlTsDAFQ3DvZuoLIym/Ns+h8AAHBzubm5kqQaTjWsNuf1ubJzs602Z3HS09P1+uuvq0WLFkY44OnpqV69Sv6oxfy3F9x2221W7xEAUL1wBUEZLZxlyw9hk0ZPY8MhAACqssWLF2vq1KnG8W233aaFCxfq+++/1++//16iOQgIAADWREBQRqW9P7A0eHwiAADVR506dTRlyhRNmjRJTk5O+v7770t87u7du41xp06dlJeXp507d+rAgQNKS0tTvXr11KNHDzVu3NjqfcfExJS4NjY21urrAwCsj4DAAqZ8v8nbMjAAAABVT7169TR79mxNnDhR3t7eZZrj+hUE9evX17fffqu3335bZ86cuaGub9++eueddxQcHGxJywUEBgZabS5UT3l5eUpMTLR3G1WSj4+PHBy4mxylR0BgJXUbtVVgu4EKbDtAbp6+9m4HAABUcAMHDtTAgQPLfH58fLzi4+ON8TPPPFNo7Zo1a9S1a1d99dVXGjZsWJnXBKwpMTFRfn5+9m6jSkpISJCvL7+ToPQICMqo55CZijm0UudORiovL1eJcYeVGHdYO9e8p3pNOyqo3UD5t+mrmq5e9m4VAIAq6dS5aEXs32iVuWLiTlplnvKUf/8B6doVCf/4xz80dOhQNWzYUElJSVq7dq1mzZqlY8eOKT09XQ8//LA2btyorl272qlrAEBFRkBQRi06DFGLDkOUkXZZpw6vUfShlTp/arfMebmKj9mp86d2aVv4bDUM6qbAtgPUtPVdquHsZu+2AQCoMnYc3qodh7fauw27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",
      "text/plain": [
       "<Figure size 1062.99x944.882 with 4 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "## plotting\n",
    "\n",
    "from labproject.plotting import generate_palette\n",
    "color_dict = {\"SW\": \"#cc241d\",\n",
    "              \"C2ST\": \"#458588\",\n",
    "              \"MMD\": \"#eebd35\",\n",
    "              \"FID\": \"#8ec07c\"}\n",
    "\n",
    "\n",
    "from omegaconf import OmegaConf\n",
    "from labproject.experiments import *\n",
    "from labproject.plotting import cm2inch\n",
    "import time\n",
    "import matplotlib.pyplot as plt\n",
    "import sklearn\n",
    "\n",
    "\n",
    "col_dark = {}\n",
    "col_light = {}\n",
    "for metric, color in color_dict.items():\n",
    "    col_dark[metric] = generate_palette(color, saturation='dark')[2]\n",
    "    col_light[metric] = generate_palette(color, saturation='light')[-1]\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "# precomputed metrics\n",
    "intra_data = {metric:None for metric in [\"C2ST\", \"MMD\", \"FID\", \"SW\"]}\n",
    "intra_data[\"FID\"] = {\"dog_nodog\": 20.39, \"dog_dog\": 15.06, \"nodog_nodog\": 22.31}\n",
    "intra_data[\"C2ST\"] = {\"dog_nodog\": 0.92, \"dog_dog\": 0.50, \"nodog_nodog\": 0.49}\n",
    "intra_data[\"MMD\"] = {\"dog_dog\": 1.6689e-06, \"dog_nodog\": 3.0637e-05, \"nodog_nodog\": 2.1458e-06}\n",
    "intra_data[\"SW\"] = {\"dog_dog\": 0.0072, \"dog_nodog\": 0.0223, \"nodog_nodog\": 0.0077}\n",
    "\n",
    "col_dark = {}\n",
    "col_light = {}\n",
    "col_lightest = {}\n",
    "for metric, color in color_dict.items():\n",
    "    col_dark[metric] = generate_palette(color, n_colors=6, saturation='dark')[3]\n",
    "    col_light[metric] = generate_palette(color, n_colors=6, saturation='light')[-1]\n",
    "    col_lightest[metric] = generate_palette(color, n_colors=6,saturation='light')[0]\n",
    "\n",
    "# make comparison plots\n",
    "fig, axes = plt.subplots(2, 2, figsize=cm2inch((9, 8)), dpi=300, sharex='col')\n",
    "\n",
    "for ax in axes.flatten():\n",
    "    # move spines outward\n",
    "    ax.spines['bottom'].set_position(('outward', 4))\n",
    "    ax.spines['left'].set_position(('outward', 4))\n",
    "    ax.locator_params(nbins=3)\n",
    "\n",
    "# set fontsize to 10\n",
    "plt.rcParams.update({'font.size': 10})\n",
    "for i, (ax, metric) in enumerate(zip(axes.flatten(), color_dict.keys())):\n",
    "    # move spines outward\n",
    "    ax.spines['bottom'].set_position(('outward', 4))\n",
    "    ax.spines['left'].set_position(('outward', 4))\n",
    "    ax.locator_params(nbins=3)\n",
    "\n",
    "    intraclass1 = intra_data[metric][\"dog_dog\"]\n",
    "    intraclass2 = intra_data[metric][\"nodog_nodog\"]\n",
    "    interclass = intra_data[metric][\"dog_nodog\"]\n",
    "\n",
    "    # barplot with different shadings\n",
    "    ax.bar([0, 1, 2], [interclass, intraclass1, intraclass2], color=[col_dark[metric], col_light[metric], col_lightest[metric]], edgecolor='black')\n",
    "\n",
    "    ax.set_ylabel(metric, color=col_dark[metric])\n",
    "    if metric == 'C2ST':\n",
    "        ax.set_ylim([0.4,1])\n",
    "        ax.set_yticks([0.5,1])\n",
    "    if metric == 'FID':\n",
    "        ax.set_ylim([-0.5,30])\n",
    "        ax.set_yticks([0,15,30])\n",
    "    if metric == 'MMD':\n",
    "        ax.set_ylim([-5e-7,3.2e-5])\n",
    "        ax.set_yticks([0,1e-5,2e-5,3e-5])\n",
    "    if metric == 'SW':\n",
    "        ax.set_ylim([-0.001,0.03])\n",
    "        ax.set_yticks([0,0.01,0.03])\n",
    "\n",
    "    if i >=2:\n",
    "        ax.set_xticks([0, 1, 2])\n",
    "        ax.set_xticklabels(['D vs.\\n~D', 'D vs.\\nD', '~D vs.\\n~D'])\n",
    "    else:\n",
    "        ax.set_xticks([])\n",
    "\n",
    "    \n",
    "fig.tight_layout()\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "labproject",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.18"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
