{
 "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 model.sdes as sdes\n",
    "import model.generate as generate\n",
    "import model.table_dnn as table_dnn\n",
    "import model.util as model_util\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 time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "ee894a54",
   "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": "4e727b25",
   "metadata": {},
   "source": [
    "### Define constants and paths"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "6af88147",
   "metadata": {},
   "outputs": [],
   "source": [
    "models_base_path = \"/data/anon/branched_diffusion/models/trained_models/\"\n",
    "\n",
    "branched_model_path = os.path.join(models_base_path, \"mnist_continuous_alldigits/2/epoch_90_ckpt.pth\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "672bbc7c",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define the branches\n",
    "letters = list(\"ABCDEFGHIJKLMNOPQRSTUVWXYZ\")\n",
    "class_to_letter = dict(enumerate(letters))\n",
    "letter_to_class = {v : k for k, v in class_to_letter.items()}\n",
    "\n",
    "classes = [letter_to_class[l] for l in letters]\n",
    "branch_defs = [(('A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z'), 0.5235235235235235, 1), (('A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'X', 'Y', 'Z'), 0.5165165165165165, 0.5235235235235235), (('B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'X', 'Y', 'Z'), 0.5115115115115115, 0.5165165165165165), (('B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'X', 'Y', 'Z'), 0.4944944944944945, 0.5115115115115115), (('I', 'J'), 0.4794794794794795, 0.4944944944944945), (('B', 'C', 'D', 'E', 'F', 'G', 'H', 'K', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'X', 'Y', 'Z'), 0.4724724724724725, 0.4944944944944945), (('B', 'C', 'D', 'E', 'G', 'H', 'K', 'M', 'N', 'O', 'Q', 'R', 'S', 'U', 'X', 'Z'), 0.45645645645645644, 0.4724724724724725), (('F', 'P', 'T', 'V', 'Y'), 0.4364364364364364, 0.4724724724724725), (('B', 'C', 'D', 'E', 'G', 'H', 'K', 'N', 'O', 'Q', 'R', 'S', 'U', 'X', 'Z'), 0.4174174174174174, 0.45645645645645644), (('B', 'C', 'D', 'E', 'G', 'H', 'K', 'N', 'O', 'Q', 'R', 'S', 'X', 'Z'), 0.4134134134134134, 0.4174174174174174), (('B', 'D', 'G', 'H', 'K', 'N', 'O', 'Q', 'R', 'S', 'X', 'Z'), 0.4094094094094094, 0.4134134134134134), (('F', 'T', 'V', 'Y'), 0.4024024024024024, 0.4364364364364364), (('B', 'D', 'G', 'H', 'K', 'O', 'Q', 'R', 'S', 'X', 'Z'), 0.3863863863863864, 0.4094094094094094), (('B', 'G', 'H', 'K', 'O', 'Q', 'R', 'S', 'X', 'Z'), 0.3813813813813814, 0.3863863863863864), (('B', 'G', 'H', 'O', 'Q', 'R', 'S', 'X', 'Z'), 0.3733733733733734, 0.3813813813813814), (('F', 'T', 'Y'), 0.36036036036036034, 0.4024024024024024), (('C', 'E'), 0.3563563563563564, 0.4134134134134134), (('T', 'Y'), 0.3533533533533533, 0.36036036036036034), (('B', 'R', 'S', 'X', 'Z'), 0.35135135135135137, 0.3733733733733734), (('G', 'H', 'O', 'Q'), 0.34134134134134136, 0.3733733733733734), (('B', 'S', 'X', 'Z'), 0.32232232232232233, 0.35135135135135137), (('B', 'S', 'X'), 0.27627627627627627, 0.32232232232232233), (('G', 'H', 'O'), 0.26426426426426425, 0.34134134134134136), (('G', 'O'), 0.25725725725725723, 0.26426426426426425), (('S', 'X'), 0.15615615615615616, 0.27627627627627627), (('W',), 0, 0.5235235235235235), (('A',), 0, 0.5165165165165165), (('L',), 0, 0.5115115115115115), (('J',), 0, 0.4794794794794795), (('I',), 0, 0.4794794794794795), (('M',), 0, 0.45645645645645644), (('P',), 0, 0.4364364364364364), (('U',), 0, 0.4174174174174174), (('N',), 0, 0.4094094094094094), (('V',), 0, 0.4024024024024024), (('D',), 0, 0.3863863863863864), (('K',), 0, 0.3813813813813814), (('F',), 0, 0.36036036036036034), (('E',), 0, 0.3563563563563564), (('C',), 0, 0.3563563563563564), (('Y',), 0, 0.3533533533533533), (('T',), 0, 0.3533533533533533), (('R',), 0, 0.35135135135135137), (('Q',), 0, 0.34134134134134136), (('Z',), 0, 0.32232232232232233), (('B',), 0, 0.27627627627627627), (('H',), 0, 0.26426426426426425), (('G',), 0, 0.25725725725725723), (('O',), 0, 0.25725725725725723), (('S',), 0, 0.15615615615615616), (('X',), 0, 0.15615615615615616)]\n",
    "\n",
    "branch_defs = [\n",
    "    (tuple(map(lambda l: letter_to_class[l], trip[0])), trip[1], trip[2])\n",
    "    for trip in branch_defs\n",
    "]\n",
    "\n",
    "input_shape = (16,)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "229b0774",
   "metadata": {},
   "outputs": [],
   "source": [
    "sde = sdes.VariancePreservingSDE(0.1, 20, input_shape)\n",
    "t_limit = 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "2e9b0896",
   "metadata": {},
   "outputs": [],
   "source": [
    "out_path = \"/home/anon/branched_diffusion/figures/letters_efficiency\"\n",
    "\n",
    "os.makedirs(out_path, exist_ok=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6869e986",
   "metadata": {},
   "source": [
    "### Create data loader"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "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)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "55ea3306",
   "metadata": {},
   "source": [
    "### Import model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "d13434f5",
   "metadata": {},
   "outputs": [],
   "source": [
    "model = model_util.load_model(\n",
    "    table_dnn.MultitaskTabularNet,\n",
    "    os.path.join(models_base_path, \"letters_continuous_allletters/1/epoch_100_ckpt.pth\")\n",
    ").to(DEVICE)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c5f55c8b",
   "metadata": {},
   "source": [
    "### Generating multiple digit classes"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "c1e50cd6",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: A\n",
      "Sampling class: B\n",
      "Sampling class: C\n",
      "Sampling class: D\n",
      "Sampling class: E\n",
      "Sampling class: F\n",
      "Sampling class: G\n",
      "Sampling class: H\n",
      "Sampling class: I\n",
      "Sampling class: J\n",
      "Sampling class: K\n",
      "Sampling class: L\n",
      "Sampling class: M\n",
      "Sampling class: N\n",
      "Sampling class: O\n",
      "Sampling class: P\n",
      "Sampling class: Q\n",
      "Sampling class: R\n",
      "Sampling class: S\n",
      "Sampling class: T\n",
      "Sampling class: U\n",
      "Sampling class: V\n",
      "Sampling class: W\n",
      "Sampling class: X\n",
      "Sampling class: Y\n",
      "Sampling class: Z\n",
      "Total time taken: 111s\n",
      "Sampling class: A\n",
      "Sampling class: B\n",
      "Sampling class: C\n",
      "Sampling class: D\n",
      "Sampling class: E\n",
      "Sampling class: F\n",
      "Sampling class: G\n",
      "Sampling class: H\n",
      "Sampling class: I\n",
      "Sampling class: J\n",
      "Sampling class: K\n",
      "Sampling class: L\n",
      "Sampling class: M\n",
      "Sampling class: N\n",
      "Sampling class: O\n",
      "Sampling class: P\n",
      "Sampling class: Q\n",
      "Sampling class: R\n",
      "Sampling class: S\n",
      "Sampling class: T\n",
      "Sampling class: U\n",
      "Sampling class: V\n",
      "Sampling class: W\n",
      "Sampling class: X\n",
      "Sampling class: Y\n",
      "Sampling class: Z\n",
      "Total time taken: 110s\n",
      "Sampling class: A\n",
      "Sampling class: B\n",
      "Sampling class: C\n",
      "Sampling class: D\n",
      "Sampling class: E\n",
      "Sampling class: F\n",
      "Sampling class: G\n",
      "Sampling class: H\n",
      "Sampling class: I\n",
      "Sampling class: J\n",
      "Sampling class: K\n",
      "Sampling class: L\n",
      "Sampling class: M\n",
      "Sampling class: N\n",
      "Sampling class: O\n",
      "Sampling class: P\n",
      "Sampling class: Q\n",
      "Sampling class: R\n",
      "Sampling class: S\n",
      "Sampling class: T\n",
      "Sampling class: U\n",
      "Sampling class: V\n",
      "Sampling class: W\n",
      "Sampling class: X\n",
      "Sampling class: Y\n",
      "Sampling class: Z\n",
      "Total time taken: 111s\n",
      "Sampling class: A\n",
      "Sampling class: B\n",
      "Sampling class: C\n",
      "Sampling class: D\n",
      "Sampling class: E\n",
      "Sampling class: F\n",
      "Sampling class: G\n",
      "Sampling class: H\n",
      "Sampling class: I\n",
      "Sampling class: J\n",
      "Sampling class: K\n",
      "Sampling class: L\n",
      "Sampling class: M\n",
      "Sampling class: N\n",
      "Sampling class: O\n",
      "Sampling class: P\n",
      "Sampling class: Q\n",
      "Sampling class: R\n",
      "Sampling class: S\n",
      "Sampling class: T\n",
      "Sampling class: U\n",
      "Sampling class: V\n",
      "Sampling class: W\n",
      "Sampling class: X\n",
      "Sampling class: Y\n",
      "Sampling class: Z\n",
      "Total time taken: 110s\n",
      "Sampling class: A\n",
      "Sampling class: B\n",
      "Sampling class: C\n",
      "Sampling class: D\n",
      "Sampling class: E\n",
      "Sampling class: F\n",
      "Sampling class: G\n",
      "Sampling class: H\n",
      "Sampling class: I\n",
      "Sampling class: J\n",
      "Sampling class: K\n",
      "Sampling class: L\n",
      "Sampling class: M\n",
      "Sampling class: N\n",
      "Sampling class: O\n",
      "Sampling class: P\n",
      "Sampling class: Q\n",
      "Sampling class: R\n",
      "Sampling class: S\n",
      "Sampling class: T\n",
      "Sampling class: U\n",
      "Sampling class: V\n",
      "Sampling class: W\n",
      "Sampling class: X\n",
      "Sampling class: Y\n",
      "Sampling class: Z\n",
      "Total time taken: 110s\n",
      "Sampling class: A\n",
      "Sampling class: B\n",
      "Sampling class: C\n",
      "Sampling class: D\n",
      "Sampling class: E\n",
      "Sampling class: F\n",
      "Sampling class: G\n",
      "Sampling class: H\n",
      "Sampling class: I\n",
      "Sampling class: J\n",
      "Sampling class: K\n",
      "Sampling class: L\n",
      "Sampling class: M\n",
      "Sampling class: N\n",
      "Sampling class: O\n",
      "Sampling class: P\n",
      "Sampling class: Q\n",
      "Sampling class: R\n",
      "Sampling class: S\n",
      "Sampling class: T\n",
      "Sampling class: U\n",
      "Sampling class: V\n",
      "Sampling class: W\n",
      "Sampling class: X\n",
      "Sampling class: Y\n",
      "Sampling class: Z\n",
      "Total time taken: 109s\n",
      "Sampling class: A\n",
      "Sampling class: B\n",
      "Sampling class: C\n",
      "Sampling class: D\n",
      "Sampling class: E\n",
      "Sampling class: F\n",
      "Sampling class: G\n",
      "Sampling class: H\n",
      "Sampling class: I\n",
      "Sampling class: J\n",
      "Sampling class: K\n",
      "Sampling class: L\n",
      "Sampling class: M\n",
      "Sampling class: N\n",
      "Sampling class: O\n",
      "Sampling class: P\n",
      "Sampling class: Q\n",
      "Sampling class: R\n",
      "Sampling class: S\n",
      "Sampling class: T\n",
      "Sampling class: U\n",
      "Sampling class: V\n",
      "Sampling class: W\n",
      "Sampling class: X\n",
      "Sampling class: Y\n",
      "Sampling class: Z\n",
      "Total time taken: 110s\n",
      "Sampling class: A\n",
      "Sampling class: B\n",
      "Sampling class: C\n",
      "Sampling class: D\n",
      "Sampling class: E\n",
      "Sampling class: F\n",
      "Sampling class: G\n",
      "Sampling class: H\n",
      "Sampling class: I\n",
      "Sampling class: J\n",
      "Sampling class: K\n",
      "Sampling class: L\n",
      "Sampling class: M\n",
      "Sampling class: N\n",
      "Sampling class: O\n",
      "Sampling class: P\n",
      "Sampling class: Q\n",
      "Sampling class: R\n",
      "Sampling class: S\n",
      "Sampling class: T\n",
      "Sampling class: U\n",
      "Sampling class: V\n",
      "Sampling class: W\n",
      "Sampling class: X\n",
      "Sampling class: Y\n",
      "Sampling class: Z\n",
      "Total time taken: 110s\n",
      "Sampling class: A\n",
      "Sampling class: B\n",
      "Sampling class: C\n",
      "Sampling class: D\n",
      "Sampling class: E\n",
      "Sampling class: F\n",
      "Sampling class: G\n",
      "Sampling class: H\n",
      "Sampling class: I\n",
      "Sampling class: J\n",
      "Sampling class: K\n",
      "Sampling class: L\n",
      "Sampling class: M\n",
      "Sampling class: N\n",
      "Sampling class: O\n",
      "Sampling class: P\n",
      "Sampling class: Q\n",
      "Sampling class: R\n",
      "Sampling class: S\n",
      "Sampling class: T\n",
      "Sampling class: U\n",
      "Sampling class: V\n",
      "Sampling class: W\n",
      "Sampling class: X\n",
      "Sampling class: Y\n",
      "Sampling class: Z\n",
      "Total time taken: 110s\n",
      "Sampling class: A\n",
      "Sampling class: B\n",
      "Sampling class: C\n",
      "Sampling class: D\n",
      "Sampling class: E\n",
      "Sampling class: F\n",
      "Sampling class: G\n",
      "Sampling class: H\n",
      "Sampling class: I\n",
      "Sampling class: J\n",
      "Sampling class: K\n",
      "Sampling class: L\n",
      "Sampling class: M\n",
      "Sampling class: N\n",
      "Sampling class: O\n",
      "Sampling class: P\n",
      "Sampling class: Q\n",
      "Sampling class: R\n",
      "Sampling class: S\n",
      "Sampling class: T\n",
      "Sampling class: U\n",
      "Sampling class: V\n",
      "Sampling class: W\n",
      "Sampling class: X\n",
      "Sampling class: Y\n",
      "Sampling class: Z\n",
      "Total time taken: 110s\n",
      "Average time taken: 110.42 +/- 0.14\n"
     ]
    }
   ],
   "source": [
    "# Sample each class individually without taking advantage of branches\n",
    "num_linear_trials = 10\n",
    "linear_times = []\n",
    "for _ in range(num_linear_trials):\n",
    "    time_a = time.time()\n",
    "    for class_to_sample in classes:\n",
    "        print(\"Sampling class: %s\" % class_to_letter[class_to_sample])\n",
    "        generate.generate_continuous_branched_samples(\n",
    "            model, sde, class_to_sample, class_time_to_branch_tensor,\n",
    "            sampler=\"pc\", t_limit=t_limit, num_steps=1000\n",
    "        )\n",
    "    time_b = time.time()\n",
    "    time_taken = time_b - time_a\n",
    "    linear_times.append(time_taken)\n",
    "    print(\"Total time taken: %ds\" % time_taken)\n",
    "print(\"Average time taken: %.2f +/- %.2f\" % (np.mean(linear_times), np.std(linear_times) / np.sqrt(len(linear_times))))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "0de8f9d2",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling branch 1/51\n",
      "Sampling branch 2/51\n",
      "Sampling branch 3/51\n",
      "Sampling branch 4/51\n",
      "Sampling branch 5/51\n",
      "Sampling branch 6/51\n",
      "Sampling branch 7/51\n",
      "Sampling branch 8/51\n",
      "Sampling branch 9/51\n",
      "Sampling branch 10/51\n",
      "Sampling branch 11/51\n",
      "Sampling branch 12/51\n",
      "Sampling branch 13/51\n",
      "Sampling branch 14/51\n",
      "Sampling branch 15/51\n",
      "Sampling branch 16/51\n",
      "Sampling branch 17/51\n",
      "Sampling branch 18/51\n",
      "Sampling branch 19/51\n",
      "Sampling branch 20/51\n",
      "Sampling branch 21/51\n",
      "Sampling branch 22/51\n",
      "Sampling branch 23/51\n",
      "Sampling branch 24/51\n",
      "Sampling branch 25/51\n",
      "Sampling branch 26/51\n",
      "Sampling branch 27/51\n",
      "Sampling branch 28/51\n",
      "Sampling branch 29/51\n",
      "Sampling branch 30/51\n",
      "Sampling branch 31/51\n",
      "Sampling branch 32/51\n",
      "Sampling branch 33/51\n",
      "Sampling branch 34/51\n",
      "Sampling branch 35/51\n",
      "Sampling branch 36/51\n",
      "Sampling branch 37/51\n",
      "Sampling branch 38/51\n",
      "Sampling branch 39/51\n",
      "Sampling branch 40/51\n",
      "Sampling branch 41/51\n",
      "Sampling branch 42/51\n",
      "Sampling branch 43/51\n",
      "Sampling branch 44/51\n",
      "Sampling branch 45/51\n",
      "Sampling branch 46/51\n",
      "Sampling branch 47/51\n",
      "Sampling branch 48/51\n",
      "Sampling branch 49/51\n",
      "Sampling branch 50/51\n",
      "Sampling branch 51/51\n",
      "Total time taken: 67s\n",
      "Sampling branch 1/51\n",
      "Sampling branch 2/51\n",
      "Sampling branch 3/51\n",
      "Sampling branch 4/51\n",
      "Sampling branch 5/51\n",
      "Sampling branch 6/51\n",
      "Sampling branch 7/51\n",
      "Sampling branch 8/51\n",
      "Sampling branch 9/51\n",
      "Sampling branch 10/51\n",
      "Sampling branch 11/51\n",
      "Sampling branch 12/51\n",
      "Sampling branch 13/51\n",
      "Sampling branch 14/51\n",
      "Sampling branch 15/51\n",
      "Sampling branch 16/51\n",
      "Sampling branch 17/51\n",
      "Sampling branch 18/51\n",
      "Sampling branch 19/51\n",
      "Sampling branch 20/51\n",
      "Sampling branch 21/51\n",
      "Sampling branch 22/51\n",
      "Sampling branch 23/51\n",
      "Sampling branch 24/51\n",
      "Sampling branch 25/51\n",
      "Sampling branch 26/51\n",
      "Sampling branch 27/51\n",
      "Sampling branch 28/51\n",
      "Sampling branch 29/51\n",
      "Sampling branch 30/51\n",
      "Sampling branch 31/51\n",
      "Sampling branch 32/51\n",
      "Sampling branch 33/51\n",
      "Sampling branch 34/51\n",
      "Sampling branch 35/51\n",
      "Sampling branch 36/51\n",
      "Sampling branch 37/51\n",
      "Sampling branch 38/51\n",
      "Sampling branch 39/51\n",
      "Sampling branch 40/51\n",
      "Sampling branch 41/51\n",
      "Sampling branch 42/51\n",
      "Sampling branch 43/51\n",
      "Sampling branch 44/51\n",
      "Sampling branch 45/51\n",
      "Sampling branch 46/51\n",
      "Sampling branch 47/51\n",
      "Sampling branch 48/51\n",
      "Sampling branch 49/51\n",
      "Sampling branch 50/51\n",
      "Sampling branch 51/51\n",
      "Total time taken: 67s\n",
      "Sampling branch 1/51\n",
      "Sampling branch 2/51\n",
      "Sampling branch 3/51\n",
      "Sampling branch 4/51\n",
      "Sampling branch 5/51\n",
      "Sampling branch 6/51\n",
      "Sampling branch 7/51\n",
      "Sampling branch 8/51\n",
      "Sampling branch 9/51\n",
      "Sampling branch 10/51\n",
      "Sampling branch 11/51\n",
      "Sampling branch 12/51\n",
      "Sampling branch 13/51\n",
      "Sampling branch 14/51\n",
      "Sampling branch 15/51\n",
      "Sampling branch 16/51\n",
      "Sampling branch 17/51\n",
      "Sampling branch 18/51\n",
      "Sampling branch 19/51\n",
      "Sampling branch 20/51\n",
      "Sampling branch 21/51\n",
      "Sampling branch 22/51\n",
      "Sampling branch 23/51\n",
      "Sampling branch 24/51\n",
      "Sampling branch 25/51\n",
      "Sampling branch 26/51\n",
      "Sampling branch 27/51\n",
      "Sampling branch 28/51\n",
      "Sampling branch 29/51\n",
      "Sampling branch 30/51\n",
      "Sampling branch 31/51\n",
      "Sampling branch 32/51\n",
      "Sampling branch 33/51\n",
      "Sampling branch 34/51\n",
      "Sampling branch 35/51\n",
      "Sampling branch 36/51\n",
      "Sampling branch 37/51\n",
      "Sampling branch 38/51\n",
      "Sampling branch 39/51\n",
      "Sampling branch 40/51\n",
      "Sampling branch 41/51\n",
      "Sampling branch 42/51\n",
      "Sampling branch 43/51\n",
      "Sampling branch 44/51\n",
      "Sampling branch 45/51\n",
      "Sampling branch 46/51\n",
      "Sampling branch 47/51\n",
      "Sampling branch 48/51\n",
      "Sampling branch 49/51\n",
      "Sampling branch 50/51\n",
      "Sampling branch 51/51\n",
      "Total time taken: 67s\n",
      "Sampling branch 1/51\n",
      "Sampling branch 2/51\n",
      "Sampling branch 3/51\n",
      "Sampling branch 4/51\n",
      "Sampling branch 5/51\n",
      "Sampling branch 6/51\n",
      "Sampling branch 7/51\n",
      "Sampling branch 8/51\n",
      "Sampling branch 9/51\n",
      "Sampling branch 10/51\n",
      "Sampling branch 11/51\n",
      "Sampling branch 12/51\n",
      "Sampling branch 13/51\n",
      "Sampling branch 14/51\n",
      "Sampling branch 15/51\n",
      "Sampling branch 16/51\n",
      "Sampling branch 17/51\n",
      "Sampling branch 18/51\n",
      "Sampling branch 19/51\n",
      "Sampling branch 20/51\n",
      "Sampling branch 21/51\n",
      "Sampling branch 22/51\n",
      "Sampling branch 23/51\n",
      "Sampling branch 24/51\n",
      "Sampling branch 25/51\n",
      "Sampling branch 26/51\n",
      "Sampling branch 27/51\n",
      "Sampling branch 28/51\n",
      "Sampling branch 29/51\n",
      "Sampling branch 30/51\n",
      "Sampling branch 31/51\n",
      "Sampling branch 32/51\n",
      "Sampling branch 33/51\n",
      "Sampling branch 34/51\n",
      "Sampling branch 35/51\n",
      "Sampling branch 36/51\n",
      "Sampling branch 37/51\n",
      "Sampling branch 38/51\n",
      "Sampling branch 39/51\n",
      "Sampling branch 40/51\n",
      "Sampling branch 41/51\n",
      "Sampling branch 42/51\n",
      "Sampling branch 43/51\n",
      "Sampling branch 44/51\n",
      "Sampling branch 45/51\n",
      "Sampling branch 46/51\n",
      "Sampling branch 47/51\n",
      "Sampling branch 48/51\n",
      "Sampling branch 49/51\n",
      "Sampling branch 50/51\n",
      "Sampling branch 51/51\n",
      "Total time taken: 67s\n",
      "Sampling branch 1/51\n",
      "Sampling branch 2/51\n",
      "Sampling branch 3/51\n",
      "Sampling branch 4/51\n",
      "Sampling branch 5/51\n",
      "Sampling branch 6/51\n",
      "Sampling branch 7/51\n",
      "Sampling branch 8/51\n",
      "Sampling branch 9/51\n",
      "Sampling branch 10/51\n",
      "Sampling branch 11/51\n",
      "Sampling branch 12/51\n",
      "Sampling branch 13/51\n",
      "Sampling branch 14/51\n",
      "Sampling branch 15/51\n",
      "Sampling branch 16/51\n",
      "Sampling branch 17/51\n",
      "Sampling branch 18/51\n",
      "Sampling branch 19/51\n",
      "Sampling branch 20/51\n",
      "Sampling branch 21/51\n",
      "Sampling branch 22/51\n",
      "Sampling branch 23/51\n",
      "Sampling branch 24/51\n",
      "Sampling branch 25/51\n",
      "Sampling branch 26/51\n",
      "Sampling branch 27/51\n",
      "Sampling branch 28/51\n",
      "Sampling branch 29/51\n",
      "Sampling branch 30/51\n",
      "Sampling branch 31/51\n",
      "Sampling branch 32/51\n",
      "Sampling branch 33/51\n",
      "Sampling branch 34/51\n",
      "Sampling branch 35/51\n",
      "Sampling branch 36/51\n",
      "Sampling branch 37/51\n",
      "Sampling branch 38/51\n",
      "Sampling branch 39/51\n",
      "Sampling branch 40/51\n",
      "Sampling branch 41/51\n",
      "Sampling branch 42/51\n",
      "Sampling branch 43/51\n",
      "Sampling branch 44/51\n",
      "Sampling branch 45/51\n",
      "Sampling branch 46/51\n",
      "Sampling branch 47/51\n",
      "Sampling branch 48/51\n",
      "Sampling branch 49/51\n",
      "Sampling branch 50/51\n",
      "Sampling branch 51/51\n",
      "Total time taken: 67s\n",
      "Sampling branch 1/51\n",
      "Sampling branch 2/51\n",
      "Sampling branch 3/51\n",
      "Sampling branch 4/51\n",
      "Sampling branch 5/51\n",
      "Sampling branch 6/51\n",
      "Sampling branch 7/51\n",
      "Sampling branch 8/51\n",
      "Sampling branch 9/51\n",
      "Sampling branch 10/51\n",
      "Sampling branch 11/51\n",
      "Sampling branch 12/51\n",
      "Sampling branch 13/51\n",
      "Sampling branch 14/51\n",
      "Sampling branch 15/51\n",
      "Sampling branch 16/51\n",
      "Sampling branch 17/51\n",
      "Sampling branch 18/51\n",
      "Sampling branch 19/51\n",
      "Sampling branch 20/51\n",
      "Sampling branch 21/51\n",
      "Sampling branch 22/51\n",
      "Sampling branch 23/51\n",
      "Sampling branch 24/51\n",
      "Sampling branch 25/51\n",
      "Sampling branch 26/51\n",
      "Sampling branch 27/51\n",
      "Sampling branch 28/51\n",
      "Sampling branch 29/51\n",
      "Sampling branch 30/51\n",
      "Sampling branch 31/51\n",
      "Sampling branch 32/51\n",
      "Sampling branch 33/51\n",
      "Sampling branch 34/51\n",
      "Sampling branch 35/51\n",
      "Sampling branch 36/51\n",
      "Sampling branch 37/51\n",
      "Sampling branch 38/51\n",
      "Sampling branch 39/51\n",
      "Sampling branch 40/51\n",
      "Sampling branch 41/51\n",
      "Sampling branch 42/51\n",
      "Sampling branch 43/51\n",
      "Sampling branch 44/51\n",
      "Sampling branch 45/51\n",
      "Sampling branch 46/51\n",
      "Sampling branch 47/51\n",
      "Sampling branch 48/51\n",
      "Sampling branch 49/51\n",
      "Sampling branch 50/51\n",
      "Sampling branch 51/51\n",
      "Total time taken: 67s\n",
      "Sampling branch 1/51\n",
      "Sampling branch 2/51\n",
      "Sampling branch 3/51\n",
      "Sampling branch 4/51\n",
      "Sampling branch 5/51\n",
      "Sampling branch 6/51\n",
      "Sampling branch 7/51\n",
      "Sampling branch 8/51\n",
      "Sampling branch 9/51\n",
      "Sampling branch 10/51\n",
      "Sampling branch 11/51\n",
      "Sampling branch 12/51\n",
      "Sampling branch 13/51\n",
      "Sampling branch 14/51\n",
      "Sampling branch 15/51\n",
      "Sampling branch 16/51\n",
      "Sampling branch 17/51\n",
      "Sampling branch 18/51\n",
      "Sampling branch 19/51\n",
      "Sampling branch 20/51\n",
      "Sampling branch 21/51\n",
      "Sampling branch 22/51\n",
      "Sampling branch 23/51\n",
      "Sampling branch 24/51\n",
      "Sampling branch 25/51\n",
      "Sampling branch 26/51\n",
      "Sampling branch 27/51\n",
      "Sampling branch 28/51\n",
      "Sampling branch 29/51\n",
      "Sampling branch 30/51\n",
      "Sampling branch 31/51\n",
      "Sampling branch 32/51\n",
      "Sampling branch 33/51\n",
      "Sampling branch 34/51\n",
      "Sampling branch 35/51\n",
      "Sampling branch 36/51\n",
      "Sampling branch 37/51\n",
      "Sampling branch 38/51\n",
      "Sampling branch 39/51\n",
      "Sampling branch 40/51\n",
      "Sampling branch 41/51\n",
      "Sampling branch 42/51\n",
      "Sampling branch 43/51\n",
      "Sampling branch 44/51\n",
      "Sampling branch 45/51\n",
      "Sampling branch 46/51\n",
      "Sampling branch 47/51\n",
      "Sampling branch 48/51\n",
      "Sampling branch 49/51\n",
      "Sampling branch 50/51\n",
      "Sampling branch 51/51\n",
      "Total time taken: 67s\n",
      "Sampling branch 1/51\n",
      "Sampling branch 2/51\n",
      "Sampling branch 3/51\n",
      "Sampling branch 4/51\n",
      "Sampling branch 5/51\n",
      "Sampling branch 6/51\n",
      "Sampling branch 7/51\n",
      "Sampling branch 8/51\n",
      "Sampling branch 9/51\n",
      "Sampling branch 10/51\n",
      "Sampling branch 11/51\n",
      "Sampling branch 12/51\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling branch 13/51\n",
      "Sampling branch 14/51\n",
      "Sampling branch 15/51\n",
      "Sampling branch 16/51\n",
      "Sampling branch 17/51\n",
      "Sampling branch 18/51\n",
      "Sampling branch 19/51\n",
      "Sampling branch 20/51\n",
      "Sampling branch 21/51\n",
      "Sampling branch 22/51\n",
      "Sampling branch 23/51\n",
      "Sampling branch 24/51\n",
      "Sampling branch 25/51\n",
      "Sampling branch 26/51\n",
      "Sampling branch 27/51\n",
      "Sampling branch 28/51\n",
      "Sampling branch 29/51\n",
      "Sampling branch 30/51\n",
      "Sampling branch 31/51\n",
      "Sampling branch 32/51\n",
      "Sampling branch 33/51\n",
      "Sampling branch 34/51\n",
      "Sampling branch 35/51\n",
      "Sampling branch 36/51\n",
      "Sampling branch 37/51\n",
      "Sampling branch 38/51\n",
      "Sampling branch 39/51\n",
      "Sampling branch 40/51\n",
      "Sampling branch 41/51\n",
      "Sampling branch 42/51\n",
      "Sampling branch 43/51\n",
      "Sampling branch 44/51\n",
      "Sampling branch 45/51\n",
      "Sampling branch 46/51\n",
      "Sampling branch 47/51\n",
      "Sampling branch 48/51\n",
      "Sampling branch 49/51\n",
      "Sampling branch 50/51\n",
      "Sampling branch 51/51\n",
      "Total time taken: 67s\n",
      "Sampling branch 1/51\n",
      "Sampling branch 2/51\n",
      "Sampling branch 3/51\n",
      "Sampling branch 4/51\n",
      "Sampling branch 5/51\n",
      "Sampling branch 6/51\n",
      "Sampling branch 7/51\n",
      "Sampling branch 8/51\n",
      "Sampling branch 9/51\n",
      "Sampling branch 10/51\n",
      "Sampling branch 11/51\n",
      "Sampling branch 12/51\n",
      "Sampling branch 13/51\n",
      "Sampling branch 14/51\n",
      "Sampling branch 15/51\n",
      "Sampling branch 16/51\n",
      "Sampling branch 17/51\n",
      "Sampling branch 18/51\n",
      "Sampling branch 19/51\n",
      "Sampling branch 20/51\n",
      "Sampling branch 21/51\n",
      "Sampling branch 22/51\n",
      "Sampling branch 23/51\n",
      "Sampling branch 24/51\n",
      "Sampling branch 25/51\n",
      "Sampling branch 26/51\n",
      "Sampling branch 27/51\n",
      "Sampling branch 28/51\n",
      "Sampling branch 29/51\n",
      "Sampling branch 30/51\n",
      "Sampling branch 31/51\n",
      "Sampling branch 32/51\n",
      "Sampling branch 33/51\n",
      "Sampling branch 34/51\n",
      "Sampling branch 35/51\n",
      "Sampling branch 36/51\n",
      "Sampling branch 37/51\n",
      "Sampling branch 38/51\n",
      "Sampling branch 39/51\n",
      "Sampling branch 40/51\n",
      "Sampling branch 41/51\n",
      "Sampling branch 42/51\n",
      "Sampling branch 43/51\n",
      "Sampling branch 44/51\n",
      "Sampling branch 45/51\n",
      "Sampling branch 46/51\n",
      "Sampling branch 47/51\n",
      "Sampling branch 48/51\n",
      "Sampling branch 49/51\n",
      "Sampling branch 50/51\n",
      "Sampling branch 51/51\n",
      "Total time taken: 67s\n",
      "Sampling branch 1/51\n",
      "Sampling branch 2/51\n",
      "Sampling branch 3/51\n",
      "Sampling branch 4/51\n",
      "Sampling branch 5/51\n",
      "Sampling branch 6/51\n",
      "Sampling branch 7/51\n",
      "Sampling branch 8/51\n",
      "Sampling branch 9/51\n",
      "Sampling branch 10/51\n",
      "Sampling branch 11/51\n",
      "Sampling branch 12/51\n",
      "Sampling branch 13/51\n",
      "Sampling branch 14/51\n",
      "Sampling branch 15/51\n",
      "Sampling branch 16/51\n",
      "Sampling branch 17/51\n",
      "Sampling branch 18/51\n",
      "Sampling branch 19/51\n",
      "Sampling branch 20/51\n",
      "Sampling branch 21/51\n",
      "Sampling branch 22/51\n",
      "Sampling branch 23/51\n",
      "Sampling branch 24/51\n",
      "Sampling branch 25/51\n",
      "Sampling branch 26/51\n",
      "Sampling branch 27/51\n",
      "Sampling branch 28/51\n",
      "Sampling branch 29/51\n",
      "Sampling branch 30/51\n",
      "Sampling branch 31/51\n",
      "Sampling branch 32/51\n",
      "Sampling branch 33/51\n",
      "Sampling branch 34/51\n",
      "Sampling branch 35/51\n",
      "Sampling branch 36/51\n",
      "Sampling branch 37/51\n",
      "Sampling branch 38/51\n",
      "Sampling branch 39/51\n",
      "Sampling branch 40/51\n",
      "Sampling branch 41/51\n",
      "Sampling branch 42/51\n",
      "Sampling branch 43/51\n",
      "Sampling branch 44/51\n",
      "Sampling branch 45/51\n",
      "Sampling branch 46/51\n",
      "Sampling branch 47/51\n",
      "Sampling branch 48/51\n",
      "Sampling branch 49/51\n",
      "Sampling branch 50/51\n",
      "Sampling branch 51/51\n",
      "Total time taken: 67s\n",
      "Average time taken: 67.54 +/- 0.04\n"
     ]
    }
   ],
   "source": [
    "# Sample each digit by taking advantage of branches\n",
    "num_branched_trials = 10\n",
    "branched_times = []\n",
    "\n",
    "# Sort the branches by starting time point (in reverse order), and generate along those\n",
    "# branches, caching results; this guarantees that we will always find a cached batch\n",
    "# (other than the first one)\n",
    "cache = {}\n",
    "sorted_branch_defs = sorted(branch_defs, key=(lambda t: -t[1]))\n",
    "\n",
    "for _ in range(num_branched_trials):\n",
    "    time_a = time.time()\n",
    "    # First branch\n",
    "    print(\"Sampling branch 1/%d\" % len(sorted_branch_defs))\n",
    "    branch_def = sorted_branch_defs[0]\n",
    "    samples = generate.generate_continuous_branched_samples(\n",
    "        # Specify arbitrary class\n",
    "        model, sde, branch_def[0][0], class_time_to_branch_tensor,\n",
    "        sampler=\"pc\", t_limit=branch_def[2], t_start=branch_def[1],\n",
    "        num_steps=int(1000 * (branch_def[2] - branch_def[1]))\n",
    "    )\n",
    "    for class_i in branch_def[0]:\n",
    "        cache[class_i] = (branch_def[1], samples)\n",
    "\n",
    "    for i, branch_def in enumerate(sorted_branch_defs[1:]):\n",
    "        print(\"Sampling branch %d/%d\" % (i + 2, len(sorted_branch_defs)))\n",
    "        cached_time, cached_samples = cache[branch_def[0][0]]\n",
    "        assert cached_time == branch_def[2]\n",
    "        samples = generate.generate_continuous_branched_samples(\n",
    "            model, sde, branch_def[0][0], class_time_to_branch_tensor,\n",
    "            sampler=\"pc\", t_limit=branch_def[2], t_start=branch_def[1],\n",
    "            num_steps=int(1000 * (branch_def[2] - branch_def[1])),\n",
    "            initial_samples=cached_samples\n",
    "        )\n",
    "        for class_i in branch_def[0]:\n",
    "            cache[class_i] = (branch_def[1], samples)\n",
    "\n",
    "    time_b = time.time()\n",
    "    time_taken = time_b - time_a\n",
    "    branched_times.append(time_taken)\n",
    "    print(\"Total time taken: %ds\" % time_taken)\n",
    "print(\"Average time taken: %.2f +/- %.2f\" % (np.mean(branched_times), np.std(branched_times) / np.sqrt(len(branched_times))))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "26111439",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 576x576 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(figsize=(8, 8))\n",
    "labels = [\"Linear\", \"Branched\"]\n",
    "times = [np.mean(linear_times), np.mean(branched_times)]\n",
    "errors = [np.std(linear_times) / np.sqrt(num_linear_trials), np.std(branched_times) / np.sqrt(num_branched_trials)]\n",
    "ax.bar(labels, times, color=\"mediumorchid\")\n",
    "ax.errorbar(labels, times, yerr=errors, fmt=\"none\", color=\"darkmagenta\")\n",
    "ax.set_xlabel(\"Time to generate one batch of each class\")\n",
    "ax.set_ylabel(\"Seconds\")\n",
    "plt.show()\n",
    "fig.savefig(\n",
    "    os.path.join(out_path, \"letters_efficiency.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
}
