{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "f35d8a3b",
   "metadata": {},
   "source": [
    "## compute the FIDs for the generated samples in the MNIST experiment"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "f7fd2617",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2026-01-04 23:03:51.850715: I tensorflow/core/platform/cpu_feature_guard.cc:210] This TensorFlow binary is optimized to use available CPU instructions in performance-critical operations.\n",
      "To enable the following instructions: AVX2 AVX512F FMA, in other operations, rebuild TensorFlow with the appropriate compiler flags.\n",
      "/data/zizgpu04/not-backed-up/howard/miniforge3/envs/log_smoothing/lib/python3.11/site-packages/keras/src/export/tf2onnx_lib.py:8: FutureWarning: In the future `np.object` will be defined as the corresponding NumPy scalar.\n",
      "  if not hasattr(np, \"object\"):\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "os.environ['CUDA_VISIBLE_DEVICES']='2'\n",
    "\n",
    "from typing import Tuple\n",
    "import numpy as np\n",
    "import tensorflow as tf\n",
    "from scipy import linalg as ln\n",
    "import gc\n",
    "\n",
    "\n",
    "def prepare_32x32_grayscale(imgs):\n",
    "    \"\"\"\n",
    "    Make to range [0,255] with 3 channels, from 32x32 grayscale inputs.\n",
    "    Accepts shapes:\n",
    "      [N, 32, 32], [N,32,32,1], [N,32,32,3] (if already RGB)\n",
    "    Returns:\n",
    "      imgs_rgb: np.float32 array shape [N, 32, 32, 3] with values in [0,255]\n",
    "    \"\"\"\n",
    "    imgs = imgs.astype(np.float32)\n",
    "    if imgs.ndim == 3:\n",
    "        imgs = imgs[..., None]  # [N,H,W,1]\n",
    "\n",
    "    if imgs.shape[-1] == 1:\n",
    "        imgs = np.repeat(imgs, 3, axis=-1)  # [N,H,W,3]\n",
    "    elif imgs.shape[-1] == 3:\n",
    "        pass\n",
    "    else:\n",
    "        raise ValueError(f\"Unsupported channel dimension: {imgs.shape}\")\n",
    "\n",
    "    imgs = imgs * 255.0\n",
    "\n",
    "    return imgs.astype(np.float32)\n",
    "\n",
    "\n",
    "def build_inception_model():\n",
    "    \"\"\"Return Keras InceptionV3 model producing 2048-d pooled features.\"\"\"\n",
    "    model = tf.keras.applications.InceptionV3(include_top=False, weights=\"imagenet\", pooling=\"avg\")\n",
    "    return model\n",
    "\n",
    "\n",
    "\n",
    "def get_activations(model: tf.keras.Model, imgs_32: np.ndarray, batch_size: int = 50) -> np.ndarray:\n",
    "    \"\"\"\n",
    "    Run imgs_32 (N,32,32,3) through the model.\n",
    "    Resizes to 299x299 per-batch, applies preprocess_input, returns activations (N,2048).\n",
    "    \"\"\"\n",
    "    n = imgs_32.shape[0]\n",
    "    acts = []\n",
    "    for start in range(0, n, batch_size):\n",
    "        end = min(n, start + batch_size)\n",
    "        batch = imgs_32[start:end]  # numpy array float32 0..255\n",
    "\n",
    "        batch_tf = tf.convert_to_tensor(batch)  # dtype float32\n",
    "        # Resize to Inception expected 299x299\n",
    "        batch_tf = tf.image.resize(batch_tf, (299, 299))\n",
    "\n",
    "        # Preprocess: maps [0,255] -> [-1,1]\n",
    "        batch_tf = tf.keras.applications.inception_v3.preprocess_input(batch_tf)\n",
    "        preds = model(batch_tf, training=False)\n",
    "        acts.append(preds.numpy())\n",
    "    acts = np.concatenate(acts, axis=0)\n",
    "    return acts  # shape (N,2048)\n",
    "\n",
    "\n",
    "def compute_statistics(features: np.ndarray):\n",
    "    mu = np.mean(features, axis=0)\n",
    "    sigma = np.cov(features, rowvar=False)\n",
    "    return mu, sigma\n",
    "\n",
    "\n",
    "def calculate_frechet_distance(mu1, sigma1, mu2, sigma2, eps=1e-6):\n",
    "    \"\"\"Numpy implementation of the Frechet Distance.\n",
    "    The Frechet distance between two multivariate Gaussians X_1 ~ N(mu_1, C_1)\n",
    "    and X_2 ~ N(mu_2, C_2) is\n",
    "            d^2 = ||mu_1 - mu_2||^2 + Tr(C_1 + C_2 - 2*sqrt(C_1*C_2)).\n",
    "    Stable version by Dougal J. Sutherland.\n",
    "    Params:\n",
    "    -- mu1   : Numpy array containing the activations of a layer of the\n",
    "               inception net (like returned by the function 'get_predictions')\n",
    "               for generated samples.\n",
    "    -- mu2   : The sample mean over activations, precalculated on an\n",
    "               representative data set.\n",
    "    -- sigma1: The covariance matrix over activations for generated samples.\n",
    "    -- sigma2: The covariance matrix over activations, precalculated on an\n",
    "               representative data set.\n",
    "    Returns:\n",
    "    --   : The Frechet Distance.\n",
    "    \"\"\"\n",
    "\n",
    "    mu1 = np.atleast_1d(mu1)\n",
    "    mu2 = np.atleast_1d(mu2)\n",
    "\n",
    "    sigma1 = np.atleast_2d(sigma1)\n",
    "    sigma2 = np.atleast_2d(sigma2)\n",
    "\n",
    "    assert mu1.shape == mu2.shape, \\\n",
    "        'Training and test mean vectors have different lengths'\n",
    "    assert sigma1.shape == sigma2.shape, \\\n",
    "        'Training and test covariances have different dimensions'\n",
    "\n",
    "    diff = mu1 - mu2\n",
    "\n",
    "    # Product might be almost singular\n",
    "    covmean, _ = ln.sqrtm(sigma1.dot(sigma2), disp=False)\n",
    "    if not np.isfinite(covmean).all():\n",
    "        msg = ('fid calculation produces singular product; '\n",
    "               'adding %s to diagonal of cov estimates') % eps\n",
    "        print(msg)\n",
    "        offset = np.eye(sigma1.shape[0]) * eps\n",
    "        covmean = ln.sqrtm((sigma1 + offset).dot(sigma2 + offset))\n",
    "\n",
    "    # Numerical error might give slight imaginary component\n",
    "    if np.iscomplexobj(covmean):\n",
    "        # if not np.allclose(np.diagonal(covmean).imag, 0, atol=1e-3):\n",
    "        #     m = np.max(np.abs(covmean.imag))\n",
    "        #     raise ValueError('Imaginary component {}'.format(m))\n",
    "        covmean = covmean.real\n",
    "\n",
    "    tr_covmean = np.trace(covmean)\n",
    "\n",
    "    return (diff.dot(diff) + np.trace(sigma1) +\n",
    "            np.trace(sigma2) - 2 * tr_covmean)\n",
    "\n",
    "\n",
    "def compute_fid_from_32x32(imgs1, imgs2, batch_size, verbose=False):\n",
    "\n",
    "    processed_imgs1 = prepare_32x32_grayscale(imgs1)\n",
    "    processed_imgs2 = prepare_32x32_grayscale(imgs2)\n",
    "\n",
    "    if verbose:\n",
    "        print(f\"Set A: {imgs1.shape}, Set B: {imgs2.shape}\")\n",
    "        print(\"Building InceptionV3 model (may download weights)...\")\n",
    "\n",
    "    model = build_inception_model()\n",
    "\n",
    "    if verbose:\n",
    "        print(\"Computing activations for set A...\")\n",
    "    features1 = get_activations(model, processed_imgs1, batch_size=batch_size)\n",
    "    if verbose:\n",
    "        print(\"Computing activations for set B...\")\n",
    "    features2 = get_activations(model, processed_imgs2, batch_size=batch_size)\n",
    "\n",
    "    if verbose:\n",
    "        print(\"Activations shapes:\", features1.shape, features2.shape)\n",
    "\n",
    "    mu1, sigma1 = compute_statistics(features1)\n",
    "    mu2, sigma2 = compute_statistics(features2)\n",
    "\n",
    "    fid = calculate_frechet_distance(mu1, sigma1, mu2, sigma2)\n",
    "    return fid"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "25f179be",
   "metadata": {},
   "source": [
    "## load results for the digit"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "48b20521",
   "metadata": {},
   "outputs": [],
   "source": [
    "digit = 4\n",
    "all_gaussian_samples = np.load(f'samples/digit_{digit}/gaussian_samples_mnist_digit_{digit}.npy')\n",
    "all_gaussian_dists_to_data = np.load(f'samples/digit_{digit}/gaussian_dists_to_data_digit_{digit}.npy')\n",
    "all_gaussian_dists_to_manifold = np.load(f'samples/digit_{digit}/gaussian_dists_to_manifold_digit_{digit}.npy')\n",
    "\n",
    "all_adapted_samples = np.load(f'samples/digit_{digit}/adapted_samples_mnist_digit_{digit}.npy')\n",
    "all_adapted_dists_to_data = np.load(f'samples/digit_{digit}/adapted_dists_to_data_digit_{digit}.npy')\n",
    "all_adapted_dists_to_manifold = np.load(f'samples/digit_{digit}/adapted_dists_to_manifold_digit_{digit}.npy')\n",
    "\n",
    "all_KDE_samples = np.load(f'samples/digit_{digit}/KDE_samples_mnist_digit_{digit}.npy')\n",
    "all_KDE_dists_to_data = np.load(f'samples/digit_{digit}/KDE_dists_to_data_digit_{digit}.npy')\n",
    "all_KDE_dists_to_manifold = np.load(f'samples/digit_{digit}/KDE_dists_to_manifold_digit_{digit}.npy')\n",
    "\n",
    "fid_images = np.load(f'samples/digit_{digit}/fid_images.npy')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6ad4b345",
   "metadata": {},
   "source": [
    "## compute FIDs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "0ae09175",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "WARNING: All log messages before absl::InitializeLog() is called are written to STDERR\n",
      "I0000 00:00:1767567835.790555 2992865 gpu_device.cc:2020] Created device /job:localhost/replica:0/task:0/device:GPU:0 with 9615 MB memory:  -> device: 0, name: NVIDIA GeForce RTX 2080 Ti, pci bus id: 0000:3d:00.0, compute capability: 7.5\n",
      "2026-01-04 23:03:58.584174: I external/local_xla/xla/stream_executor/cuda/cuda_dnn.cc:473] Loaded cuDNN version 91701\n",
      "/tmp/ipykernel_2992865/2158393331.py:104: DeprecationWarning: The `disp` argument is deprecated and will be removed in SciPy 1.18.0.\n",
      "  covmean, _ = ln.sqrtm(sigma1.dot(sigma2), disp=False)\n",
      "/tmp/ipykernel_2992865/2158393331.py:104: LinAlgWarning: Matrix is singular. The result might be inaccurate or the array might not have a square root.\n",
      "  covmean, _ = ln.sqrtm(sigma1.dot(sigma2), disp=False)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Gaussian sample 0 FID: 9.49876545856408\n",
      "Gaussian sample 1 FID: 23.938848973037523\n",
      "Gaussian sample 2 FID: 65.46106626948384\n",
      "Gaussian sample 3 FID: 85.74146732829695\n",
      "Gaussian sample 4 FID: 90.96696628114664\n",
      "Gaussian sample 5 FID: 91.7539786217366\n",
      "Gaussian sample 6 FID: 90.15391127766763\n",
      "Gaussian sample 7 FID: 87.10448817493364\n"
     ]
    }
   ],
   "source": [
    "all_gaussian_fids = []\n",
    "for i in range(all_gaussian_samples.shape[0]):\n",
    "    samples = all_gaussian_samples[i]\n",
    "    fid = compute_fid_from_32x32(samples, fid_images, batch_size=10, verbose=False)\n",
    "    all_gaussian_fids.append(fid)\n",
    "    print(f\"Gaussian sample {i} FID: {fid}\")\n",
    "\n",
    "    tf.keras.backend.clear_session()   # releases TF graphs/vars\n",
    "    gc.collect()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "a6f66b06",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_2992865/2158393331.py:104: DeprecationWarning: The `disp` argument is deprecated and will be removed in SciPy 1.18.0.\n",
      "  covmean, _ = ln.sqrtm(sigma1.dot(sigma2), disp=False)\n",
      "/tmp/ipykernel_2992865/2158393331.py:104: LinAlgWarning: Matrix is singular. The result might be inaccurate or the array might not have a square root.\n",
      "  covmean, _ = ln.sqrtm(sigma1.dot(sigma2), disp=False)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Adapted sample 0 FID: 9.498700171351814\n",
      "Adapted sample 1 FID: 11.698663308149378\n",
      "Adapted sample 2 FID: 15.690553831298935\n",
      "Adapted sample 3 FID: 16.885456441244543\n",
      "Adapted sample 4 FID: 17.372686837196404\n",
      "Adapted sample 5 FID: 18.494801134364046\n",
      "Adapted sample 6 FID: 21.346096014110877\n",
      "Adapted sample 7 FID: 23.10902081185384\n"
     ]
    }
   ],
   "source": [
    "all_adapted_fids = []\n",
    "for i in range(all_adapted_samples.shape[0]):\n",
    "    samples = all_adapted_samples[i]\n",
    "    fid = compute_fid_from_32x32(samples, fid_images, batch_size=10, verbose=False)\n",
    "    all_adapted_fids.append(fid)\n",
    "    print(f\"Adapted sample {i} FID: {fid}\")\n",
    "\n",
    "    tf.keras.backend.clear_session()   # releases TF graphs/vars\n",
    "    gc.collect()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3e37c348",
   "metadata": {},
   "source": [
    "## make plots"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "c4932d86",
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "from src.plotting import add_arrows\n",
    "\n",
    "# plot formatting\n",
    "import seaborn as sns\n",
    "from matplotlib.lines import Line2D\n",
    "sns.set_theme()  # sets a nicer global style for plots\n",
    "plt.rcParams.update({'font.size': 18})\n",
    "plt.rcParams.update({'font.family': 'serif'})"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "9f39f0e6",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 300x300 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "avg_gaussian_data_distances = np.mean(all_gaussian_dists_to_data, axis=1)\n",
    "avg_gaussian_manifold_distances = np.mean(all_gaussian_dists_to_manifold, axis=1)\n",
    "\n",
    "avg_adapted_data_distances = np.mean(all_adapted_dists_to_data, axis=1)\n",
    "avg_adapted_manifold_distances = np.mean(all_adapted_dists_to_manifold, axis=1)\n",
    "\n",
    "# Plot scatter plot with arrows\n",
    "plt.figure(figsize=(3,3))\n",
    "\n",
    "# Plot and extract line objects\n",
    "\n",
    "pal = sns.color_palette()   # nice, colorblind-safe seaborn palette\n",
    "orange, green = pal[1], pal[2]  \n",
    "\n",
    "line_gaussian = plt.plot(avg_gaussian_data_distances, all_gaussian_fids,  marker='o', ms=10, label='Gaussian', color=orange)[0]\n",
    "line_adapted = plt.plot(avg_adapted_data_distances, all_adapted_fids,  marker='^', ms=10, label='Adapted', color=green)[0]\n",
    "\n",
    "# Add arrows to the lines\n",
    "add_arrows(line_gaussian, segment_index=1, color='tab:orange')\n",
    "add_arrows(line_adapted, segment_index=2, color='tab:green')\n",
    "\n",
    "plt.xlabel(\"Distance to data\")\n",
    "plt.ylabel(\"FID\")\n",
    "plt.grid(True)\n",
    "\n",
    "# plt.plot([min_val, max_val], [min_val, max_val], 'k--', label='y = x')\n",
    "plt.legend(loc='upper left')\n",
    "\n",
    "fig_dir = f'figures/mnist/digit_{digit}'\n",
    "os.makedirs(fig_dir, exist_ok=True)\n",
    "\n",
    "plt.savefig(f\"{fig_dir}/mnist_fid_{digit}s.pdf\", bbox_inches='tight')\n",
    "\n",
    "plt.show()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "log_smoothing",
   "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.11.14"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
