{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "4ce6f328",
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2\n",
    "import sys\n",
    "sys.path.append(\"../../src\")\n",
    "import feature.scrna_dataset as scrna_dataset\n",
    "import model.sdes as sdes\n",
    "import model.generate as generate\n",
    "import model.scrna_ae as scrna_ae\n",
    "import model.util as model_util\n",
    "import analysis.fid as fid\n",
    "import torch\n",
    "import numpy as np\n",
    "import scipy.stats\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.font_manager as font_manager\n",
    "import os\n",
    "import h5py\n",
    "import json"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "647469ac",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Plotting defaults\n",
    "font_list = font_manager.findSystemFonts(fontpaths=[\"/home/anon/modules/fonts\"])\n",
    "for font in font_list:\n",
    "    font_manager.fontManager.addfont(font)\n",
    "plot_params = {\n",
    "    \"figure.titlesize\": 22,\n",
    "    \"axes.titlesize\": 22,\n",
    "    \"axes.labelsize\": 20,\n",
    "    \"legend.fontsize\": 18,\n",
    "    \"font.size\": 13,\n",
    "    \"xtick.labelsize\": 16,\n",
    "    \"ytick.labelsize\": 16,\n",
    "    \"font.family\": \"Roboto\",\n",
    "    \"font.weight\": \"bold\",\n",
    "    \"svg.fonttype\": \"none\"\n",
    "}\n",
    "plt.rcParams.update(plot_params)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "df22d86f",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define device\n",
    "if torch.cuda.is_available():\n",
    "    DEVICE = \"cuda\"\n",
    "else:\n",
    "    DEVICE = \"cpu\""
   ]
  },
  {
   "cell_type": "markdown",
   "id": "806f0a68",
   "metadata": {},
   "source": [
    "### Define constants and paths"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "767236f1",
   "metadata": {},
   "outputs": [],
   "source": [
    "models_base_path = \"/data/anon/branched_diffusion/models/trained_models/\"\n",
    "\n",
    "branched_model_path = os.path.join(models_base_path, \"scrna_covid_flu_continuous_branched_9classes_latent_d200/3/last_ckpt.pth\")\n",
    "label_guided_model_path = os.path.join(models_base_path, \"scrna_covid_flu_continuous_labelguided_9classes_latent_d200/3/last_ckpt.pth\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "93fc3767",
   "metadata": {},
   "outputs": [],
   "source": [
    "def import_classes_branch_points(json_path):\n",
    "    with open(json_path, \"r\") as f:\n",
    "        d = json.load(f)\n",
    "        return d[\"classes\"], \\\n",
    "            [(tuple(trip[0]), trip[1], trip[2]) for trip in d[\"branches\"]]\n",
    "    \n",
    "classes, branch_defs = import_classes_branch_points(\n",
    "    \"/home/anon/branched_diffusion/data/config/classes_branch_points/scrna_covid_flu/redset.json\"\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "c574c65f",
   "metadata": {},
   "outputs": [],
   "source": [
    "latent_space = True\n",
    "latent_dim = 200"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "2efe8729",
   "metadata": {},
   "outputs": [],
   "source": [
    "data_file = \"/data/anon/branched_diffusion/data/scrna/covid_flu/processed/covid_flu_processed_reduced_genes.h5\"\n",
    "autoencoder_path = \"/data/anon/branched_diffusion/models/trained_models/scrna_vaes/covid_flu/covid_flu_processed_reduced_genes_ldvae_d%d/\" % latent_dim"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "79325406",
   "metadata": {},
   "outputs": [],
   "source": [
    "out_path = \"/home/anon/branched_diffusion/figures/scrna_covid_flu_sample_quality\"\n",
    "\n",
    "os.makedirs(out_path, exist_ok=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4f058fa4",
   "metadata": {},
   "source": [
    "### Create data loader"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "8ef8d5c7",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[rank: 0] Global seed set to 0\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[34mINFO    \u001b[0m File                                                                                                      \n",
      "         \u001b[35m/data/anon/branched_diffusion/models/trained_models/scrna_vaes/covid_flu/covid_flu_proc\u001b[0m\n",
      "         \u001b[35messed_reduced_genes_ldvae_d200/\u001b[0m\u001b[95mmodel.pt\u001b[0m already downloaded                                                \n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "No GPU/TPU found, falling back to CPU. (Set TF_CPP_MIN_LOG_LEVEL=0 and rerun for more info.)\n",
      "/home/anon/miniconda3/envs/scanpy/lib/python3.9/site-packages/scvi/data/fields/_layer_field.py:91: UserWarning: adata.X does not contain unnormalized count data. Are you sure this is what you want?\n",
      "  warnings.warn(\n"
     ]
    }
   ],
   "source": [
    "dataset = scrna_dataset.SingleCellDataset(data_file, autoencoder_path=(autoencoder_path if latent_space else None))\n",
    "\n",
    "# Limit classes\n",
    "inds = np.isin(dataset.cell_cluster, classes)\n",
    "dataset.data = dataset.data[inds]\n",
    "dataset.cell_cluster = dataset.cell_cluster[inds]\n",
    "\n",
    "data_loader = torch.utils.data.DataLoader(dataset, batch_size=128, shuffle=True, num_workers=0)\n",
    "input_shape = next(iter(data_loader))[0].shape[1:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "49b23ce6",
   "metadata": {},
   "outputs": [],
   "source": [
    "sde = sdes.VariancePreservingSDE(0.1, 5, input_shape)\n",
    "t_limit = 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "51fbc829",
   "metadata": {},
   "outputs": [],
   "source": [
    "# TODO: this is currently rather inefficient; a decision-tree-style structure\n",
    "# would be better\n",
    "\n",
    "def class_time_to_branch(c, t):\n",
    "    \"\"\"\n",
    "    Given a class and a time (both scalars), return the\n",
    "    corresponding branch index.\n",
    "    \"\"\"\n",
    "    for i, branch_def in enumerate(branch_defs):\n",
    "        if c in branch_def[0] and t >= branch_def[1] and t <= branch_def[2]:\n",
    "            return i\n",
    "    raise ValueError(\"Undefined class and time\")\n",
    "        \n",
    "def class_time_to_branch_tensor(c, t):\n",
    "    \"\"\"\n",
    "    Given tensors of classes and a times, return the\n",
    "    corresponding branch indices as a tensor.\n",
    "    \"\"\"\n",
    "    return torch.tensor([\n",
    "        class_time_to_branch(c_i, t_i) for c_i, t_i in zip(c, t)\n",
    "    ], device=DEVICE)\n",
    "\n",
    "def class_to_class_index_tensor(c):\n",
    "    \"\"\"\n",
    "    Given a tensor of classes, return the corresponding class indices\n",
    "    as a tensor.\n",
    "    \"\"\"\n",
    "    return torch.argmax(\n",
    "        (c[:, None] == torch.tensor(classes, device=c.device)).int(), dim=1\n",
    "    ).to(DEVICE)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c5f55c8b",
   "metadata": {},
   "source": [
    "### Import models"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "efedd4e4",
   "metadata": {},
   "outputs": [],
   "source": [
    "branched_model = model_util.load_model(\n",
    "    scrna_ae.MultitaskResNet, branched_model_path\n",
    ").to(DEVICE)\n",
    "\n",
    "label_guided_model = model_util.load_model(\n",
    "    scrna_ae.LabelGuidedResNet, label_guided_model_path\n",
    ").to(DEVICE)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "82bd65a8",
   "metadata": {},
   "source": [
    "### Sample objects"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "1f7c9fb4",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████| 500/500 [00:29<00:00, 16.89it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████| 500/500 [00:29<00:00, 16.87it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 2\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████| 500/500 [00:29<00:00, 16.84it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 3\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████| 500/500 [00:29<00:00, 16.79it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 4\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████| 500/500 [00:29<00:00, 16.80it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████| 500/500 [00:29<00:00, 16.74it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 7\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████| 500/500 [00:29<00:00, 16.75it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 10\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████| 500/500 [00:29<00:00, 16.75it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 12\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████| 500/500 [00:29<00:00, 16.73it/s]\n"
     ]
    }
   ],
   "source": [
    "# Sample digits of each class from branched model\n",
    "branched_samples = {}\n",
    "for class_to_sample in classes:\n",
    "    print(\"Sampling class: %s\" % class_to_sample)\n",
    "    sample = generate.generate_continuous_branched_samples(\n",
    "        branched_model, sde, class_to_sample, class_time_to_branch_tensor,\n",
    "        sampler=\"pc\", t_limit=t_limit, num_samples=500, verbose=True\n",
    "    )\n",
    "    if latent_space:\n",
    "        branched_samples[class_to_sample] = sample.cpu().numpy()\n",
    "        sample = dataset.decode_batch(sample).cpu().numpy()\n",
    "    branched_samples[class_to_sample] = sample"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "a7395b41",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████| 500/500 [00:28<00:00, 17.34it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████| 500/500 [00:28<00:00, 17.33it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 2\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████| 500/500 [00:28<00:00, 17.34it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 3\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████| 500/500 [00:28<00:00, 17.34it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 4\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████| 500/500 [00:28<00:00, 17.34it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████| 500/500 [00:28<00:00, 17.34it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 7\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████| 500/500 [00:28<00:00, 17.34it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 10\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████| 500/500 [00:28<00:00, 17.33it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 12\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████| 500/500 [00:28<00:00, 17.34it/s]\n"
     ]
    }
   ],
   "source": [
    "# Sample digits of each class from label-guided model\n",
    "label_guided_samples = {}\n",
    "for class_to_sample in classes:\n",
    "    print(\"Sampling class: %s\" % class_to_sample)\n",
    "    sample = generate.generate_continuous_label_guided_samples(\n",
    "        label_guided_model, sde, class_to_sample, class_to_class_index_tensor,\n",
    "        sampler=\"pc\", t_limit=t_limit, num_samples=500, verbose=True\n",
    "    )\n",
    "    if latent_space:\n",
    "        label_guided_samples[class_to_sample] = sample.cpu().numpy()\n",
    "        sample = dataset.decode_batch(sample).cpu().numpy()\n",
    "    label_guided_samples[class_to_sample] = sample"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "eb232f52",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 7\n",
      "Sampling class: 10\n",
      "Sampling class: 12\n"
     ]
    }
   ],
   "source": [
    "# Sample objects from the original dataset\n",
    "true_samples = {}\n",
    "if latent_space:\n",
    "    true_latent_samples = {}\n",
    "for class_to_sample in classes:\n",
    "    print(\"Sampling class: %s\" % class_to_sample)\n",
    "    inds = np.where(dataset.cell_cluster == class_to_sample)[0]\n",
    "    sample_inds = np.random.choice(inds, size=min(500, len(inds)), replace=False)\n",
    "    sample = dataset.data[sample_inds]\n",
    "    if latent_space:\n",
    "        true_latent_samples[class_to_sample] = dataset.encode_batch(torch.tensor(sample, device=DEVICE)).cpu().numpy()\n",
    "    true_samples[class_to_sample] = sample"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "0144b5f5",
   "metadata": {},
   "outputs": [],
   "source": [
    "# # Rescale sampled values to match range of true values based on standard deviation\n",
    "# def rescale_samples(samps, samp_class):\n",
    "#     t = true_samples[samp_class]\n",
    "\n",
    "#     s_z = (samps - np.mean(samps, axis=0, keepdims=True)) / np.std(samps, axis=0, keepdims=True)\n",
    "#     s_trans = (s_z * np.std(t, axis=0, keepdims=True)) + np.mean(t, axis=0, keepdims=True)\n",
    "    \n",
    "#     return np.nan_to_num(s_trans)  # Convert NaNs to 0s\n",
    "\n",
    "# if latent_space:\n",
    "#     branched_samples_scaled, label_guided_samples_scaled = {}\n",
    "#     for class_to_sample in classes:\n",
    "#         branched_samples_scaled[class_to_sample] = rescale_samples(branched_samples[class_to_sample], class_to_sample)\n",
    "#         label_guided_samples_scaled[class_to_sample] = rescale_samples(label_guided_samples[class_to_sample], class_to_sample)\n",
    "# else:\n",
    "#     branched_samples_scaled, label_guided_samples_scaled = branched_samples, label_guided_samples"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5206c88c",
   "metadata": {},
   "source": [
    "### Compute FID"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "ca83ee3e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "FID of 0\n",
      "Branched FID: 375.3394\n",
      "Label-guided FID: 375.9273\n",
      "FID of 1\n",
      "Branched FID: 162.3776\n",
      "Label-guided FID: 166.3055\n",
      "FID of 2\n",
      "Branched FID: 149.6286\n",
      "Label-guided FID: 153.1675\n",
      "FID of 3\n",
      "Branched FID: 151.0971\n",
      "Label-guided FID: 156.2666\n",
      "FID of 4\n",
      "Branched FID: 64.6149\n",
      "Label-guided FID: 65.1992\n",
      "FID of 5\n",
      "Branched FID: 1604.8071\n",
      "Label-guided FID: 1609.5925\n",
      "FID of 7\n",
      "Branched FID: 84.4859\n",
      "Label-guided FID: 85.2968\n",
      "FID of 10\n",
      "Branched FID: 318.3276\n",
      "Label-guided FID: 316.7784\n",
      "FID of 12\n",
      "Branched FID: 173.8220\n",
      "Label-guided FID: 175.6205\n"
     ]
    }
   ],
   "source": [
    "# Compute FID scores\n",
    "fid_scores = {}\n",
    "for class_to_sample in classes:\n",
    "    print(\"FID of %d\" % class_to_sample)\n",
    "    branched = branched_samples[class_to_sample]\n",
    "    label_guided = label_guided_samples[class_to_sample]\n",
    "    true = true_samples[class_to_sample]\n",
    "    branched_fid = fid.compute_fid(branched, true)\n",
    "    label_guided_fid = fid.compute_fid(label_guided, true)\n",
    "    fid_scores[class_to_sample] = (branched_fid, label_guided_fid)\n",
    "    print(\"Branched FID: %.4f\" % branched_fid)\n",
    "    print(\"Label-guided FID: %.4f\" % label_guided_fid)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "edc81176",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAApIAAAFhCAYAAADDZkpMAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/YYfK9AAAACXBIWXMAAAsTAAALEwEAmpwYAABTa0lEQVR4nO3dd5xU1f3/8ddHBFZAVJCmgFhQkWBFxQr2FrtGE2PE+IuJxBpj+8aCMRpLVIyixsSIJcYSBXsHEUWUIjZsqCAgAqIivX5+f5xzd4fZmd3Zu7vMlvfz8ZjH3bn3zJlz+2fPPeeMuTsiIiIiIlW1VrELICIiIiL1kwJJEREREUlFgaSIiIiIpKJAUkRERERSUSApIiIiIqkokBQRERGRVBp0IGlm3czMC3gNqsZ33BzzeMzMmlUjnyExn1+mzaMmmVlzM/vAzL4ys/bFLs+aYGb/NrMfzeyAYpelMma2wsw8a169KX9dYGb94jn3crHLImJmv4zH45Bil0XKM7NL4/4ZWM18/hXz6Z8x7yAzm29m/6xm3gPNbGFm3mtCgw4kMywA/lPBa1yaTM3sOOBc4F7gZ+6+rCYKWxsygurJBX6kKbAx0B5oVXslW7PMrH/cDv/KsbgrsC7Qbg0Xq6ZUu/yVbB+ponhhdzO7tNhlKRYzmxK3Qedil0UapgZwjLUj3Gc3qWY+nYEWwEaZM2u7omrt2si0Dprl7jW6Ac2sJfAz4GrgMm9gI7u7+wIz6was7e5zi12eNeRQoJ27zyh2QVKq7+UXEWl03P0BM3sVmF3NrH4L/Nndv6p+qQrXWALJajGzJu6+MnOeuy8kBJINlrvPK3YZ1qRYo1xvg7D6Xn4RMzPA3H1Vscsia5aZre3uK4pdjmJx9+k1kMdKYI0GkckXN9gX0A1wYHKB6fvF9E8SIvtPgRVA/6x0xwGvAz8C3wPPAbvnyXMj4BbgC2AJMBMYAnTLSjckfvcvgV8RHrcvjvk/CnTNk/9OwOOE/2QWAm9nljdjncq9CtgeK7LTAa/Gz28NnAN8ENdrNvBvYIM8eR0PjAS+AxbF9ft1nrTrAX8BJse8p8Zt2CYr3cBYlp8DZwLvxm32LfAgsEmOtNmvVzPSvBzn9ctRph2AhwmB2pJ4bPwVWD9H2ikxn9bxeycDS+NnbwKaV+EYPgx4M26z74D/AVvl2Tc5y094ZHIF8H5GPk8De1Rl+8R0mwN3xX2yFJgTj79eOcrucZ3XB24mXOCWEc6FPwFr5fhMc+AC4D3C8fwD4fzaO8/26U5oWjIj7vv3gT8ATapwvr8M7AG8RDinFxCO83LHQfycAb8Bxscyzon7pWeO8zn7NSQunw6spPwxvWtM93iO770zLjulKteAqpY9a9v8h3CuPwrMJRz77wBHF7B9c153ku0K9I/vbwMujcfUqozlU+LyzjnyruhcLfj6nKfcyXmbnC/zgVHAkTnSvkoVr4dACXBVXL+lwOfAlcBpmcdIAeVcDxgUj6WlwGfAZUDfmM+/0p4vadYrfm5f4EXCNeZH4DXgp3mOjR+BvYAxMe8hGcubA+cR7hMLyHPPqOwYS3OdIFyvbiXcq5cAH8ZtcFnMe2CB+2dj4D7C/Wgx4Zw7CfhXzKd/Rtpf5tv3wC7AC5Rdm0YCBwD/iJ/ZMyPtanln7Mfs18Cs7/gpMCLu38WEe+m5hCeSla9roSdXfXyRPpBcEKdvAY8A+2WkuTnjJHg+nijL4+unWfn1jAejE26M/wM+ju+/J+PmS9mN533CxfQ14Im4Yx34KHunEoKzpYSg4nXgWcKN14FrY5oewAPA0Dh/fnz/QAHbo6JAciwhKHiREJT8GOe/nCOfWzO+++l4Uiwix8UO6EQ4cZ0QgA0jBG3JNlg/I+3AOH963P7DgceAb+L8mUCXmPaYuN6jM/J+APhTRn75ArET43ZeRbih/I8QFHks28ZZ6adkbKOFwDNxnZfkWucKtv8v43d6LPdj8Xu/T+ZnpS9XfkKbyWR7fhnzeDu+XwkcV4Xtsxdl58bYuB3eo+yc2TarPB738weEIOQJwsVqZVx2aVb6VoSg2QkX/P9llGcl8Kus9HtTdry/DTxF2fny3yqc77MIx/JHhIBpfJy/HDgk6zNNYrmcEIQ9RbgRJsd375ju9Lj9ku0zMb4/PS6/O84/Oiv/v2Rsz5KsZR/H/d6xKteAqpY9a9t8Qji3P4/Hzrtx/ipgr0q27wPxlRwzj8f3PeLy/nH+wrh/RxKut72yzqOCA0mqcH3OU+YOlF1vknV+lbJj9rTqXA8J/RJeyCjjE7GcSwjnSM5gIkc5WxACeo+fG0r4R2gZISDPdW0t+Hyp6nrFz5wbj4vFhGvxy5Rd53+X49qwMqadTDjvzslYt+Q68HVctxcJx7kDVxV6jKVY75aEczU5Rx4nXLNWZOyfgQXsn06U3SO+YvVrWZJP/6xrfbl9D+xD2X3jvbid3o/bbhqVB5J/itvj8zj/jfj+mIzPXEXZtfo5wv0q2Z5PEJ4QVLy+lSWozy/SB5IOHJ9j+fFx2QdkXNyAPQkn2zfE2ibCBSM50c/PSGuEdpUOjMqYPyTO+4HVa4o2yDggD8qYv2nc2Quy0m9M+M90FbBD2m0RP1NRIDkN2Dor//lx2VYZ84+m7Ia0Ucb8zQgBoLN6oP5MnHcjscYqbsskGB2UkXYgZSf8thnzWxIuzg48nFX+/uT/bz1XILYZ4Ua3CNg3Y35Tyk7aF7PymULZid8pY/4uhAvAMmDdSrZ9W2Be3I/HZ8xvkvG92fsmV/nPj/OeJOO/b+AUyoLttQrcPh/EZSdkzU8uRI9mzU/OpVeA1hnzj0u+Oyv9LXH+/8iotSXUyq6Ix9f6cd56hGBzJatfFNen7IJ9VBXO9wuylp1N2XHeNGP+BXH+CFb/p+Zncf7ErHySYzQ7aE6uJbdmzX83o0w/zZjfIc57pxrXgILLnrVtrss6dm6jwGA963zonDU/OdaWZJa/ss9VcKwXfH2uoKzJug0m4wYK7EfZPx2Z81/NOE4KuR7+hrJ/QDtlpf+CwgPJ5B7yVta+3JKyAONfGfOrdL6kWK8+Me+vgW0y5vcgBE6Ls9Y3ObZuJStQAf5IWRCTee79hHAdXkbWk6AKjrGqrneyXd9g9WvWDoRaVqewQPI/Me1jrH4tO4CygLh/xvxygSSh6WGyXpdk5T8gYxvmDSQz5g+J83+ZNb894dr6I7B5xvwulMUdh1a6voVcCOrri7LgqbQWLscr8+DqF9OPzpPfhLi83GOSjB14VHzfN75/L0daIwRRq4gBRcaO/mOO9MkNNjMgvSnO+78c6ZODclCObVFTgeRxOdI/EZcdmzFvRPa8jGWnx2V/i++3je8nUf5RS4t4sH9HvPBQdpO+MkfeW8TyLwPWy5jfn6oFkoPivJtypG9BWe3njhnzp8R5vXN8JgkUdqpk2ycXilyPOJsRa0gKKH8SgJ+TI5/B8bjLvGBWtH12zFXuuK0d+CRrvsdjvH2Oz8yLy9tmbMsFhIvsRjnSP0hGcEVZoHdXjrR7xmXDKtnG/WK68XnO0aQm95A4rwkhkFhOjqYmGdt/+4x5yTGaHUiuH4/PDzLmdY1p789eN+CEOO+vGfMKvgZUtewZ2+Z9yt/od0iWVbR9c5wP+QLJB6vyuQqO9YKvzxWUtTvQmxz/6FH2j29mQPQqVbseJk8DjsiRvqBH2/HYnJV9rGUs/x3lA8kqnS8p1uvxOO8XOdJfGpedmzHPCdewkhzpN4n7YMMcy16Pn92twGOs4PXO2q7b5kif/MM8sJL9sz7hvrM0zzok53f/HOfrkIx5h5L1z2NWPsnThOoEkjvl+w5Ck7EhFBBINpbONq0IbRNymUw4CTItyk5kZh0IF9Bv3H10jnwmxumOhMex/eL7J7ITurubWV/CY8clWYu/yZH3nDjNHIbnkDjNLnt2WWpLpeU0syaEx6GrCDVi2f5DuLAmvcKTdXrSy3duWmRmnxIO/E0J/70nlmdn7O6TzWwCsDOwHeERVxoHx+m9Ob5jkZk9TLhY7Ue4kWUqdF/m0idOn87xvcuyx5CswBuE9qN/NLOZwDMeOorh7r8vMI/keyeY2VpxnMq9CY9vSghBIIR2TdlWuXuunohzCG3RWhH2/86EmuTX3f3rHOnPA64nPLaDmj3+v8+eEc/Rp4BtYtmeI9zc2gNvee5ekRMJx8GOGWXIyd1/MLO3gN3NrIO7zwIOj4sHE5rFHG5m5uGq3jcuez4jm6psg7RlnxW/P1Ohx3Chyl1vqyrF9Tknd/8s5retmR1CeCLRghBkrBeT5TrOC7keNotlXEF4vJ1tab5yZdmUsC+nu/vEHMuz7ymQ/nwpZL2McJ1cQe7rfL783d3LldXdpwJTzWwLMzuZUMvairAPNo/Jcu2DXKqy3sl2nebu7+VIX+j+2ZHwxGqUu39bjXzy3gOiXPu5qj4iVM5sb2bXEALuKQDu/l/gv4Vk0lgCyc/dfYtq5tElTjtWchPfME47xem0XInc/fMUZbAc5fkonMcVlmVNSwrUjlATMtPdcwV7C1n9ppWs00VmdlEF+W/I6oFkPlMJQUB1BlRPyvRlnuWfZ6UrVN6dFnWI06kVpqqEuz9kZlsD/0foLLTczN4kXJzuyXOhy8nMuhBuwjX5D0qyHSo7X2YRagsSyfZ+rhaP/6QsyfGTfOeuBV4DKvM8sDvhn86HCYHkbMLjyicItZm7Emoe+hGerGQGSVW5BtR02aHyY3hNqur1OScza05ov5qv4iGNZDu1Jdxzp7t7ocFELsm1YUoVPlMb50vmeq0T/55f3fxjYHo9oTNMdce6rsp618g1txbymVLNfPKKlSFHECp1LgEuiWNNv0ioHR1bSD6NJZCsCcl/QHMIGzmf8XGaHLW1NYxFUp7kkV8uuf6bXJOqug2SdRpDWYCWyw8F5pfUalbnhpdcyMoFwlHTauRdiGqfo+4+0MzuILRN3J/Qs3Jv4FIzO8XdhxWY1aOEIPIl4BrKOtFsTJ4AsArSHitPEB6J57Iyz/xCJUORJGVLvvNLVg/osn1SYP7PA38G9jWzZwjB4n9ibWgSSB5hZl8Q2poNy/qHrCrXgJoue11T1etzPlcSgsjJhEeyrwFz3H1FvMFuXtGHC1Tdczo5R6qST22eL0neiwgdY/L5oMD8fktoJzkHuJhwvZkVn8S8TKg5r2rZqrLeNRUXFWM/V5m7v2FmWxBqbw8GDiQ0rRpgZvcAv8l+QphNgWThksdzC7ywwc2TC3jXWizPRoRGuGt+3KjCzCUEYJ3MrGmuWsksyTZ+zt3/XAPfv2lWvmlMJ7QB3IzQZi7fd1R7DLAsM+O0qjWdOcUavcHAYDNrSmhHdQtwn5l1dvcfK/q8mXUn1I7NIrRTXJaxrCaKWNXzZTZhv9zk7mmbLVQmKcvsrOmUAq8BlRlHuFnuQ7h4Nyf0KMXdJ5rZVOBIQqc9WP2xdlKeQq8BNV32NSEJjgs5wKp6fc7n5Dg9zt3frUY+uSTXww5m1szT/xJammtDbZ4vcwlBT1NC27zqjgWZ7IPfuHu5pmFVVJX1rqlrbl3Lp1Jxnz0VX5hZb+Ah4FTCP1NDKvp8Y/mJxJowmXCz62ZmOR/txcciiVFxenietCPNbJqZpW1nlOR/bAFlgfw1FrUmXijfJhxnh2UvN7MjzWy6mV0WZyXrdJSZlTs2zayp5Y5ayv1DZGYbE9pTLmf1x+dV3Q6vxOmvcnzHOoSeopC7zVN1JLVGx+T43hIKrGU1sxFmNtbM1k3muftyd7+V0Kt8XUIHg9LFebJKfnZxao4bYE00oRhLaPOzs5l1zF5oZtfGY2X/OKuqx39FNsgzP2kf+1acvk1oRL9rvp9iq8p5F9sevkTY/mcQ1v+ljCRPENpo/r/4PjuQrMo2SFP2mpL22jM/TssdD4T2tJmqen3OJznOP8v6bDPCuZJaPG/GEc7do3IkaZFjXq58viL0RN7YzHbJkSR720DNni/Z5VlCOH+bkv9+V5X8k33waY5lbfMVI8/8gtc7tg38mvzbtaD9Q6j1XgrsEu9DafNJ7gFH57nv5drP+eTcPmZ2vpmNM7NfrJbYfRxhBAMI99EKKZAsULzo30S4CAzJvhib2VnAZDP7SZz1CqEGa3szOycr7bmER4uT3T1fdXtlBhH+C7zSzPbMyr8v8KWZ/TxjdtKhpX0MRNaUv8fpXzMDBDPbCLiW8Fg0OdlfJDz+2AG4LnbWSdK3Jfy3NDRHkDnAzHplpG1BaOfUBHjMV/+FnmQ7FFrzdQvhBnxm3K7Jd6xNOB7aAy/VQu3FI4ROIAebWWkwGb/3dgo/d38kdLS4JPNiZGZbEh7TrWL1x9L5ts+nhEdA25nZthn5dAHuiW+rczNaEPMpAW6JN+7kO/YEziJcOMfF2XcQhhT5vZmtdpOIx8KHZvbHAr9+RzP7v6w8ziI8xv+K0EOYeBz9i3AjuM/MNshIv7aZ/RWYGI/tRGXHWxIc7g8MTzpCRUltzIHAx7ETQqZBFHgNSFn2mlLVcy6RPIY+LXOmmf0/yjoiAKmuz/l8FKdJrVjyD+PdlLWVrU7A/Y84vcpCB6HkOzYljPlXqH/G6U1m1jojn60I7aGz1eT5ksuNcXqrmW2Tlf9xhM4z+xSYV7IPSv95j8fotcD2cVb2Psh3jFV1ve+K05sy//mO/5wMKKTw7j6f0EmlKTAo61p2MKGJUSFGEJp4bUNoL5pZ9rMI1/VC5ds+XxACxf8zs9IKgXifOSC+nVJp7pV1667PL9KPI1lusNW4fC3KBvRdTKiF+h9lg32OBlpkpO9FeHTlhGFfHqVsSJGvWX3cpiHk6J4flyXDJwzMmn8u4UayKn73w4T/DJ3wCHKrrPTj4rLPCcMoVDaWYUXD/+yZI32+oQeSEfjnE4LBZykb8PT6rLQ94rZxwk38UUKP2QVxPf+QkXZgTJc8Mnol7o9kSJ6vKT8cRHvKBskdS8bA7OQf5Lg/IYhaSajmf4TQ1swJg0Rnf8cUqjBsSQXb/2eUDUj+OmUDof8Yj7/sfZNvHMxke0wmjGv2EmVjmf21Ctvnb5Qd+8kg60spGzJjGdAsI70DK/Ks2+S4vFvGvHUzjtHpcX1fi9tgGVm/LELZYNxOePz7MOGfkpXxeCl3jOY536fE4+fjeLwlZVgG7J/1mRaUDUEyj/jPDWXH7OOsPi5nL8rO0TeIQ11lbe9kH/82a9nalI1dd3OedTiXAq8BVSk7FVwLgc7JdqvsGM46br4lDDq+T8Z55eQZoJ8QyC+LacbH42FSfP8D5Y/1Kl2f83znERn7YxShc9kcQm1x8p37pL0eEgLdpzL2wTDC9W0JZefRkAK26TqUDZw/h7LzehnhH9By25UqnC9VXa84/+Y4fzkhCHqUsrFnPyVjGDAqvjb0pmwQ7vHxuJwWy5ncT08t5BhLsd7rUHb+z47bdTjhXpjsn4EF7J/2lN0jpsZtkYxbmeSTeVzkG5C8b8a2eIdw7/mAcIwmQ6gVMvzPTym7po0kDiVIOB4fpuz+/BzhmpAMdfUxGcPD5V3fQi4E9fVFDQeSMc1awK8JI+8ngwFPBC4k4yaakb4L4b+caZT96sDfgXZZ6YZQxUAy40B7mhBMLSX8h/F3oEOOtD1juRcSLq6tKtkeNRJIxmUnxxNpQTxgR5LnZ9YIvdVujmVcRrh4PEPGwOUx3cD4fZcRhrh5L5503xHad3TLk/8xhEdXC4GhGfPzBnnAboQaom8JN6mPCR0l1s+Rdgo1EEjG9AfHfZb8XObQuB8/z7Fv8gXCHeIx8XncPt8Qgu7Dq7h9jBC4fBL3y0zCz/Z1INw4JrL6ANgV3SzKBZJxfgvgckLAsCTu+0eB7fLks33c18mv00wj1GxuUYXz/Z+EYapeJgTpCwnHec5fbiHUhpxPuLAvip8ZQxgXtdyvQBBqMpJ/AP6eY3ly49o4x7IH4rKDKliPqlwDCio7NRtIrke4Wc0jBK394vz+VBBIxjT7Ea4biwnXjjfi8Znc/LKP9Spdn/N85yHxO5fEMj9JCGpviHn9JiPtq1Q94GpO6NQzJWN/XU2oeS4okIz5tCbUSs8gHPufEX4K9bfJcZ32fEmzXnHZ0YTAax5l18mryaq0oIJrQ1y+KyEwXhj34XBCJ8Hz4j64opBjLM11Iub1d8L1bSmhhvSPGcfrwAL3TyfCsHFzKftp0VMp+6nF/hlpcwaScdnOhH+K5lP2E4kHZuyjzB8jqGjfXEW49n8PXJh1zpxOOObnxe95L5az0iDS3UsHdhapd8xsIOHCeZm7/6XIxRERKTozu4QwqsIt7n5ukYsjtcTMPiGMsbmD5x5PdI1RG0kREZF6JrPNdpakvexHeZZLPWFm3cysW475GxFGDFlBeLpTVAokRURE6hEz6w+8bGan5Jh/KKHZQq5fc5H6ZQjwrIVxHgGII738g9CZZ6in77BbYzSOpIiISP0ygdCebYiZXUhoi9id0LlrJaHj1pwKPi/1w0vAX4BJZvYacYg0QmeeKcA5xStaGdVIioiI1CMefgt6G0Jv5aaEXrmdCZ2u+rr7A0UsntQQd7+a0PHtacI/CQcQOiDdDOzs7jPzf3rNUWcbEREREUlFj7YLtOGGG3q3bt2KXQwRERGRSo0fP/5bd29XecrqUSBZoG7dujFu3LjKE4qIiIgUmZll/xpWrVAbSRERERFJRYGkiIiIiKSiQFJEREREUlEgKSIiIiKpKJAUERERkVQUSIqIiIhIKgokRURERCQVjSNZw+bNm8e3337LsmXLil0UEYmaNWvGhhtuyHrrrVfsooiINCgKJGvQkiVLmDVrFp07d2adddbBzIpdJJFGz91ZvHgx06dPp3nz5pSUlBS7SCIiDYYebdegOXPm0K5dO1q0aKEgUqSOMDNatGjBhhtuyJw5c4pdHBGRBkWBZA1asmQJrVq1KnYxRCSHddddlyVLlhS7GCIiDYoebdegFStWsPba2qQiddHaa6/NihUril2MxunGWnxCc77XXt4iUilFPTVMj7RF6iadmxXbd8BXtZb38M1rLWsRKTI92hYRERGRVBRISqPRrVs3zIz+/fvX2ncMGTIEM8PMmDJlSq19j4iISF2gQFKqLAnIMl/rrrsuO+20E9dccw2LFi0qdhFFRERkDVAbyTWkNtsfpTH89q41mt+CBQuYMGECEyZM4Pnnn+fVV19lrbX0f4qIiEhDpju9pNanTx+mTZvGV199xZgxY9hvv/0AGDVqFKNGjcr5Gf3ij4iISMOhQFJSa968OZ07d6ZLly7suuuunHPOOaXLZs2axcCBAzEzunTpwl/+8hfatWvHVlttVZrm0UcfZY899qBVq1a0b9+eQw45hAkTJpQuT9obNmnShPHjx9O3b19atGjBpptuyg033FCuPHfccQfbbrstJSUltGnThuOOO47JkyeXS7dq1Souu+wyOnbsyAYbbMDhhx/OtGnTVkszZ84czjjjDDp37kxJSQk9e/bkjjvuKJfXDTfcQLdu3SgpKWHffffls88+S7UtRURE6iM92pYaMXPmTB544AEAmjZtSp8+fZg0aRIA06dP57LLLgOgZcuWANx2222cddZZpZ9fuHAhzz//PCNGjGDMmDFsv/32pctWrVpF3759WbhwIQBTpkzhwgsvZJtttuGwww4D4A9/+AM333xz6WeWLl3KY489xsiRI5k4cSIbb7xx6bKHH354tZrRp59+mh9//JGRI0cCMHfuXHbddVe+/PLL0jSTJk1iwIABzJ49myuuuAKAQYMGceGFF5amGTFiBCNGjEi7CUVEROod1UhKaiNHjiztbLPRRhvxyCOP0KZNG+677z66dl29DeaAAQOYNGkSw4cPB+DKK68EYK+99uKDDz7gmWeeoaSkhKVLlzJ48OBy3/XTn/6Ud999l6eeeormzZsD8PjjjwMhyEuCyBNPPJEPPviAhx9+mObNm/Ptt99yyy23rJZXq1atGDZsGB999BGHH344AK+99hozZ84E4NJLL+XLL7+kpKSEhx56iI8//rg06L366quZNWsWS5cu5brrrgOge/fuvPHGG4wcOZKtt966+htWRESknlCNpNSoxYsX53ycfOutt5Z2vnF3XnrpJQA22mgj2rdvT8+ePdlpp5144403VqsJTFx77bV069aNbbfdlh122IExY8Ywffp0AIYNGwbAWmutxeDBg2nTpg09e/Zk3LhxvPbaa+V+zeTwww/nyCOPBEKA+9RTTwHw1Vdf0alTJx5++GEAjjrqKPbYYw8AzjzzTG699VaWL1/Oyy+/zFZbbcU333wDwMCBA9l9990BuPDCC/n1r3+dfgOKiIjUIwokJbU+ffrw6KOPAjB//nyuvvpq/vOf/3DZZZfRu3fv1dJm9uA2M7bYYgtuu+02HnroIaZOncoPP/xQuryyn7FLaiSXLl0KwNSpUwFo164dbdq0KU13/fXXV7oOJSUlpX8vXryYuXPn8v333wPw0EMP8dBDD5X7zLRp01b7XK9evVZbNxERkcaizjzaNrO2Zna2mU00sz4Z87c1M8/zyky3t5mNNrPFZjbXzIaYWdus7+hlZi+a2UIzm2dmj5vZJmtyPRuSpLNN586d6dGjx2ptFF944YUKP3vooYdyySWX8O67764WRKZRk8FbEpxWZMmSJSxfvrz0fZMmTWrs+0VEROqTogeSZraXmT0KfA3cAmyXlaRznC4DZmW9lsU8egEvArvFtBsApwBPmdlaMU1H4FXgAEJNbEvgaGC4mbWojXVrbNy99O/MQCvbhx9+WDo80EknncTEiROZNm0affr0yfuZimy66aYAfPvtt6sFpQ888AAXX3wxQ4YMKTiv9u3bl3YI6t+/P+6+2mvVqlUMHDhwtTagH3zwQenfmdtARESkoSt6IAmcDxwH5Is8ku62T7p7x6xXMlbMBUBz4ElgXaAnsIgQWO4X0wwA2gBj47QLIXjdDDipZlepcVi6dCnTp09n+vTpvPvuu5x++umly7IfbWfKDLa6dOlC8+bNGTNmDJ9//nmqchxxxBGYGStXruTss89m0qRJPP300/z+97/nuuuuWy3Qq8zaa6/NscceC8B9993Hddddx8cff8zIkSM55phj+N3vfle6fhtssAEAl19+OW+++SavvfZaQY/TRUREGoq6EEi+SAgk2+VZntRITq8gj75xeo+7r3D3jwi1jwB7ZaV50N0XuvtMYFhWGqmCMWPG0KVLF7p06cL222/PE088AcB2223Hz3/+87yf69GjB9tuuy0QOtH06NGD448/njlz5gDhV3KqokePHpx33nkA3H///fTs2ZPDDz+cH3/8kQ4dOpQuK9T111/PpptuyqpVq7j44ovp0aMH/fr1Y+jQoQwdOpRZs2bRrFmz0mGAPvnkE3bffXf69u3Lp59+WqXvEhERqc+KHki6++3u/pi7L86TJAkk9zGzGbEN5Etm1iMjTac4nZ0xb2acdqxCGkmpWbNmdO/enQsuuICRI0eWdojJpUmTJjzzzDP87Gc/o02bNnTo0IHTTz+dM888E4DPPvusyr+Ac+ONN3L77bfTq1cvmjdvTtu2bfnFL37BmDFjVhtDshAdOnTg7bff5uyzz6Zr1640bdqUTp06ceqppzJ+/Hg6dOgAwDnnnMPNN99M165dad68OXvttRf3339/lb5LRESkPrO61KbLzJLC7ObuY+K854GD4vz5hEfXAFOBrdx9aZ7P3Qn8FrjX3fub2RRgE+Dn7v5QTHMx8FdgpLv3q6hsvXv39nHjxlVY/o8++ogePXpUmEZEikfnaH77Dviq1vIevnkt9mk8v+7cw0TqEjMb7+7525nVkKLXSBbgTOD3QG93bw3sBCwmBIXH1+YXm9npZjbOzMYlj11FREREJKjzgaS7T46Pv8fH9xMI7SoBkqqFZODBzHExkzFZlsTp8gLSZH/3Xe7e2917t2uXrwmniIiISONUpwNJM2tiZufGV4fMRXHaNE6/idONMtIk7R5nViGNiIiIiBSoTgeS7r4SuBC4GbjEgq0pG9JnbJyOitPTzKyZmXUH9onzXstKc5KZtTaz9sBRWWlEREREpEB1OpCMrorTcwidbSYRBhN/F3giLruB8Oj6QGAe8ElM8zplwwANjsu2Bb4ljCHZGfgcKP87eCIiIiJSoTofSLr7HYQBwycQ2jTOAe4G9nP3ZTHNO8DBwFuEx97fA/cCR3rslu7uM4B+wHBC0LkIGBrzyTf0kIiIiIjksXblSdYcd8/5o8nu/iDwYCWfHQ5U+Bt77j6RssfiIiIiIlINdb5GUkRERETqJgWSIiIiIpKKAkkRERERSUWBpIiIiIikokBSRERERFJRICnSSI0YMYLJkycXuxip/fe//2XevHnFLoaISKOmQFKKqn///pgZ3bp1q5P5VdV1111HixYt2G233Vi2bFnedK+++ipmhpnx6quvVvt7hwwZUprflClTKk3/z3/+k/3224/HH38cgG7dumFm9O/fH4DZs2ez1VZb0bp1ax58sMKRt4pi1apVDBgwgJ133pnZs2cXuzgiIo1WnRpHskG7MecQmcVzvqf+aLdu3Zg6dSp9+/atkSCoIfnuu+9YvHgxs2bNYuXKlcUuTk6vv/46Z5xxBttvvz0DBgzImWbZsmXMnTuXBQsW1Mlav7XWWovBgwdz0kkncfTRRzNq1CjWWkv/F4uIrGm68orUoOuuu465c+fy2Wefsc466xS7OOWsWrWKM888E3dnyJAhtGrVKme6zp078/XXX/P9999zxhlnrOFSFuYXv/gFRx99NKNHj+af//xnsYsjItIoKZAUqWFt2rShSZMmxS5GTi+88ALvvvsuBx10ENtuu22FaZs1a8Z66623hkpWOHdnxYoVAPzxj38E4Prrryf+GqqIiKxBCiSl1syfP5+LL76YzTffnJKSErbYYgvOP/98fvzxx5zpr776ajp16kSLFi3Yf//9+fDDD1dbvmLFCq699lq23nprSkpK6NKlC+effz4LFy6sVjmHDRvGdtttR/Pmzdliiy24//772WCDDTAzhgwZAkC/fv0wM/r161f6uSlTppS2S0zS5Wujee+997LVVlvRvHlzdt55Z95+++2cZZkzZw5nnHEGnTt3pqSkhJ49e3LHHXeUS3fDDTfQrVs3SkpK2Hffffnss88KWtcHHngAgMMOO6zCdLnWbeDAgZgZm266Ka+88gq9e/dmnXXWYauttuL+++8vl8ddd93F9ttvT4sWLejQoQO/+c1vmDNnzmpp3nnnHY466ijatm1L69at2XnnnXnkkUdWS5OU48ILL2SPPfagWbNmpeux22670aZNG7744gs1sxARKQK1kZRasXLlSg444ADeeuut0nmff/45N910E6NHj2b06NGYlbUbnTp1Kpdeemnp+1deeYV99tmHSZMmseGGGwJw7LHH8uSTT5ammT59OjfddBMTJ07kpZdeStVG7tlnn+XYY49l1apVpWX81a9+VeV8KjJs2LDSTiwA48aNY9y4ceXSzZ07l1133ZUvv/yydN6kSZMYMGAAs2fP5oorrgBg0KBBXHjhhaVpRowYwYgRIwoqy8iRIwHYe++906wKADNmzOCQQw5h+fLlAHz66aeccsop7LDDDvzkJz8B4Oyzz+bWW28t/czixYv517/+xejRoxk7diwtWrRg7Nix7L333ixZsqQ03bhx4zjhhBP47rvv+N3vfrfa995www3lymJm7Lnnnjz55JOMGDGCffbZJ/V6iYhI1alGUmrF888/XxpEXnvttXz66aecffbZAIwZM4bRo0eX+8yf//xnPvjgA6699log1M4NHjwYgIcffrg0iLz00kv55JNP+Pe//81aa63F8OHDGTZsWN6yTJ8+vdwrcdFFF7Fq1SrWW289/ve//zFx4kROOeWUGtkGiSuvvBKAdu3a8fzzzzNmzBj22muvcukuvfRSvvzyS0pKSnjooYf4+OOPOeuss4BQWztr1iyWLl3KddddB0D37t154403GDlyJFtvvXWl5ZgzZw4zZswAKCh9PsuXL+eMM87gww8/5J577gHC4+ahQ4cC8NZbb5UGkaeddhofffQRw4YNo2XLlkyaNIk777wTCI+jlyxZQvv27Rk1ahQTJkwoLdeNN95Y7ns33nhjnn32WT7//HOOOuqo0vk9e/YEYOzYsanXSURE0lEgKbWiT58+vPPOO7zzzjucf/75dO/evTQoAlardQPo2rUrl112GT179uSiiy4qbb+XPK586KGHANhkk0347W9/S4sWLTjggANKa8Cee+65vGXp0qVLuReEAPODDz4A4JxzzuHYY49lu+22Y+DAgTWyDQC+/fZbJk6cCMC5557LQQcdxK677loaXGZ6+OGHATjqqKPYY489aNmyJWeeeSYQgreXX36Z999/n2+++QYIj5p333139t5779VqKPNJPteqVSuaNm1arfW65ZZb2Gabbejfvz8dO3YEKA3Qk321zjrr8H//93+0atWKnXbaib59+wJl++qaa67hnXfe4c0332TPPfdkhx124NhjjwXKHx8Av/71rznkkEPYbLPNWH/99Uvnt2nTBoBZs2ZVa51ERKTq9GhbakXbtm2ZOXMm119/PW+88QYzZsxg6dKlpcuTzhKJzMfcEGrM3nvvPb7++muA0oGzp06dWhoIZpo2bVqVy5j5mZ122qnKny/E1KlTS//u1atX6d/Z6zt37ly+//57IARiSTCWadq0aZSUlBSUXy7JMD6ZQVhNaN68OUDp/k321eLFi9l8883LpU+2+xZbbMGDDz7IP/7xDz799FNmz55d2mEm19BJ+ZoubLDBBgD88MMP1VsRERGpMgWSUis+/PBDdtllFxYvXpzq89lBQ2YQmktmO7tshfTmra1e1kk7wsq+o7L1g7COheaXS5K+kO+qjkL31RVXXMFVV11V7e9LtkmLFi2qnZeIiFSNHm1LrRgyZAiLFy+mSZMmDBkyhMmTJ/Pmm2/mTZ8d7CW1Wp06dQJg0003BWDzzTdn5cqVuHvpa+XKlal67G6yySalf0+YMKHCtM2aNQOo8NdqcunatWvp38ljdCi/vu3bt6dly5ZA6PmduX7uzqpVqxg4cGDB+eWS1NwlNZ+1JdlXrVq1YsGCBeX2VfLLO0lv9O23357XX3+dqVOnct5551X5++bOnQuEWnAREVmzFEhKagsWLGDixInlXt99911pYFNSUkLnzp1ZvHhxaWeMXL766iv+/Oc/8/HHH3PbbbeV9mpO2tWdeOKJQOhVffLJJzNx4kTGjx/Pn/70J3bbbbe8QwpVZKONNipti3nTTTcxdOhQ3nvvvZxtJJMAbsKECTz77LOMHTuWCy64oKDvSDqD3HTTTbz00ku8/fbbpT2wE2uvvXZp+8D77ruP6667jo8//piRI0dyzDHHlPZg7t27d2lAePnll/Pmm2/y2muvcf3111dalk022YS1116bFStW8O2331aaPq1kXy1YsIBjjjmGt956i/fff5+//e1v/OQnPynt8JMcIx07dqR169ZMmTKFUaNGVfn7kraR22yzTQ2tgYiIFEqPtiW18ePHs8MOO5Sbf88993DccccxaNAgFi5cyP77718uzYIFC1Z736xZM6644orVAqy2bdvy+9//HoBTTjmFoUOH8tRTT/Hggw+u9vvPa621Fm+//XbO76nM9ddfz6GHHsoPP/zAMccckzfdCSecwN13383SpUsrHYMx21/+8heOPvpoZs2axYEHHlha5lxlGTVqFF9++SUXX3wxF198cemydu3a8ec//5kOHTpwxRVXcO655/LJJ5+w++67580v2zrrrMN2223H+PHjGTNmDD/96U+rtB6F2meffRgwYAC33347L774Ii+++OJqy1988UVOPfVUTjjhBG6//Xaef/55nn/++XL5LFiwIO8v72RKarr79OlTMysgIiIFU42k1Io+ffowbNgwdtxxR1q0aME222zDfffdx8Ybbwys/lgW4IgjjuDqq6+mY8eOlJSU0K9fP4YPH0779u2BECg9/vjjpbVazZo1Y/311+fggw/mtddeSxVEAhx00EEMHTqUXr160bRpU7bccsvVaveSAO2AAw7gzjvvpHv37rRs2ZIdd9yRxx57rKBOLkcddRT/+c9/2HLLLWnWrBk77LADTzzxRLl0HTp04O233+bss8+ma9euNG3alE6dOnHqqacyfvx4OnToAIQe5jfffDNdu3alefPm7LXXXjkHBM9XFigbT7K2DB48mH//+9+lg5a3atWKvffemyeffJJTTz0VCEP8XHjhhXTu3JnWrVtz8MEHlw73BOWPkVx++OEH3nnnHZo3b84RRxxRa+sjIiK5mX5WrDC9e/f2XINIZ/roo4/o0aPHGiqR1JQZM2aUBrgATz31VGlQ8txzz3HwwQcXq2g1bsaMGWy22Wa0bduWL7/8srTHdX3197//nXPOOYeTTz6Z++67r9L0Okfz23fAV7WW9/DNN6k8UVrn6x4mkouZjXf33rX9PaqRlEbts88+o2fPntx444189tlnvPTSS/zhD38AQi/g3XbbrcglrFkbb7wx5513XunQTPXZ3Llzueaaa2jZsiVXX311sYsjItIoKZCURu3JJ59k3rx5/PGPf2TLLbfkwAMPLO0xfsMNN7DeeusVuYQ17/LLL2eXXXbhyiuvzPmIvT5YtWoVRx55JLNnz+aOO+7IObaoiIjUPgWS0qj94Q9/YMiQIeyyyy6ss846rL/++uy33348++yzDBgwoNjFqxUtWrTgmWeeYb/9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\n",
      "text/plain": [
       "<Figure size 720x360 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(figsize=(10, 5))\n",
    "bar_width = 0.4\n",
    "x = np.arange(len(fid_scores))\n",
    "ax.bar(\n",
    "    x, [fid_scores[c][0] for c in classes], bar_width, label=\"Branched\", color=\"royalblue\"\n",
    ")\n",
    "ax.bar(\n",
    "    x + bar_width, [fid_scores[c][1] for c in classes], bar_width, label=\"Label-guided (linear)\", color=\"darkorange\"\n",
    ")\n",
    "ax.set_xticks(x + (bar_width / 2), labels=[(\"%d\" % c) for c in classes])\n",
    "ax.set_xlabel(\"MNIST class\")\n",
    "ax.set_ylabel(\"FID\")\n",
    "ax.set_title(\"Fréchet inception distance between true and generated digits\")\n",
    "ax.legend()\n",
    "plt.show()\n",
    "fig.savefig(\n",
    "    os.path.join(out_path, \"scrna_covid_flu_fid.svg\"),\n",
    "    format=\"svg\"\n",
    ")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
