{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### This notebook has been heavily adapted from https://github.com/nec-research/tf-imle"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "# choose a particular GPU\n",
    "# os.environ[\"CUDA_VISIBLE_DEVICES\"] = \"2\"\n",
    "\n",
    "import tensorflow as tf\n",
    "\n",
    "from tensorflow.keras.layers import Dense, Flatten\n",
    "from tensorflow.keras import Model\n",
    "from tensorflow.keras.utils import plot_model\n",
    "\n",
    "import tensorflow_probability as tfp\n",
    "\n",
    "import numpy as np\n",
    "\n",
    "%matplotlib inline\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# physical_devices = tf.config.list_physical_devices('GPU') \n",
    "# tf.config.experimental.set_memory_growth(physical_devices[0], True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "class IdentityLayer(tf.keras.layers.Layer):\n",
    "  \n",
    "    def __init__(self, _k=10, _tau=10.0, _lambda=10.0):\n",
    "        super(IdentityLayer, self).__init__()\n",
    "        \n",
    "        self.k = _k\n",
    "        self._tau = _tau\n",
    "        self._lambda = _lambda\n",
    "        self.samples = None\n",
    "        self.gumbel_dist = tfp.distributions.Gumbel(loc=0.0, scale=1.0)\n",
    "        \n",
    "    @tf.function\n",
    "    def sample_gumbel(self, shape, eps=1e-20):\n",
    "        return self.gumbel_dist.sample(shape)\n",
    "    \n",
    "    @tf.function\n",
    "    def sample_gumbel_k(self, shape):\n",
    "        \n",
    "        s = tf.map_fn(fn=lambda t: tf.random.gamma(shape, 1.0/self.k,  self.k/t), \n",
    "                  elems=tf.constant([1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 9.0, 10.0]))\n",
    "        # now add the samples\n",
    "        s = tf.reduce_sum(s, 0)\n",
    "        # the log(m) term\n",
    "        s = s - tf.math.log(10.0)\n",
    "        # divide by k --> each s[c] has k samples whose sum is distributed as Gumbel(0, 1)\n",
    "        s = self._tau * (s / self.k)\n",
    "\n",
    "        return s\n",
    "       \n",
    "    @tf.custom_gradient\n",
    "    def imle_layer(self, logits, hard=False):\n",
    "\n",
    "        # toggle between Gumbel(0, 1) and sum-of-Gamma\n",
    "        #forward_sample = self.sample_gumbel(tf.shape(logits))\n",
    "        threshold = tf.expand_dims(tf.nn.top_k(logits, self.k, sorted=True)[0][:,-1], -1)\n",
    "        y_map = tf.cast(tf.greater_equal(logits, threshold), tf.float32)\n",
    "\n",
    "        def custom_grad(dy):\n",
    "            return dy, hard\n",
    "\n",
    "        return y_map, custom_grad\n",
    "\n",
    "    def call(self, logits, hard=False):\n",
    "        logits = logits + self.sample_gumbel_k(tf.shape(logits))\n",
    "        # Project onto the hyperplane\n",
    "        logits = logits - tf.reduce_mean(logits, 1)[:, None]\n",
    "        # Project onto the sphere\n",
    "        logits = logits / tf.norm(logits, axis=1)[:, None]\n",
    "        return self.imle_layer(logits, hard)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "PARAMS = {\n",
    "    \"batch_size\": 100,\n",
    "    \"data_dim\": 784,\n",
    "    \"M\": 20,\n",
    "    \"N\": 20,\n",
    "    \"nb_epoch\": 100, \n",
    "    \"epsilon_std\": 0.01,\n",
    "    \"anneal_rate\": 0.0003,\n",
    "    \"init_temperature\": 1.0,\n",
    "    \"min_temperature\": 0.5,\n",
    "    \"learning_rate\": 1e-3,\n",
    "    \"hard\": False,\n",
    "}\n",
    "\n",
    "class DiscreteVAE(tf.keras.Model):\n",
    "    \n",
    "    def __init__(self, params):\n",
    "        super(DiscreteVAE, self).__init__()\n",
    "        \n",
    "        self.params = params\n",
    "                \n",
    "        # encoder\n",
    "        self.enc_dense1 = tf.keras.layers.Dense(512, activation='relu')\n",
    "        self.enc_dense2 = tf.keras.layers.Dense(256, activation='relu')\n",
    "        self.enc_dense3 = tf.keras.layers.Dense(params[\"N\"]*params[\"M\"])\n",
    "        \n",
    "        # this is our new Gumbel layer\n",
    "        self.imleLayer = IdentityLayer(_k=10, _tau=10.0, _lambda=10.0)\n",
    "\n",
    "        # decoder\n",
    "        self.flatten = Flatten()\n",
    "        self.dec_dense1 = tf.keras.layers.Dense(256, activation='relu')\n",
    "        self.dec_dense2 = tf.keras.layers.Dense(512, activation='relu')\n",
    "        self.dec_dense3 = tf.keras.layers.Dense(params[\"data_dim\"])\n",
    "\n",
    "\n",
    "    def sample_gumbel(self, shape, eps=1e-20): \n",
    "        \"\"\"Sample from Gumbel(0, 1)\"\"\" \n",
    "        # test\n",
    "        #U = gumbel_dist.sample(shape)\n",
    "        U = tf.random.uniform(shape, minval=0, maxval=1)\n",
    "        return -tf.math.log(-tf.math.log(U + eps) + eps)\n",
    "    \n",
    "    def gumbel_softmax_sample(self, logits, temperature): \n",
    "        \"\"\" Draw a sample from the Gumbel-Softmax distribution\"\"\"\n",
    "        # logits: [batch_size, n_class] unnormalized log-probs\n",
    "        y = logits + self.sample_gumbel(tf.shape(logits))\n",
    "        return tf.nn.softmax(y / temperature)  \n",
    "\n",
    "    def gumbel_softmax(self, logits, temperature, hard=True):\n",
    "        \"\"\"\n",
    "        logits: [batch_size, n_class] unnormalized log-probs\n",
    "        temperature: non-negative scalar\n",
    "        hard: if True, take argmax, but differentiate w.r.t. soft sample y\n",
    "        \"\"\"\n",
    "        y = self.gumbel_softmax_sample(logits, temperature)\n",
    "        if hard: \n",
    "            # \n",
    "            y_hard = tf.cast(tf.equal(y, tf.reduce_max(y, 1, keepdims=True)),y.dtype)\n",
    "            y = tf.stop_gradient(y_hard - y) + y\n",
    "        return y\n",
    "    \n",
    "    def decoder(self, x):\n",
    "        # decoder\n",
    "        h = self.flatten(x)\n",
    "        h = self.dec_dense1(h)\n",
    "        h = self.dec_dense2(h)\n",
    "        h = self.dec_dense3(h)\n",
    "        return h\n",
    "\n",
    "    def call(self, x, tau, hard=False):\n",
    "        N = self.params[\"N\"]\n",
    "        M = self.params[\"M\"]\n",
    "\n",
    "        # encoder\n",
    "        x = self.enc_dense1(x)\n",
    "        x = self.enc_dense2(x)\n",
    "        x = self.enc_dense3(x)   # (batch, N*M)\n",
    "        logits_y = tf.reshape(x, [-1, M])   # (batch*N, M)\n",
    "\n",
    "        ###################################################################\n",
    "        ## here we toggle between methods #################################\n",
    "        # here we can switch between traditional and our method\n",
    "        # \"traditional\" Gumbel Softmax trick\n",
    "        #y = self.gumbel_softmax(logits=logits_y, temperature=tau, hard=False)\n",
    "        # IMLE approach -- note: we don't anneal so set temperature once at init\n",
    "        y = self.imleLayer(logits=logits_y, hard=False)\n",
    "        ###################################################################\n",
    "        \n",
    "        assert y.shape == (self.params[\"batch_size\"]*N, M)\n",
    "        y = tf.reshape(y, [-1, N, M])\n",
    "        self.sample_y = y\n",
    "\n",
    "        # decoder\n",
    "        logits_x = self.decoder(y)\n",
    "        return logits_y, logits_x\n",
    "\n",
    "\n",
    "def gumbel_loss(model, x, tau, hard=True):\n",
    "    M = 20\n",
    "    N = 20\n",
    "    data_dim = PARAMS['data_dim']\n",
    "    logits_y, logits_x = model(x, tau, hard)\n",
    "    \n",
    "    # cross-entropy\n",
    "    cross_ent = tf.nn.sigmoid_cross_entropy_with_logits(labels=x, logits=logits_x)\n",
    "    cross_ent = tf.math.reduce_sum(cross_ent, 1)\n",
    "    cross_ent = tf.math.reduce_mean(cross_ent, 0)\n",
    "    \n",
    "    # KL loss\n",
    "    q_y = tf.nn.softmax(logits_y)   # (batshsize*N, M)  softmax\n",
    "    log_q_y = tf.math.log(q_y + 1e-20)   # (batshsize*N, M)  \n",
    "    kl_tmp = tf.reshape(q_y*(log_q_y-tf.math.log(1.0/M)), [-1,N,M])  # (batch_size,N,K)\n",
    "    KL = tf.math.reduce_sum(kl_tmp, [1, 2])    # shape=(batch_size, 1)\n",
    "        \n",
    "    KL_mean = tf.math.reduce_mean(KL)\n",
    "    #print(\"**\", cross_ent.numpy(), KL_mean.numpy())\n",
    "    return cross_ent + KL_mean\n",
    "\n",
    "\n",
    "def compute_gradients(model, x, tau, hard):\n",
    "    with tf.GradientTape() as tape:\n",
    "        loss = gumbel_loss(model, x, tau, hard)\n",
    "    return tape.gradient(loss, model.trainable_variables), loss\n",
    "\n",
    "\n",
    "def apply_gradients(optimizer, gradients, variables):\n",
    "    optimizer.apply_gradients(zip(gradients, variables))\n",
    "\n",
    "\n",
    "def get_learning_rate(step, init=PARAMS[\"learning_rate\"]):\n",
    "    return tf.convert_to_tensor(init * pow(0.95, (step / 1000.)), dtype=tf.float32)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2022-09-29 02:05:21.300238: W tensorflow/core/platform/profile_utils/cpu_utils.cc:128] Failed to get CPU frequency: 0 Hz\n",
      "2022-09-29 02:05:21.347388: W tensorflow/python/util/util.cc:368] Sets are not currently considered sequences, but this may change in the future, so consider avoiding using them.\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 1 , TRAIN loss: 102.435585 , Temperature: 0.9997000449955004\n",
      "Eval Loss: 99.39311 \n",
      "\n",
      "Epoch: 2 , TRAIN loss: 89.74848 , Temperature: 0.9991004048785274\n",
      "Eval Loss: 89.475 \n",
      "\n",
      "Epoch: 3 , TRAIN loss: 88.06725 , Temperature: 0.9982016190284373\n",
      "Eval Loss: 85.51461 \n",
      "\n",
      "Epoch: 4 , TRAIN loss: 87.33134 , Temperature: 0.997004495503373\n",
      "Eval Loss: 83.8279 \n",
      "\n",
      "Epoch: 5 , TRAIN loss: 80.53149 , Temperature: 0.9955101098295706\n",
      "Eval Loss: 82.06636 \n",
      "\n",
      "Epoch: 6 , TRAIN loss: 80.56935 , Temperature: 0.9937198033910547\n",
      "Eval Loss: 80.67386 \n",
      "\n",
      "Epoch: 7 , TRAIN loss: 82.75039 , Temperature: 0.9916351814230984\n",
      "Eval Loss: 80.17215 \n",
      "\n",
      "Epoch: 8 , TRAIN loss: 81.33204 , Temperature: 0.9892581106136482\n",
      "Eval Loss: 79.635 \n",
      "\n",
      "Epoch: 9 , TRAIN loss: 79.27237 , Temperature: 0.9865907163177327\n",
      "Eval Loss: 78.97988 \n",
      "\n",
      "Epoch: 10 , TRAIN loss: 76.0836 , Temperature: 0.9836353793906725\n",
      "Eval Loss: 78.47988 \n",
      "\n",
      "Epoch: 11 , TRAIN loss: 80.245285 , Temperature: 0.9803947326466972\n",
      "Eval Loss: 78.1621 \n",
      "\n",
      "Epoch: 12 , TRAIN loss: 75.89484 , Temperature: 0.9768716569503434\n",
      "Eval Loss: 77.91544 \n",
      "\n",
      "Epoch: 13 , TRAIN loss: 76.795364 , Temperature: 0.9730692769487556\n",
      "Eval Loss: 77.171005 \n",
      "\n",
      "Epoch: 14 , TRAIN loss: 77.92663 , Temperature: 0.9689909564537397\n",
      "Eval Loss: 77.1945 \n",
      "\n",
      "Epoch: 15 , TRAIN loss: 77.28584 , Temperature: 0.964640293483123\n",
      "Eval Loss: 76.87686 \n",
      "\n",
      "Epoch: 16 , TRAIN loss: 76.52683 , Temperature: 0.9600211149716509\n",
      "Eval Loss: 76.919716 \n",
      "\n",
      "Epoch: 17 , TRAIN loss: 74.811104 , Temperature: 0.9551374711623026\n",
      "Eval Loss: 76.740364 \n",
      "\n",
      "Epoch: 18 , TRAIN loss: 73.66559 , Temperature: 0.9499936296895314\n",
      "Eval Loss: 76.2061 \n",
      "\n",
      "Epoch: 19 , TRAIN loss: 77.91887 , Temperature: 0.9445940693665233\n",
      "Eval Loss: 76.409935 \n",
      "\n",
      "Epoch: 20 , TRAIN loss: 74.66787 , Temperature: 0.9389434736891332\n",
      "Eval Loss: 76.5391 \n",
      "\n",
      "Epoch: 21 , TRAIN loss: 74.346344 , Temperature: 0.9330467240696794\n",
      "Eval Loss: 75.7782 \n",
      "\n",
      "Epoch: 22 , TRAIN loss: 80.342125 , Temperature: 0.9269088928142736\n",
      "Eval Loss: 76.11216 \n",
      "\n",
      "Epoch: 23 , TRAIN loss: 79.00188 , Temperature: 0.9205352358578188\n",
      "Eval Loss: 75.68276 \n",
      "\n",
      "Epoch: 24 , TRAIN loss: 75.74218 , Temperature: 0.9139311852712282\n",
      "Eval Loss: 76.1896 \n",
      "\n",
      "Epoch: 25 , TRAIN loss: 74.182465 , Temperature: 0.9071023415558017\n",
      "Eval Loss: 75.41245 \n",
      "\n",
      "Epoch: 26 , TRAIN loss: 78.45046 , Temperature: 0.9000544657400421\n",
      "Eval Loss: 75.35782 \n",
      "\n",
      "Epoch: 27 , TRAIN loss: 77.87514 , Temperature: 0.8927934712944959\n",
      "Eval Loss: 75.10194 \n",
      "\n",
      "Epoch: 28 , TRAIN loss: 74.464195 , Temperature: 0.8853254158804752\n",
      "Eval Loss: 75.49076 \n",
      "\n",
      "Epoch: 29 , TRAIN loss: 75.63541 , Temperature: 0.8776564929487385\n",
      "Eval Loss: 75.43032 \n",
      "\n",
      "Epoch: 30 , TRAIN loss: 73.42955 , Temperature: 0.8697930232043984\n",
      "Eval Loss: 74.92193 \n",
      "\n",
      "Epoch: 31 , TRAIN loss: 76.70047 , Temperature: 0.8617414459544691\n",
      "Eval Loss: 74.75726 \n",
      "\n",
      "Epoch: 32 , TRAIN loss: 77.25605 , Temperature: 0.8535083103545701\n",
      "Eval Loss: 74.89761 \n",
      "\n",
      "Epoch: 33 , TRAIN loss: 75.86187 , Temperature: 0.8451002665713722\n",
      "Eval Loss: 74.88129 \n",
      "\n",
      "Epoch: 34 , TRAIN loss: 70.92391 , Temperature: 0.8365240568773926\n",
      "Eval Loss: 74.71342 \n",
      "\n",
      "Epoch: 35 , TRAIN loss: 74.50814 , Temperature: 0.8277865066947337\n",
      "Eval Loss: 74.660194 \n",
      "\n",
      "Epoch: 36 , TRAIN loss: 76.14128 , Temperature: 0.8188945156043044\n",
      "Eval Loss: 74.569565 \n",
      "\n",
      "Epoch: 37 , TRAIN loss: 75.30998 , Temperature: 0.8098550483369699\n",
      "Eval Loss: 73.9898 \n",
      "\n",
      "Epoch: 38 , TRAIN loss: 74.88908 , Temperature: 0.8006751257629464\n",
      "Eval Loss: 74.69085 \n",
      "\n",
      "Epoch: 39 , TRAIN loss: 74.02948 , Temperature: 0.791361815895584\n",
      "Eval Loss: 74.56966 \n",
      "\n",
      "Epoch: 40 , TRAIN loss: 74.94245 , Temperature: 0.7819222249254774\n",
      "Eval Loss: 74.33116 \n",
      "\n",
      "Epoch: 41 , TRAIN loss: 74.752014 , Temperature: 0.7723634883006051\n",
      "Eval Loss: 73.96436 \n",
      "\n",
      "Epoch: 42 , TRAIN loss: 74.566605 , Temperature: 0.7626927618679156\n",
      "Eval Loss: 73.80967 \n",
      "\n",
      "Epoch: 43 , TRAIN loss: 75.38042 , Temperature: 0.7529172130914742\n",
      "Eval Loss: 74.33616 \n",
      "\n",
      "Epoch: 44 , TRAIN loss: 78.42018 , Temperature: 0.74304401236194\n",
      "Eval Loss: 73.86679 \n",
      "\n",
      "Epoch: 45 , TRAIN loss: 72.03265 , Temperature: 0.7330803244117685\n",
      "Eval Loss: 74.12672 \n",
      "\n",
      "Epoch: 46 , TRAIN loss: 73.57828 , Temperature: 0.7230332998501351\n",
      "Eval Loss: 74.019066 \n",
      "\n",
      "Epoch: 47 , TRAIN loss: 75.85057 , Temperature: 0.7129100668311394\n",
      "Eval Loss: 73.50869 \n",
      "\n",
      "Epoch: 48 , TRAIN loss: 70.04462 , Temperature: 0.7027177228683977\n",
      "Eval Loss: 73.493355 \n",
      "\n",
      "Epoch: 49 , TRAIN loss: 71.34128 , Temperature: 0.6924633268086435\n",
      "Eval Loss: 74.05084 \n",
      "\n",
      "Epoch: 50 , TRAIN loss: 74.27101 , Temperature: 0.6821538909764523\n",
      "Eval Loss: 73.92295 \n",
      "\n",
      "Epoch: 51 , TRAIN loss: 73.74945 , Temperature: 0.6717963735016784\n",
      "Eval Loss: 73.60079 \n",
      "\n",
      "Epoch: 52 , TRAIN loss: 75.02587 , Temperature: 0.6613976708406429\n",
      "Eval Loss: 73.60572 \n",
      "\n",
      "Epoch: 53 , TRAIN loss: 69.05267 , Temperature: 0.6509646105015452\n",
      "Eval Loss: 73.59644 \n",
      "\n",
      "Epoch: 54 , TRAIN loss: 73.25265 , Temperature: 0.6405039439839885\n",
      "Eval Loss: 73.80253 \n",
      "\n",
      "Epoch: 55 , TRAIN loss: 70.80734 , Temperature: 0.6300223399419125\n",
      "Eval Loss: 73.72022 \n",
      "\n",
      "Epoch: 56 , TRAIN loss: 72.8452 , Temperature: 0.6195263775786136\n",
      "Eval Loss: 73.63894 \n",
      "\n",
      "Epoch: 57 , TRAIN loss: 73.91891 , Temperature: 0.609022540281914\n",
      "Eval Loss: 73.64124 \n",
      "\n",
      "Epoch: 58 , TRAIN loss: 72.875656 , Temperature: 0.5985172095069092\n",
      "Eval Loss: 73.90841 \n",
      "\n",
      "Epoch: 59 , TRAIN loss: 69.65963 , Temperature: 0.5880166589130854\n",
      "Eval Loss: 74.12495 \n",
      "\n",
      "Epoch: 60 , TRAIN loss: 75.52129 , Temperature: 0.5775270487619548\n",
      "Eval Loss: 73.38233 \n",
      "\n",
      "Epoch: 61 , TRAIN loss: 72.79599 , Temperature: 0.5670544205807092\n",
      "Eval Loss: 73.65838 \n",
      "\n",
      "Epoch: 62 , TRAIN loss: 72.20951 , Temperature: 0.556604692096744\n",
      "Eval Loss: 74.40301 \n",
      "\n",
      "Epoch: 63 , TRAIN loss: 72.68349 , Temperature: 0.5461836524472542\n",
      "Eval Loss: 73.63641 \n",
      "\n",
      "Epoch: 64 , TRAIN loss: 73.227234 , Temperature: 0.5357969576674562\n",
      "Eval Loss: 73.63278 \n",
      "\n",
      "Epoch: 65 , TRAIN loss: 74.16976 , Temperature: 0.5254501264603462\n",
      "Eval Loss: 73.26527 \n",
      "\n",
      "Epoch: 66 , TRAIN loss: 72.08741 , Temperature: 0.5151485362502642\n",
      "Eval Loss: 73.09439 \n",
      "\n",
      "Epoch: 67 , TRAIN loss: 78.456024 , Temperature: 0.5048974195219025\n",
      "Eval Loss: 73.14621 \n",
      "\n",
      "Epoch: 68 , TRAIN loss: 73.73921 , Temperature: 0.5\n",
      "Eval Loss: 74.06846 \n",
      "\n",
      "Epoch: 69 , TRAIN loss: 73.1819 , Temperature: 0.5\n",
      "Eval Loss: 73.321 \n",
      "\n",
      "Epoch: 70 , TRAIN loss: 73.43863 , Temperature: 0.5\n",
      "Eval Loss: 73.120926 \n",
      "\n",
      "Epoch: 71 , TRAIN loss: 75.06397 , Temperature: 0.5\n",
      "Eval Loss: 73.60335 \n",
      "\n",
      "Epoch: 72 , TRAIN loss: 72.03889 , Temperature: 0.5\n",
      "Eval Loss: 73.40978 \n",
      "\n",
      "Epoch: 73 , TRAIN loss: 75.08796 , Temperature: 0.5\n",
      "Eval Loss: 73.0464 \n",
      "\n",
      "Epoch: 74 , TRAIN loss: 72.796104 , Temperature: 0.5\n",
      "Eval Loss: 73.26022 \n",
      "\n",
      "Epoch: 75 , TRAIN loss: 76.295715 , Temperature: 0.5\n",
      "Eval Loss: 73.18456 \n",
      "\n",
      "Epoch: 76 , TRAIN loss: 71.578224 , Temperature: 0.5\n",
      "Eval Loss: 73.15289 \n",
      "\n",
      "Epoch: 77 , TRAIN loss: 70.28218 , Temperature: 0.5\n",
      "Eval Loss: 72.846886 \n",
      "\n",
      "Epoch: 78 , TRAIN loss: 74.7035 , Temperature: 0.5\n",
      "Eval Loss: 72.7771 \n",
      "\n",
      "Epoch: 79 , TRAIN loss: 75.2387 , Temperature: 0.5\n",
      "Eval Loss: 73.4482 \n",
      "\n",
      "Epoch: 80 , TRAIN loss: 72.119896 , Temperature: 0.5\n",
      "Eval Loss: 73.46986 \n",
      "\n",
      "Epoch: 81 , TRAIN loss: 72.23955 , Temperature: 0.5\n",
      "Eval Loss: 73.30801 \n",
      "\n",
      "Epoch: 82 , TRAIN loss: 77.28792 , Temperature: 0.5\n",
      "Eval Loss: 73.22431 \n",
      "\n",
      "Epoch: 83 , TRAIN loss: 73.461136 , Temperature: 0.5\n",
      "Eval Loss: 73.27042 \n",
      "\n",
      "Epoch: 84 , TRAIN loss: 72.06599 , Temperature: 0.5\n",
      "Eval Loss: 73.13477 \n",
      "\n",
      "Epoch: 85 , TRAIN loss: 70.887 , Temperature: 0.5\n",
      "Eval Loss: 73.26312 \n",
      "\n",
      "Epoch: 86 , TRAIN loss: 74.2153 , Temperature: 0.5\n",
      "Eval Loss: 73.258644 \n",
      "\n",
      "Epoch: 87 , TRAIN loss: 72.1132 , Temperature: 0.5\n",
      "Eval Loss: 72.68868 \n",
      "\n",
      "Epoch: 88 , TRAIN loss: 73.21365 , Temperature: 0.5\n",
      "Eval Loss: 73.26696 \n",
      "\n",
      "Epoch: 89 , TRAIN loss: 73.57863 , Temperature: 0.5\n",
      "Eval Loss: 73.22333 \n",
      "\n",
      "Epoch: 90 , TRAIN loss: 70.49953 , Temperature: 0.5\n",
      "Eval Loss: 73.76415 \n",
      "\n",
      "Epoch: 91 , TRAIN loss: 71.64477 , Temperature: 0.5\n",
      "Eval Loss: 73.14918 \n",
      "\n",
      "Epoch: 92 , TRAIN loss: 72.94967 , Temperature: 0.5\n",
      "Eval Loss: 73.173584 \n",
      "\n",
      "Epoch: 93 , TRAIN loss: 75.92674 , Temperature: 0.5\n",
      "Eval Loss: 73.466774 \n",
      "\n",
      "Epoch: 94 , TRAIN loss: 77.468285 , Temperature: 0.5\n",
      "Eval Loss: 73.103004 \n",
      "\n",
      "Epoch: 95 , TRAIN loss: 77.08339 , Temperature: 0.5\n",
      "Eval Loss: 73.26614 \n",
      "\n",
      "Epoch: 96 , TRAIN loss: 76.25474 , Temperature: 0.5\n",
      "Eval Loss: 73.25732 \n",
      "\n",
      "Epoch: 97 , TRAIN loss: 71.37439 , Temperature: 0.5\n",
      "Eval Loss: 73.261406 \n",
      "\n",
      "Epoch: 98 , TRAIN loss: 71.957466 , Temperature: 0.5\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Eval Loss: 73.22725 \n",
      "\n",
      "Epoch: 99 , TRAIN loss: 73.86409 , Temperature: 0.5\n",
      "Eval Loss: 73.20703 \n",
      "\n",
      "Epoch: 100 , TRAIN loss: 72.182945 , Temperature: 0.5\n",
      "Eval Loss: 73.219055 \n",
      "\n",
      "CPU times: user 1h 23min 57s, sys: 5min 28s, total: 1h 29min 26s\n",
      "Wall time: 23min 22s\n"
     ]
    }
   ],
   "source": [
    "%%time\n",
    "\n",
    "tf.random.set_seed(1234)\n",
    "\n",
    "model = DiscreteVAE(PARAMS)\n",
    "# model.build()\n",
    "# plot_model(model, to_file='model_plot.pdf', show_shapes=True, show_layer_names=True)\n",
    "learning_rate = tf.Variable(PARAMS[\"learning_rate\"], trainable=False, name=\"LR\")\n",
    "\n",
    "optimizer = tf.keras.optimizers.Adam()\n",
    "\n",
    "# data\n",
    "(x_train, y_train), (x_test, y_test) = tf.keras.datasets.mnist.load_data()\n",
    "x_train = x_train.astype('float32') / 255.\n",
    "x_test = x_test.astype('float32') / 255.\n",
    "x_train = x_train.reshape((len(x_train), np.prod(x_train.shape[1:])))\n",
    "x_test = x_test.reshape((len(x_test), np.prod(x_test.shape[1:])))\n",
    "\n",
    "TRAIN_BUF = 60000\n",
    "BATCH_SIZE = 100\n",
    "TEST_BUF = 10000\n",
    "\n",
    "train_dataset = tf.data.Dataset.from_tensor_slices(x_train).shuffle(TRAIN_BUF).batch(BATCH_SIZE)\n",
    "test_dataset = tf.data.Dataset.from_tensor_slices(x_test).shuffle(TEST_BUF).batch(BATCH_SIZE)\n",
    "\n",
    "# temperature\n",
    "tau = PARAMS[\"init_temperature\"]\n",
    "anneal_rate = PARAMS[\"anneal_rate\"]\n",
    "min_temperature = PARAMS[\"min_temperature\"]\n",
    "\n",
    "results = []\n",
    "\n",
    "# Train\n",
    "for epoch in range(1, PARAMS[\"nb_epoch\"] + 1):\n",
    "    \n",
    "    # this is only needed for the standard Gumbel softmax trick\n",
    "    tau = np.maximum(tau * np.exp(-anneal_rate*epoch), min_temperature)\n",
    "\n",
    "    for train_x in train_dataset:\n",
    "        gradients, loss = compute_gradients(model, train_x, tau, hard=PARAMS[\"hard\"])\n",
    "        apply_gradients(optimizer, gradients, model.trainable_variables)\n",
    "\n",
    "    print(\"Epoch:\", epoch, \", TRAIN loss:\", loss.numpy(), \", Temperature:\", tau)\n",
    "\n",
    "    if epoch % 1 == 0:\n",
    "        losses = []\n",
    "        for test_x in test_dataset:\n",
    "            losses.append(gumbel_loss(model, test_x, tau, hard=True))\n",
    "        eval_loss = np.mean(losses)\n",
    "        results.append(eval_loss)\n",
    "        print(\"Eval Loss:\", eval_loss, \"\\n\")\n",
    "\n",
    "    if PARAMS['hard'] == True:\n",
    "        model.save_weights(\"model.h5\")\n",
    "    else:\n",
    "        model.save_weights(\"model_hard.h5\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "sample_y:  (100, 20, 20) , logits_x.shape: (100, 784)\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAjkAAAIuCAYAAABdOBlOAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8qNh9FAAAACXBIWXMAAAsTAAALEwEAmpwYAABeQ0lEQVR4nO3dd7wVxfnH8bkq0jtIsYOKGGzYFYIajVEsaOxB1NgbxthiTaKJMYk1dqxRiVEjJrEbCwoqNlQ0iCIoIIJKUUCqen9//F55/M54z2HZu3vKnM/7r+e4c/YsZ+/eO87zzExdfX29AwAAiM1K5b4AAACAPNDJAQAAUaKTAwAAokQnBwAARIlODgAAiBKdHAAAEKVVih2sq6tjfnkZ1NfX1+V16uU1qKv7/kcnWWagofc1eAENnCvJe9NeQ5bvS/pvbOBcmd9Pns3yyOvZ5H6WB89mPArdS0ZyAABAlOjkAACAKNHJAQAAUaKTAwAAolS08BhxSVJgm3Qvs6yKhVekXZpryPJ9AIDqwkgOAACIEp0cAAAQJTo5AAAgStTkwFOOepW09T1pa3nyXNyQeh8AqByM5AAAgCjRyQEAAFGikwMAAKJUdTU5/fr1s/ill17yjvXq1cviPffc0+KBAwd67R555JEGz/3iiy96r0ePHp36OoFa06pVK4sPOugg79iOO+5o8aqrrmrx5ptv7rVr2rSpxUuWLLE4fNb/9Kc/WTx+/HiL09Zp4f+ttNJ3/9/bunVri9dff32v3fbbb29x7969LR4wYIDXbtGiRRa///77Ft96661euzFjxli8YMGCFb1sNGDllVe2uEuXLhbvuuuuXju9f23atLG4b9++XrtVVvmuuzBp0iSLw3v5/PPPW7x48eIVvezMVV0nB6WX5R+OLAtzs1yQMKtr4I8sAFQO0lUAACBKdHIAAECU6ooNr9fV1ZVl7F3zgs45N3z4cIt33nlnizXf65yf69f6gKTC8y1cuNDiE044weJ//OMfK3zuFVFfX5/LYisN3c8k67xUQrqqoWvIM12V5XU65zK/n+V6NjXP75xzP/rRjyy+5ZZbLF5ttdW8dt98802D5yv2PWt9SGjatGkWH3XUURZrPUAeSvlslkKHDh2814cffrjFRx55pMVt27b12jVv3rzBWH8HO+fc119/bbH+fv3qq6+8dk899ZTFv/nNbyzW+5yHPO5nue5ls2bNvNeDBw+2+Mwzz7Q4/Nuoz6DWxIXn03b6PH/++edeu8svv9ziO++80+L58+cX/wc0UqF7yUgOAACIEoXH8OQ9YpL2/KU+d5a7sQMAyqMi01U33HCD9/q4445L9L53333XYh1CmzdvXsH36B+pcKq50qG2/v37e8fGjRuX6PqSKme6qsD7ltumMSmttOdPMrOp1J2cGNNV+m/UqeDOOfevf/3LYh0GD6eOvv322xa/9957Fs+aNctr165dO4t1aqvGzvkpkv/+978W77///l67yZMnuyzFkK7ScoDLLrvMO3bIIYdYrFOGw7TR1KlTLdap/nPmzPHa6fOw5pprWqzLfTjn33dNOer1OOfc3LlzXZaqPV2lKV1NNTrn3BVXXGFxy5YtLZ45c6bX7q233rJYnxe9/84516NHD4s32WQTi/XeOeensu644w6LzzvvPK9d1ukr0lUAAKCm0MkBAABRqph01Q9+8AOLR44c6R3r2LGjxR9//LHFQ4YM8dp98MEHFn/xxRcWF1tBU4f7LrzwQu/Y+eefb7HOKhkxYoTX7uijj7Y4i+HUSktXJTz39/5b3vU9pU5zNeLfU9VD4joDR9NOzjnXtWtXi3X4+eyzz/baPfjggxbrM7ds2TKvnc5obNKkicVhKvmmm26yWGeBhL879t13X4vDGT1pxJCu2mmnnSzWdKNz/nf58ssvW3zBBRd47SZMmGCx3s9whqqmLfXcu+22m9du2LBhFusMn4suushrd+mll1pcaMbeiqj2Z7N9+/YWa9rWOec6d+5s8SeffGJxeC+ffvppi/W71+fPOT8tudZaa1l8xhlneO1+/OMfW6w/G3/961+9djpjWWfhpUW6CgAA1BQ6OQAAIEp0cgAAQJQqpiZn2223tTjcDVyvcejQoRZfd911uV7TJZdcYrHmHcOpdXvttZfFhXY4XxHU5FCTszylfDZ1Gu9tt93mHdNaNZ3Cet9993ntvv32W4s11x/WVehr/W7DVVqvvfZai3XauH6Oc84deuihFj/00EOusaqxJif8GdW6ll/84hfeMa272GeffSwOV5LWZ0HrLsLvX1/rdYQrI2vtzcknn2xxuJruZpttZrHWXaZV7c+m7ige1lfp3yn9fv/85z977fSZa9GihcXh7zutl9N7Ge5QoPVVWhMX/mxst912Fr/22muusQrdSxYDhCfvbR3yvIa0nZUs/83sQg4AlYN0FQAAiFLFpKsGDBhg8bPPPusd01UTf/7zn5fqkjyTJk2yeN111/WO3X777RbrZoFpVeMGnVmmdxqSZ9qpMSM5Ma54HG6M+eSTT1r8wx/+0Dum04x32WUXizXt4Zz/PWkcDmEXEn7POoX15ptvtrhfv35eO51WG157ON05iWpMV4X3U9MaOt3XOefGjBljsU7zDlewzlqXLl0sfu655yzu1q2b1+6AAw6wWH8u06r2dNXFF19s8VlnneUd09XE+/TpY3GY5tNnq1jqUX+3Ffv92KlTJ4unTJlisabCnPOnrmvabXnnL4Qp5AAAoKbQyQEAAFGi8LiG5JnyaUzBbZL0UaklvYY8i5gBAI1TMZ0czS2GNO9fLk888YTFxx9/vHdMp78D1U7rXZxzbsstt7Q4zNOfdtppFi9durTgOZPm85O83znnpk+fbvGVV15p8TbbbOO1W3vttS3WJfCdS1eTU410x3bn/B2kw075VVddZXGx+5m12bNnWzxq1CiLtQbHOed69uxZsmuqRLrtgnP+die6nINzzt16660W65Yrxf4HNWmNXDG6E/3rr79ucf/+/b1266yzjsXhtWexzcP/kK4CAABRopMDAACiVLZ0VY8ePbzX3bt3t/jLL7/0joU7H5fDM888Y3GYrgKqnaYtwl2KNd0RPpu6tEIpa490ldaJEycWvAZdjTVMw+nOzDHbeOONvde6q3w41b9c91PTJJMnT7Y4TGNsv/32Ft94443esVqofQvvpf4dDVM8jz76qMVZ7NielN5L3Xl8hx128Nrpsxmm4bJMV1VMTQ6qS9otFbL8RVSJBcu18IsWAKoF6SoAABClso3kDB482Hutw24PPPCAdyzcsBNAtlq2bGlxOKysRo8e7b1esGBBbtdUjI6Y6WyOkKY7wo11Y6ajmuGqxvqdTJ061Tv26aef5nthBej91FWXw9WadWXkhKuNVz39Dvbee2/vmG52G97L9957z+JyfTe6qnGYGlXhfc4SIzkAACBKtfO/Nsh1H6e862Gy3PMqSZss9+KK9f8wAaDSMZIDAACiVLaRnIMPPth7rVNTr7766lJfDlDT2rZta3Hr1q0LtnvooYe811lO9UxLa0xWXXVV75iOoi1btqxk11RuWn+06aabesd09HH8+PHesXnz5llcyhFIvSa9h+EU8mbNmlkc1nFksVpvJdK6G51C75z/HbzwwgvesXLVyym9vmLT2PW+Ouev0Nzoa8jsTAAAABWETg4AAIhSxRQeT5gwweJwmiryk+cu5GmLmJMOk2dV7JxlkXG16tu3r8Vhymfx4sUW65RQ5yqjqHr11Ve3OJwmrkPk5ZoeXQ66mmzXrl29Y3o/77jjDu9YsWm+paIbqYbpqm7dupX6csquc+fOFoc7BWi6OLyX5Uol6+9J3ckgfDaLpSWzxEgOAACIEp0cAAAQpZKmq3RVVa0YB1B6Oqysm1eGz6bOfPz888/zv7AEdNbG4YcfbnE47P3hhx9aXEvpKk0NhGnW2bNnW/zKK694x8o1Q0mvUWcQhSmOcs3+KidNPerfUOf89OK4ceO8Y+X6fvQZ3HHHHS0O7+WsWbMszjO1xkgOAACIUsUUHqM88lzxOO3/SSQt8M3zutJeQ8zFyQBQbRjJAQAAUSrpSM6BBx5occ+ePb1jmp+rROHur6oSVn0FVpTmzjfbbLMG/7tzzn388ccWV8qqsp06dbJ4yJAhFofXd+ONN1q8aNGi/C+sQug07HB0cfr06RZXyu8unU5c7HetrupbbAXdmPTp08fiFi1aeMe0Rq5c30f486Urpu+zzz4F36f1clmucBxiJAcAAESJmpwakmSBu7wXA0xyXWnPlfTz0u4czo7jAFBd6OQUscUWW1i85557Fmx37rnnluJygExpWmq99dazOEz5zJ071+JwGmipVsgNh+lvuOEGi9u1a2dxOMX9zjvvzPW6Kol2wjWdt3DhQq/d0qVLLW7VqpV3bM6cORbnmZoMU6IDBgywWFfJDX++hg0blts1VaqwtEPp9xjeS51un6fwd0Lv3r0t3nDDDS0O02lPPvmkxXlunku6CgAARIlODgAAiBLpKqHpKeec++Uvf2mxDolrhb9zzj3xxBO5XheQB01HaJoipCmDUtYc6Yybiy66yDumM3D033H99dd77Yr9u2Kj9+aTTz5p8L8756+aqytHh6+zTlfpuTfddFPv2F133dVgu5deeslrpxs514rJkydbHKb5NEXVvHlz71ip7uW6667rHbvpppssbtasmcXhDOq//e1vuV2fopMDT96LAea5WF6Wu5dn+e+hGBkAyoN0FQAAiBKdHAAAEKWSpqs++ugji/Nc4XBFaI7zjDPO8I4ddNBBFusqoWG7Slk1FFgR+rNfbLVUXT1X62Sc81cR1rRcWOuhirXTGoM//OEPFh9zzDEFr13rNK644oqCn1VL9L5MmTLFO9atWzeLw+m/WdP7tPPOO1t83333ee10p239XTt48GCvXS3+rtUVxxcvXuwd0+dn/fXX947pfS/2HCR9RvSzNtpoI4tHjBjhtVtnnXUavN7zzz/fa6crHueJkRwAABAlCo9rSJ47cjemyDjJqst57jjemIJldiEHgMpV0k7Os88+a7EOSTrnD1fqap3ONX7zzk022cR7feKJJ1rct29fi7fccsuC59Bh05dffrlR1wNUAh36f/vtty3+8Y9/7LXTVUu33XZb79hzzz1nsU4DDdNQumKxPt/hZownnXSSxWussYbF4dRZXdlYN/5dsGCBg58m0BWrnfM3fNx44429Y9OmTbNY72c4xVc783qvdakN55w755xzLD722GMt1k0cnXNu9uzZFh9++OENXk+tmjRpksVhuqpz584WH3bYYd4xXepE05fhvdT7p/e1bdu2XruhQ4dafPzxxzd4DeE13nPPPRbfcccdXrtSpZJJVwEAgCjRyQEAAFGqmJoc3dTr8ccf947NmDGjUecOh9g7duzYYLswLfbvf//b4ldffbVR11AJ0g4P5j2smPb8aXdQT9Mmj/eWm86oevTRRy0+6qijvHZdu3a1OFxR+OGHH7ZYh7fDoe4111zT4h49elgcpi3CtNT/hLMxDz30UIv/+9//NvieWqabcr755pvesX322cdiXdXdOT+tqL8Pw41PNUWhKcdBgwYVbKfPSrgStc5kHTlypMN39LsaN26cd2yXXXaxOEz96uyq9957z+Lw76k+3/3797d44MCBXjtNM2uKK9xE9e6777ZYU1zFZnDmiZEcAAAQJTo5AAAgSnRyAABAlOqK1UPU1dXlVoyx7777eq91NcTNN988r491zhXefTlcLfXSSy/N9ToKqa+vz6vQo6TLvyatVyl3TU7e8rifWT+buvLtCSec4B3TZzOstSk0lTiktTbF7ok+m7oiqtZsOOfcG2+80eB78pbXs5n1/dTvWKfiO+fcVVddZXG/fv28Y3oPmzZtanH4HWvtTpMmTQpehy5ToDuK/+xnP/Pa6aq+pVyluhqeTbXeeut5r7V+da211vKO6T3T73TZsmVeO90pvNi91PfpzuinnHKK105rqsLPylOhe1kxhcfIX5Y7ZifpYDTm/Mv7vKRK/XkAgMpBugoAAESpbOmqUPfu3S0Op5DrCp1p3Hzzzd5rHeq+8cYbG3XuPJRzSDzLkZy02ywk+byGZDnMneVITrUNiYdD1vvtt5/Fv/rVr7xjOh1cN+8M74WugqpD2OH05r/85S8WP/300w2+v5yqJV2lwjSi/j4N76dOIdbp3+FGnjodWFNS4VIbev7XXnutwfeUU7U9m+HvJd0o85JLLvGO7bDDDhbrUg3hMg16L5cuXWqx3i/nnLvsssssfuaZZyzW1ZTLqdC9ZCQHAABEiU4OAACIEoXHNSTPWQtJU1NZ7uSd1c7kjUlNsQs5AFSuiqnJwXeqYQp52hqZWuzkVFveH4VVY00OCuPZjAc1OQAAoKbQyQEAAFGikwMAAKJE4XENyXjtl8zOVeo1cNKeu6HrTLvyMwAgf4zkAACAKNHJAQAAUaKTAwAAokRNTo1Lu9ZMkjYNnTvJ2jlZ7p+VdK0eAEB8GMkBAABRopMDAACiRCcHAABEiU4OAACIUtENOgEAAKoVIzkAACBKdHIAAECU6OQAAIAo0ckBAABRopMDAACiRCcHAABEiU4OAACIEp0cAAAQJTo5AAAgSnRyAABAlOjkAACAKNHJAQAAUaKTAwAAokQnBwAARIlODgAAiBKdHAAAECU6OQAAIEp0cgAAQJRWKXawrq6uvlQXgu/U19fX5XXq8D/U1fkfVV+f7paH5yl0robaJZHkXEmvPck1ZHyuzO8nz2Z55PVscj/Lg2czHoXuJSM5AAAgSnRyAABAlOjkAACAKBWtyUH80tbglPoaktS+JK0LSittPREAoDwYyQEAAFGikwMAAKJEJwcAAESJmpwakqRmJW1dS9I1cdKunZO2biftudN+D9TtAEDlYCQHAABEiU4OAACIEp0cAAAQpaquydliiy2814MGDbL4pz/9qcW9evXy2mndhNZZjB071mv37rvvWnzJJZdYPGHChHQXDERGn6W2bdta3LNnT6/d3nvvbfGBBx5YsN3KK69s8fz58y0eM2aM1+7EE0+0+MMPP7S4EtZ9qiarrOL/CWjTpo3F6667rsWrrbaa16579+4Wb7DBBhZvueWWXrtu3bpZPHv2bIvD37X6+/Xzzz+3+Ntvvy3+D4Bp0qSJ93rTTTe1WJ+XPn36eO3atWtnsX7fHTt29No1a9bM4oULF1r8wQcfeO3OO+88i1944QWLly1bVvT681LVnRxkr9SbajbULunnJdmgs9SFwPyRBYDKQboKAABEiU4OAACIUsWkq4499liLN9xwQ+9Y//79G3xP3759vdeaKihUd+Occ8OGDbP4wQcftPjJJ59cgSsGak+rVq281xdeeKHFWgfXunVrr92qq67a4Dm0Bsc557755huLv/76a4u1BsA559q3b2/xtGnTLC5X3r+SrbSS//+yWmtx8skne8cOOuggi1u0aGHx0qVLC56zU6dOFmvdhnP+/dCfiZdfftlr17x5c4v1ZyX8XGp0fE2bNrX4rLPO8o6deeaZFuv3u2DBAq/dlClTLNbnMazXUnosvCdrrbWWxZMmTbL4k08+8drps54nRnIAAECUKmYkB/nLc8XeLFc8zrJgOe9VipNcOwCgPOqK/VKuq6sr2W9sHfIKr0mHqnX69qhRo7x2ekynIWpKqhrU19fnNSVoufeTTk5haa/dOZf5/Szls6nTgP/5z396x8KU8f8888wz3usrrrjC4rfeesviefPmee0KDWGH02N1mF7PoSku57LvdOb1bGZ9P/XnsnPnzt6x6667zmKd2u+cn4b64osvLH7ooYe8dlOnTrVY0xDvvPOO1+6zzz6z+Kuvvmowds65JUuWWKw/A+HzpcfCe5vmXudxP/N8NsP07q677mrx3Xff7R3r0KGDxXovhwwZ4rXTv6N6H8I0p6YR9XkM2+k90r/dYSo569RyoXtJugoAAESJTg4AAIhSxdTkjBgxwmJdudg5Pw211VZbleqSsAKy3HG8IXmmuSphEcFKEs6q0OcxXC1Vh6avuuoqi88//3yvXZhGWlHhLBsd6q7le1WIfic6s8Y5fyXj8L5MnjzZ4nPPPdfi5557zmu3aNEii4ulifRYsZlRhWbGFru31L85t8Yaa1isKVzn/Hs0dOhQix955BGvXdLvUc9XbPayHtNUVrnuFyM5AAAgSnRyAABAlOjkAACAKFVMTc4JJ5xgcbi7+Nprr22xrqao0xgBZCPMnWveX6eYOudPKb/gggssbmwNTqhY/ZXW51Cn8f/0e9BaCuecmzt3rsVag+OccxdddJHFTz31lMXhfS9U8xO2KzblO8m1cz99xeqawmP691HrcJazbIzF4dTwQvel2PnS3P+sVUwnB/krddFv2nMlLWLO6qFpTOFqVmv1AACyR7oKAABEqWJGcnSFYt1A0znnfve731msm8GRrgKyFw5762qpL730knfsjjvusLhUG+45V3yFdPjfid4/55z7z3/+Y7H+Pg3bhlPPVZcuXSzWdFiYGuPeZCv8PnWF6XHjxnnH9F7qSsnFRs6Lpas0BZ102YBKwEgOAACIEp0cAAAQpYpJVyF/aVf2Tdsm7eeVerizMZ/HarsAULkqspMT5gL1D0nv3r0b/O/FvPvuu95r3RkVgC98rnQ34/bt23vHWrZsaXHr1q0tDqcSFxLW/2hdj/4eCOt9sp6iHrPwu/rwww8t3mijjbxjulyH1t107NjRa6f349VXX7V4/vz5Xruk2z8gnZkzZ1oc7uqt3/3mm29u8Xvvvee1W7x4scX6zIU7xYev/6fS7yvpKgAAECU6OQAAIEp1y1n9sGTjUJ07d7b4lVde8Y7pKseFdqstduzBBx/02g0fPrzgsUpQX1+fV6HH9+5nnjUllVCTk/fifAn/PZl/yXk+m82aNfNe6zTxH/3oR94xTX3okg7Tpk3z2rVt29birl27Wjxjxgyv3ccff2zxRx99ZPHjjz/utfv0008tLrYKbNbyejbzvJ+rrrqq9/rss8+2+JhjjvGO6U7WmpIKU/yaCtF78frrr3vtPvjgA4t1+YGwhKCUyw+oans2Qz169LD44Ycf9o7pSuULFiyweOLEiV47TS1ravqFF17w2j355JMW63T16dOne+1K+TyqQveyImtyUDpJ/uDn3TFJu2pw2lWXK7H4GQCQPdJVAAAgSmVLV2l6yjnnnnvuOYt79erlHRs7dqzFOsw5evTogufXYdhwVU+dQaD//q233tprp59VyhlZpUxXJZHlyEeSUZosR3Ky3K+rITGmq8KZNDfddJPF/fr1844tXbrUYh22DlfZ7d69u8WargpnYelMoFatWlk8Z84cr51u6Pv8889bnHfaoxrTVauttpr3+p577rF4xx139I7p7Br9nRf+/tOUhK6MrCvrNvT6fzT14ZxzRx99tMWzZs1q8HPyUG3PZvj7ZoMNNrD4mWee8Y7pM6fPaXgv9f4VSlc65z+bekw3/3TOuZNOOslivZd5j44XupeM5AAAgChRk1ND8tyFvDHnSiLLAuKs6pAaOheLAwJA5WAkBwAARKlsNTlhbl9rckaMGOEdO+CAAxr1WWFNzuDBgy0eNGiQxf379/fajR8/vsFrmDBhQqOuZ3kqLe+f5cymJNKOoiQ9V6lHcqoh76+1GOGqxjvvvLPFugquc87NnTvXYq2R01VUQzqluU2bNt4xXS7ilFNOsXjbbbf12mmNwRlnnGHxvffe67ULV4FtrEp7Ngtp0qSJxXr/nHPusssus3idddbxjumKxXfddZfF4dTwyZMnW6zPwCabbOK123PPPS3eYYcdLA7rPXQpgeuuu87ixx57zGun09Wz+J1TDc+mCpcD0BWrf/vb33rH9FnSZ3PKlCleO13JWOt4NHbOuZ122qnBc4f0Hp1//vkW//3vf/fahTvWNxY1OQAAoKbQyQEAAFGqmBWPK8Gxxx7rvdZp6DrtfPfdd/fahUO5jVXKIfG0hbN5poqSSju1Pcv3xZKu0uteZZXC8xGKfXeaGsqiwLxFixYW/+IXv/DaHX/88Rbraq46PO6cc//6178szmJTz2pJV+m04J/85CfesT322MNiTTs559xDDz1ksa6MG6b9kk7t1vvZrl07iw888ECv3XHHHWexLi+iq1475993LSdwLt3PXDU8m5pK1inezvnpK01ROuffI502XuxeFvsO9fy6Ufbll1/utdtyyy0t1iUEwlXLhwwZYnGx9HZSpKsAAEBNoZMDAACiRLqqCJ2VpbO/whVhdfXVLDb8JF2VDOmq2hEO0+tMD01hhLN2/vCHP1j85ptvesfSpK+qJV2lqYXWrVt7xzQdGa4knUVKL4kwtbLxxhtbrCnH7bbbzmv3zjvvWHz44Yd7xz755JMVvo5qeDY15RP+ftGf93LttxfO+Np3330tvvrqqy0Ofw4vvPBCi6+88krvWJqVrklXAQCAmsKKxzWkmnfyTjuKkvc6OaxwDACVi5EcAAAQJWpyEvrhD39ocThlTqeXX3LJJd6xq666aoU/q5S7kMc+ktOQLEdyEn5exef9K114P7SuZNNNN7U4rNPQmoWHH37YO6a7NietAaiWmhz9vsLvLu+dvdPQuhNdhfmGG27w2um9fuqpp7xjRx11lMVJpyRXw7Op969cdTcrQp9NXbX8nHPO8drpaul675zzV2hOipocAABQU6jJwXLlOYupoffmXeeSZPSqGv6PCQBQHOmqFMINP3V6ea9evbxjxVaPLaTS0lWV2snJqnNUgg5NxQ+JV5tCKzRvs802XruzzjrL4iVLlnjHdJXdcCp1IdWSrqpmmroaOHCgd+yWW26xOJzurhssT5o0KdFnVUO6qtros6nTxsPVyPfbbz+LX3vtNe/YEUccYXFjU4+kqwAAQJTo5AAAgChRk5PCrFmzvNdaCb7hhhuW+nKAsinXzA/9LE1bhEPbOly+7rrrescqcZYR/Psyf/5879iiRYssbt++vXcsTWkAsqfP5ldffWWxlnU459zWW29tcc+ePb1juopyYzfv5KeixmVV15Ll9O2k70s7/X1550n6eQ2hYBkAKgfpKgAAECU6OQAAIEqkq1II624GDRpk8fjx40t8NUC2VlrJ/38frXUI61j0dblSdZpG3H333b1jHTt2tLh58+besXBKea2o9BV09edtr7328o7p7uXh/fv444/zvbAyCZ9HVel1ZXrt3bt3944Vu/Ys/12M5AAAgCgxklNDsiyczWoRwaTnSnOepNeV9Nob828EAJRexXRyTjvtNIs///xz79jdd99d6sv5Ht2E8/e//713rEWLFhYfcMABJbsmICuaBlh//fW9Y3379rX4zTff9I5NmDDB4lIOneuquNttt53Fe+yxh9dOl3v45z//6R2LOV2lnW/9/eScc23btrU4XOlZv5M8U1nh/xxoKnHw4MEWh/dTN3W8+OKLvWMLFy7M8hLLStM8bdq0sXittdby2s2bN8/imTNnesd0aQXdqDZr4b3U6d/bbrutxbrCsXP+z9cDDzzgHcvy2SRdBQAAokQnBwAARKls6ap9993Xe33ZZZdZPGzYMO9Y1umqzp07F7wOpcd0yP6zzz7z2g0ZMsRiHb6vBqVeDDBprUsSWQ2nN6b+phJnp6Shw+MDBgzwjv32t7+1+IsvvvCOnXzyyRa/9NJLFi9dutRrV+h7Cr9TvY6WLVta3KFDB6/dIYccYrHOqJoxY4bXbvjw4RaPGDHCO1bpM1MaQ79XTbU759yRRx5p8TvvvOMde/zxxy3We532u9I0qK5QHM5QPeaYYyzecsstLZ49e7bX7tprr7X473//e6prqgb6vGj6Z//99/fa6d+ocHbZY489ZvHYsWMtDtN6OptNnzlNhTnnXNOmTS3W+xqmQw888ECLt9hiC4u//PJLr90rr7xi8ZVXXukdW7ZsmcsKIzkAACBKdHIAAECU6OQAAIAo1RWrKairq8ut4CCshdF8eZj/1bystgvz+Zrn1amjuiJx+D7994fne/fddy1+4oknLL7kkku8duGu5I1VX1+fy+IrDd3PrGpyGqPUdS1p1wtqRE1O5l9q1s+m/tt0d2Dn/LqWddZZxzumu0Trczp16lSvne4krPU64ZTYBQsWWKzTxDV2zt/deNKkSRbfcccdXrsxY8ZYnEUNTimfzUaez+JNN93UO6Y1Vptvvrl3TL9LrZn46KOPvHY6rVlrOnbZZRevXatWrSzWmo7wWdL7OXHiRIuvvvpqr53WllTq/cz6Xur3Fk7D1nsZPps6bVy/K33GnPPvhU7d1s8Nj+m5w3upr6dNm2bxXXfd5bXTWtvwmtIodC8rZp0c5C/PHcArQdprynIxwFgKkQEgBqSrAABAlMqWrgrttttuFofpJaVpLp0K7py/OaYOnWvayTk/vfTggw8W/CydDl7K1TTzGhJ3zi33fpZjO4M8p4Kn/fyMR3IqfkhchcPUW221lcWnnHKKd6xPnz4Wr7baahaH6aVCQ+K6gq1zzn3wwQcW6+rKzz//vNdO0yrTp09v8Nx5qJZ0leratav3+qSTTrJYl79wzp+qr/dQ040hTYWEU3912vBbb71l8ciRI712b7zxhsXjxo2zOO/fu9WQrlK6WrVz/rM5cOBA75hO3+7SpYvF4Yafep81lRxO+dYp5ZpSnDx5stdOn+mHH37Y4nA5AF2ROQuF7iUjOQAAIEp0cgAAQJQoPK4hea42nGX6Ku9VkSuxaBoAkL2KqcnBdyptCnmSDkZjOg55XkPaOpparslBYdVYk5OHQstwVJtqq8lBYdTkAACAmkInBwAARImaHADACqnmFBVqC52cGpJlEW6WWyMkOVeSGpkst2LglzgAVD/SVQAAIEp0cgAAQJTo5AAAgChRk1ND8lx/JmltT5br1iR5X57r6zTmXACA/DGSAwAAokQnBwAARIlODgAAiBKdHAAAEKWiG3QCAABUK0ZyAABAlOjkAACAKNHJAQAAUaKTAwAAokQnBwAARIlODgAAiBKdHAAAECU6OQAAIEp0cgAAQJTo5AAAgCjRyQEAAFGikwMAAKJEJwcAAESJTg4AAIgSnRwAABAlOjkAACBKdHIAAECU6OQAAIAorVLsYF1dXX2pLgTfqa+vr8vr1OF/qKvzP6q+PtktD9/X4Ic1cK6070tyDeW49rBdgTaZ30+ezfLI69nkfpYHz2Y8Ct1LRnIAAECU6OQAAIAo0ckBAABRopMDAACiVLTwGHFJUnCbpLi2MdIWI6e9riTnTirJ52X5XQEAGoeRHAAAECU6OQAAIEp0cgAAQJSoyYEnywX8ktbRpK11yXJhwTTnzvLzAADZYyQHAABEiU4OAACIEp0cAAAQJTo5AAAgShQe17hSF/3mXcSc5vMag0JjAKhcjOQAAIAo0ckBAABRIl0FAAA8K6+8sve6c+fOFnfo0MHi6dOne+0WLFhg8TfffJPT1SXHSA4AAIgSIzk1JG2hbt4FvuH5S70TetJrT1IQTSEyAFSOmujkrL322t7rU045xeKtttrK4pNOOslr98477+R7YUCN23DDDb3XZ511lsXffvutxTfeeKPX7o033rC4EobEa1mTJk0s3mmnnbxjvXr1sviZZ56xeOLEiV67ZcuWWcz/KJTWSit9l9DRNNQFF1zgtdtvv/0afM/kyZO9dg888IDFDz/8sMUzZszw2i1cuNDiPO856SoAABAlOjkAACBKdcWGierq6qpq3HCDDTaw+OSTT7Z4yJAhXrs2bdo0+P6wSnyvvfayeM0117R4ypQpXrtx48at+MUWUV9fn8sKdg3dz6x26U56nkpYDDBtDVAjdibP/H5W27O56qqrWnzmmWdarKlj55xr166dxTokPm/ePK/dddddZ7GmsmbNmuW1+/rrry3OYki8lM9mJQl/9nv27GnxTTfdZPGWW25Z8Bya/j/nnHO8Y5MmTbJYZ/VoGss55+bOnWvx4sWLl3fZy1WLz6Y+V845161bN4vPPvtsiwcNGuS102dYU8SffPKJ107viz6Pw4YN89o9++yzDb4nrUL3kpEcAAAQJTo5AAAgSnRyAABAlKquJkfzib179/aO/ec//7G4a9eujf6s+fPnW9y6dWuLX3rpJa9d//79LdZpr2lRk0NNzvJU4rOp1ltvPe/1o48+avE666xT8H36vWttRng/li5darHW0ukUdOece/LJJy3WKatp1VJNziqrfLfCyMEHH+wd09qNNdZYI9H5dCXct99+2zvWsmVLi3Vl3aefftprd+WVV1r84YcfesfS/C6rlWdT/25usskm3rHzzjvP4n79+lms9WzO+c+Z/m1s1qyZ106XDdA6Hn0WnXPu1FNPtXjmzJnesSzvZU2sk4P/V+oF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D+xwDgCVI6p0VYcOHSweOXKkxeEqrdOmTbN4/fXXt3jp0qX5XdwKKOeQeJZ/3LPs5GSZMssz9VWgTcUPiWdBh9J1mQbnnNtkk00sHjFihMXdunXz2unKxscff7zF4VTzcAmKUoktXZWWTiceNWqUxeHvWl1R+ZBDDrH4iSee8NqVa5XjWnk2NSUVpry22morix999FGL27Zt67XTlKJucj1+/HivXbkmWpCuAgAANYVODgAAiFJUNTm6GVw4bKouuugiiyslRQVUOx0G15VunXPu4IMPtrjYBry6svF9991ncbnSU6Wiac5KWVdJryncSFVnovbu3dvi8No1zfjUU09ZzCacpaXpKp0N55y/YrGmqMJnTjfvrKYVx6Pq5KDxkta15Ln7dtq6oDx3OAcAVB/SVQAAIEp0cgAAQJSqegp5jx49vNe6Q7nuQv7II4947QYNGmRxJeb6q3GaamPSVUlSX3mvXJwEU8iL07x/WBOn9RidO3e2eMGCBV67/v37W6yr51ZK3r8an8209Gd53XXX9Y698sorFmv91dy5c712uqv5559/nvUlNlqtPJutW7e2+Nxzz/WO6c7xWns1ceJEr912221nse5CXimYQg4AAGoKhcc1JMkISSUU72a8b9Ry35flLuSVMuIAAKjCTs7qq69u8dNPP+0d0xSVrmp80kknee0qMUUFVDudQj506FDvmK5Grs+fTkt1zrn//ve/FtNhLK9mzZpZfM0113jH9H7qdPBTTz3Vazdr1qycrg7Lo8/jLrvsYnG4oaamqBYvXmzxoYce6rWbM2dO1pdYEqSrAABAlOjkAACAKFVduqpv374Whys3an3EbbfdZvHUqVPzv7AqkOcu2lnOYspydlWWKuEaKtkWW2xh8f777+8d0+9OZ0H+6U9/8totW7Ysn4tDInqffv7zn1us6Y6w3euvv27xvffe67Xj+SifddZZx+JLL73U4k6dOnntNN04fPhwi3V2o3PVey8ZyQEAAFGikwMAAKJEJwcAAESpKlY83nrrrS1+/PHHLW7Xrp3XTncU79evn8WvvfZafheXg1KuqppVTU5DkuZw81zxOMvPa0jCa4h2VdWWLVtarM+mro7qnHPz5s2zeOedd7Z43LhxXrtK35069hWP11hjDYtfeukli3XpDuec++KLLyzW1a0/+eST/C4uBzGteNy8eXPvte4ufthhh1msU8udc27ChAkWDxgwwOJKXKG6mEL3suoKj5FelovZVcvu3kmuszGLD7IYIABULtJVAAAgShU5kqND4M4599vf/tbiMEWldHO4cOM/AI23yir+r4whQ4ZYvNlmm1kcrip+4403WvzBBx9YXOnpqdjpqsbOOXfWWWdZrBupaimAc84df/zxFn/66ac5XR2WRzfF1RIN55zba6+9LNYR5/nz53vtTjjhBIvDDVZjwEgOAACIEp0cAAAQpYpMV6Gy5DmzKaksdyFPcp60s6tYFRkAKkdFdnKOPfZY7/Vuu+3WYLuZM2d6r/fYYw+LdVocgPS04xZupaL5fJ2aOmbMGK+d7ja+cOHCrC8RK0DrOLp37+4d22GHHSzWeqnnnnvOa6fLBYT1VygdranafvvtvWP6PH711VcWX3755V473ZYjxntJugoAAESJTg4AAIhSRaarwiGzL7/80uIrr7zS4ptvvtlrN2PGjHwvDKhBOm1cV7d1zk93TJkyxWJdYdU5/9lk2nh5abqqSZMm3jFNJT7xxBMWhyUE4TRklIf+rRw7dqx3bNSoURY/8sgjFt99991eu0WLFlkcY/1gRXZyUFnSFu8mPVdWqxInLSDO80GO8ZcEAFQr0lUAACBKVbFBZ63JaxNA59z37mdWey9lOZKTdoPOLKd9J1VrG3Ruuumm3uurrrrK4v32289i3cDRuXhGuGLboHPVVVf1Xvfo0cPiiRMnWhzjrBvn4tqgs9YVupeM5AAAgChRk1ND0o5YZLmAX9rPy3IH9STS1g4BACoHIzkAACBK1ORUoHLm/bMcwUg6spJ2e4Y864nS1gUV+ogUl1X8hDybZRFbTU6toyYnHtTkAACAmkInBwAARKlougoAAKBaMZIDAACiRCcHAABEiU4OAACIEp0cAAAQJTo5AAAgSnRyAABAlOjkAACAKNHJAQAAUaKTAwAAokQnBwAARIlODgAAiBKdHAAAECU6OQAAIEp0cgAAQJTo5AAAgCjRyQEAAFGikwMAAKJEJwcAAERplWIH6+rq6kt1IfhOfX19XR7nbeh+1tfXh20aup6GzrXcNgWuIVG7JNeQ5Nxp35f0GhJ+D5nfT57N8ijls4n85XE/uZflUeheMpIDAACiRCcHAABEiU4OAACIUtGaHMQlSU1JQ9LW0eRdI5O2LiittP8eAEB5MJIDAACiRCcHAABEiU4OAACIEjU5NSRJrUs5akyyqgtKu95N0lol6m8AoLrQyQGQubCTqK9XXnlli1dayR9MbtOmjcXffPONxYsWLfLaha9RmcL7W8i3336b85WgsZo0aWKxPsPhvVu2bJnFlfA/hqSrAABAlOjkAACAKEWVrtIhtL322svi008/3Wv3l7/8xWIdWivmmWeesXjevHlpLxGoamEaSoewmzVrZnGLFi28dv3797d4iy22sHjQoEFeu86dO1u8yirf/Xr65JNPvHbDhg2z+JZbbrF4/vz5Ra8fxen91ThMO7Vr187i7t27W7z//vt77XbffXeLu3btavH06dO9dsOHD7f4jjvusJj7mZz+/XPOv2ea+g3TS9pOnzn9G+qccyeeeKLFa6+9tsVTp0712v3zn/+0+O6777Z47ty5XrtSpbKi6uRgxeW58WXaxQeTSru5aNprohgZAKoL6SoAABAlOjkAACBKdcWG2+vq6qpqLL558+YWL1iwINNz33zzzRYff/zxmZ47VF9fn11ORzR0P/NMVyU9V1qlTlcluYZCH5H6AwqdsEzPZtOmTb3XLVu2tFhrM3r06OG1+9GPfmTxTjvtZPE666zjtdNaHr0vWlPgnJ/ff+yxxyz+5S9/6bWbM2fO9/8RjVDKZ7MUwp99rc9YddVVLW7fvr3X7oc//KHFBx98sMXbbbed105rd7RmpNj91DqOiy++2Gv35Zdffv8f0Qh53M9y3UutiXPOr70p9ntKn7ldd93V4j/+8Y9eO32+tXbn66+/9trNnDnT4gceeKDg+T7//POC15RGoXvJSA4AAIgShcfw5L3jeNpi5CTvy/LcDclyFAoAkL+KTFfpELhz/pTTpUuXWjxmzBivXZ7pKh36C6eQDxw4sOA1pVHOdFXarREak9Kq5k5OLaSrNIXRoUMH79hWW21lsU4TD6eQb7TRRhZriqpbt25eO02RFFsZWb93/Z3w+OOPe+2GDBlicRbTkWNIVxVafdo5//vX+7Tpppt67Q455BCLt9xyS4s7derktSt2DwtZsmSJxZqKdM65ww8/3OIslvKo9nSV3q9WrVp5x3R5B00vhfdc08dnnXWWxeuvv77XTt9X7PeqPptfffWVxQ899JDXbujQoRbPnj274PmSIl0FAABqCp0cAAAQpYpMV11++eXe61/84hcWaxrq5z//udfuX//6l8W33nqrxYMHD874Cn0DBgywePTo0Y0+X15D4s655d7PLGco5Vl/0xhZzgxLmLarqiHx8N/Qs2dPi0877TTv2CabbGKxpnQ1deycn8bQY+HQ+cKFCy1u27atxWH6S4fi9Xo1deWcc/fcc4/FRxxxhGusakxXhWkiLQfo0qWLd2zDDTe0eOutt7a4X79+XrvevXtbrPdJ0yfFriPp86ypK+ecu//++y3WVKRz6Z7rans29efeOX822w477OAdW7x4scU64ymc+bjffvtZrCmqcCalzqIqtFmnc/59LpRWds65v//97xaHf8vTbNhKugoAANQUOjkAACBKdHIAAECUylaTE04Tv+iiiyw+6aSTvGNhHvJ/rr/+eu/1KaecYrHufnvppZd67X7wgx9YnEWtx/jx4y3eeOONG30+anKoyUlwLZk+m5pX1x2GnfOnlYa7huvUVF3FNsy/a62NtpsxY4bXTncb112rV199da/duuuua3FYO6B0xWNdisI556ZMmVLwfYVUS02O3hfd2d0553beeWeLDz30UO+YrmqrdVRhrY1+53qs2HOq1xQqVK8TPl96P8OanHC6eRLV8Gzqdx2uKK2r7/ft29c7pqtDa41LeC91NWutfQtXMtbnVv9+hzU5eo7wmJo1a5bF4Y7nr7zyisVJf18XupcsBlhDkvyRTtrBSNtRSHL+SujQJJVkrR4AQHmQrgIAAFEq20hOmEI68cQTV/gcutKmc/7UOB26DIcx77jjDosPO+wwi6+++uqC5w+n5yldGTQcdgtXeQQqhaYIdOXhcCM9XclYpwsXE26kOGnSJIvvvfdei9977z2vnQ6d6wrFeg3OObfLLrtYrEP44SaFbdq0sTicBv3xxx9bHG4YWe00ZXDAAQd4x3S67lprreUd0+9P00vLli3z2um9ef/99y0OUxyactQUZjgCqmmNNdZYw+KwVEHv57777usde+mllyz+4osvXDXT72fNNde0+PTTT/fa6TT/cJkFTVPq+cJp+bpy9MiRIy0On2GdXq6p3jANqe20fCNMK+vmrXvssYd37M033yx4vSuKkRwAABAlOjkAACBKFB7XkLxnLaU9d9rNPpOcP++iaQBA5cq9k6NLuP/617+2+Ljjjmv0uTUf6Zy//PzkyZMLvu/UU0+1+Nprr7X4rbfe8tqtttpqFo8YMcLi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\n",
      "text/plain": [
       "<Figure size 720x720 with 25 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "(x_train, _), (x_test, y_test) = tf.keras.datasets.mnist.load_data()\n",
    "x_train = x_train.astype('float32') / 255.\n",
    "x_train = x_train.reshape((len(x_train), np.prod(x_train.shape[1:])))\n",
    "x_test = x_test.astype('float32') / 255.\n",
    "x_test = x_test.reshape((len(x_test), np.prod(x_test.shape[1:])))\n",
    "\n",
    "# Model Forward\n",
    "logits_y, logits_x1 = model(x_test[10:110], tau=1.0, hard=True)\n",
    "logits_y, logits_x2 = model(x_test[10:110], tau=1.0, hard=True)\n",
    "logits_y, logits_x3 = model(x_test[10:110], tau=1.0, hard=True)\n",
    "\n",
    "\n",
    "sample_y = model.sample_y.numpy()\n",
    "\n",
    "logits_x1 = tf.sigmoid(logits_x1).numpy()\n",
    "logits_x2 = tf.sigmoid(logits_x2).numpy()\n",
    "logits_x3 = tf.sigmoid(logits_x3).numpy()\n",
    "\n",
    "print(\"sample_y: \", sample_y.shape, \", logits_x.shape:\", logits_x1.shape)\n",
    "code = model.sample_y\n",
    "\n",
    "\n",
    "def save_plt(original_img, construct_img1, construct_img2, construct_img3, code):\n",
    "    plt.figure(figsize=(10, 10))\n",
    "    for i in range(0, 25, 5):\n",
    "        # input img\n",
    "        plt.subplot(5, 5, i+1)\n",
    "        plt.imshow(original_img[i, :].reshape(28, 28), cmap='gray')\n",
    "        plt.axis('off')\n",
    "\n",
    "        # code\n",
    "        plt.subplot(5, 5, i+2)\n",
    "        plt.imshow(code[i, :, :], cmap='gray')\n",
    "        plt.axis('off')\n",
    "\n",
    "        # output im\n",
    "        plt.subplot(5, 5, i+3)\n",
    "        plt.imshow(construct_img1[i, :,].reshape((28, 28)), cmap='gray')\n",
    "        plt.axis('off')\n",
    "        \n",
    "        # output im\n",
    "        plt.subplot(5, 5, i+4)\n",
    "        plt.imshow(construct_img2[i, :,].reshape((28, 28)), cmap='gray')\n",
    "        plt.axis('off')\n",
    "        \n",
    "        # output im\n",
    "        plt.subplot(5, 5, i+5)\n",
    "        plt.imshow(construct_img3[i, :,].reshape((28, 28)), cmap='gray')\n",
    "        plt.axis('off')\n",
    "\n",
    "    #plt.savefig('vae-pic/vae_rebuilt.png')\n",
    "\n",
    "save_plt(x_test[10:110], logits_x1,  logits_x2,  logits_x3, code)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def make_squares(images, nr_images_per_side):\n",
    "    images_to_plot = np.concatenate(\n",
    "        [np.concatenate([images[j*nr_images_per_side+i].reshape((28,28)) for i in range(0,nr_images_per_side)],\n",
    "                        axis=1)\n",
    "         for j in range(0,nr_images_per_side)],\n",
    "        axis=0)\n",
    "    return images_to_plot\n",
    "\n",
    "def plot_squares(originals, reconstructs, nr_images_per_side):\n",
    "    originals_square = make_squares(originals, nr_images_per_side)\n",
    "    plt.imsave('original.pdf', originals_square, cmap='viridis', format='pdf')\n",
    "    reconstructs_square = make_squares(reconstructs, nr_images_per_side)\n",
    "    plt.imsave('recons.pdf', reconstructs_square, cmap='viridis', format='pdf')\n",
    "    combined = np.concatenate([originals_square, reconstructs_square], axis=1)\n",
    "    plt.imsave('combined.pdf', combined, cmap='viridis', format='pdf')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plot_squares(x_test[10:110], logits_x1, 8)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "gumbel",
   "language": "python",
   "name": "gumbel"
  },
  "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.8.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
