{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### This notebook is taken 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 IMLESubsetkLayer(tf.keras.layers.Layer):\n",
    "  \n",
    "    def __init__(self, _k=10, _tau=10.0, _lambda=10.0):\n",
    "        super(IMLESubsetkLayer, 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",
    "        forward_sample = self.sample_gumbel_k(tf.shape(logits))\n",
    "        gumbel_softmax_sample = logits + forward_sample\n",
    "        threshold = tf.expand_dims(tf.nn.top_k(gumbel_softmax_sample, self.k, sorted=True)[0][:,-1], -1)\n",
    "        y_map = tf.cast(tf.greater_equal(gumbel_softmax_sample, threshold), tf.float32)\n",
    "\n",
    "        def custom_grad(dy):\n",
    "            \n",
    "            # perturb and MAP on the target distribution parameters\n",
    "            gumbel_softmax_sample = logits - (self._lambda*dy) + forward_sample\n",
    "            threshold = tf.expand_dims(tf.nn.top_k(gumbel_softmax_sample, self.k, sorted=True)[0][:,-1], -1)\n",
    "            map_y = tf.cast(tf.greater_equal(gumbel_softmax_sample, threshold), tf.float32)    \n",
    "            # now compute the gradient of the conditional log-likelihood\n",
    "            grad = (1.0 / self._lambda) * tf.math.subtract(y_map, map_y)\n",
    "\n",
    "            # return the gradient function\n",
    "            return grad, hard\n",
    "\n",
    "        return y_map, custom_grad\n",
    "\n",
    "    def call(self, logits, hard=False):\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 = IMLESubsetkLayer(_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": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "model.build()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 1 , TRAIN loss: 111.35895 , Temperature: 0.9997000449955004\n",
      "Eval Loss: 107.25453 \n",
      "\n",
      "Epoch: 2 , TRAIN loss: 96.3875 , Temperature: 0.9991004048785274\n",
      "Eval Loss: 97.03229 \n",
      "\n",
      "Epoch: 3 , TRAIN loss: 96.13809 , Temperature: 0.9982016190284373\n",
      "Eval Loss: 93.88198 \n",
      "\n",
      "Epoch: 4 , TRAIN loss: 94.87975 , Temperature: 0.997004495503373\n",
      "Eval Loss: 90.70981 \n",
      "\n",
      "Epoch: 5 , TRAIN loss: 88.63848 , Temperature: 0.9955101098295706\n",
      "Eval Loss: 89.02783 \n",
      "\n",
      "Epoch: 6 , TRAIN loss: 87.88991 , Temperature: 0.9937198033910547\n",
      "Eval Loss: 87.591515 \n",
      "\n",
      "Epoch: 7 , TRAIN loss: 89.2109 , Temperature: 0.9916351814230984\n",
      "Eval Loss: 86.7343 \n",
      "\n",
      "Epoch: 8 , TRAIN loss: 86.708084 , Temperature: 0.9892581106136482\n",
      "Eval Loss: 86.02666 \n",
      "\n",
      "Epoch: 9 , TRAIN loss: 86.708145 , Temperature: 0.9865907163177327\n",
      "Eval Loss: 85.17785 \n",
      "\n",
      "Epoch: 10 , TRAIN loss: 82.90347 , Temperature: 0.9836353793906725\n",
      "Eval Loss: 84.94581 \n",
      "\n",
      "Epoch: 11 , TRAIN loss: 85.80649 , Temperature: 0.9803947326466972\n",
      "Eval Loss: 84.31941 \n",
      "\n",
      "Epoch: 12 , TRAIN loss: 82.052864 , Temperature: 0.9768716569503434\n",
      "Eval Loss: 83.894806 \n",
      "\n",
      "Epoch: 13 , TRAIN loss: 83.08639 , Temperature: 0.9730692769487556\n",
      "Eval Loss: 83.79714 \n",
      "\n",
      "Epoch: 14 , TRAIN loss: 82.406235 , Temperature: 0.9689909564537397\n",
      "Eval Loss: 83.11628 \n",
      "\n",
      "Epoch: 15 , TRAIN loss: 83.9195 , Temperature: 0.964640293483123\n",
      "Eval Loss: 83.25606 \n",
      "\n",
      "Epoch: 16 , TRAIN loss: 83.02221 , Temperature: 0.9600211149716509\n",
      "Eval Loss: 82.68739 \n",
      "\n",
      "Epoch: 17 , TRAIN loss: 80.08184 , Temperature: 0.9551374711623026\n",
      "Eval Loss: 82.622406 \n",
      "\n",
      "Epoch: 18 , TRAIN loss: 80.630455 , Temperature: 0.9499936296895314\n",
      "Eval Loss: 82.37978 \n",
      "\n",
      "Epoch: 19 , TRAIN loss: 84.886925 , Temperature: 0.9445940693665233\n",
      "Eval Loss: 82.11848 \n",
      "\n",
      "Epoch: 20 , TRAIN loss: 80.36934 , Temperature: 0.9389434736891332\n",
      "Eval Loss: 82.00074 \n",
      "\n",
      "Epoch: 21 , TRAIN loss: 80.84902 , Temperature: 0.9330467240696794\n",
      "Eval Loss: 81.36838 \n",
      "\n",
      "Epoch: 22 , TRAIN loss: 85.88857 , Temperature: 0.9269088928142736\n",
      "Eval Loss: 81.79444 \n",
      "\n",
      "Epoch: 23 , TRAIN loss: 84.59303 , Temperature: 0.9205352358578188\n",
      "Eval Loss: 81.28827 \n",
      "\n",
      "Epoch: 24 , TRAIN loss: 80.55386 , Temperature: 0.9139311852712282\n",
      "Eval Loss: 81.571144 \n",
      "\n",
      "Epoch: 25 , TRAIN loss: 78.75278 , Temperature: 0.9071023415558017\n",
      "Eval Loss: 81.17473 \n",
      "\n",
      "Epoch: 26 , TRAIN loss: 85.36581 , Temperature: 0.9000544657400421\n",
      "Eval Loss: 80.84536 \n",
      "\n",
      "Epoch: 27 , TRAIN loss: 83.939255 , Temperature: 0.8927934712944959\n",
      "Eval Loss: 80.74687 \n",
      "\n",
      "Epoch: 28 , TRAIN loss: 79.82164 , Temperature: 0.8853254158804752\n",
      "Eval Loss: 80.80222 \n",
      "\n",
      "Epoch: 29 , TRAIN loss: 81.546036 , Temperature: 0.8776564929487385\n",
      "Eval Loss: 80.33481 \n",
      "\n",
      "Epoch: 30 , TRAIN loss: 78.92857 , Temperature: 0.8697930232043984\n",
      "Eval Loss: 80.33829 \n",
      "\n",
      "Epoch: 31 , TRAIN loss: 84.5845 , Temperature: 0.8617414459544691\n",
      "Eval Loss: 80.59088 \n",
      "\n",
      "Epoch: 32 , TRAIN loss: 83.78554 , Temperature: 0.8535083103545701\n",
      "Eval Loss: 80.55267 \n",
      "\n",
      "Epoch: 33 , TRAIN loss: 82.34808 , Temperature: 0.8451002665713722\n",
      "Eval Loss: 80.321335 \n",
      "\n",
      "Epoch: 34 , TRAIN loss: 77.63566 , Temperature: 0.8365240568773926\n",
      "Eval Loss: 80.191475 \n",
      "\n",
      "Epoch: 35 , TRAIN loss: 79.91792 , Temperature: 0.8277865066947337\n",
      "Eval Loss: 80.06816 \n",
      "\n",
      "Epoch: 36 , TRAIN loss: 80.79948 , Temperature: 0.8188945156043044\n",
      "Eval Loss: 79.663635 \n",
      "\n",
      "Epoch: 37 , TRAIN loss: 80.96091 , Temperature: 0.8098550483369699\n",
      "Eval Loss: 79.58615 \n",
      "\n",
      "Epoch: 38 , TRAIN loss: 80.19628 , Temperature: 0.8006751257629464\n",
      "Eval Loss: 80.06235 \n",
      "\n",
      "Epoch: 39 , TRAIN loss: 80.21133 , Temperature: 0.791361815895584\n",
      "Eval Loss: 79.66266 \n",
      "\n",
      "Epoch: 40 , TRAIN loss: 80.91074 , Temperature: 0.7819222249254774\n",
      "Eval Loss: 79.571686 \n",
      "\n",
      "Epoch: 41 , TRAIN loss: 80.02299 , Temperature: 0.7723634883006051\n",
      "Eval Loss: 79.71212 \n",
      "\n",
      "Epoch: 42 , TRAIN loss: 79.38112 , Temperature: 0.7626927618679156\n",
      "Eval Loss: 79.55637 \n",
      "\n",
      "Epoch: 43 , TRAIN loss: 79.4232 , Temperature: 0.7529172130914742\n",
      "Eval Loss: 79.5078 \n",
      "\n",
      "Epoch: 44 , TRAIN loss: 84.66866 , Temperature: 0.74304401236194\n",
      "Eval Loss: 79.45644 \n",
      "\n",
      "Epoch: 45 , TRAIN loss: 76.50813 , Temperature: 0.7330803244117685\n",
      "Eval Loss: 79.08239 \n",
      "\n",
      "Epoch: 46 , TRAIN loss: 77.24256 , Temperature: 0.7230332998501351\n",
      "Eval Loss: 78.811165 \n",
      "\n",
      "Epoch: 47 , TRAIN loss: 79.50446 , Temperature: 0.7129100668311394\n",
      "Eval Loss: 79.00496 \n",
      "\n",
      "Epoch: 48 , TRAIN loss: 74.53944 , Temperature: 0.7027177228683977\n",
      "Eval Loss: 79.3904 \n",
      "\n",
      "Epoch: 49 , TRAIN loss: 75.41926 , Temperature: 0.6924633268086435\n",
      "Eval Loss: 79.22826 \n",
      "\n",
      "Epoch: 50 , TRAIN loss: 79.15691 , Temperature: 0.6821538909764523\n",
      "Eval Loss: 78.97445 \n",
      "\n",
      "Epoch: 51 , TRAIN loss: 78.24916 , Temperature: 0.6717963735016784\n",
      "Eval Loss: 78.96268 \n",
      "\n",
      "Epoch: 52 , TRAIN loss: 79.557594 , Temperature: 0.6613976708406429\n",
      "Eval Loss: 79.01817 \n",
      "\n",
      "Epoch: 53 , TRAIN loss: 74.21629 , Temperature: 0.6509646105015452\n",
      "Eval Loss: 78.712555 \n",
      "\n",
      "Epoch: 54 , TRAIN loss: 79.30884 , Temperature: 0.6405039439839885\n",
      "Eval Loss: 78.4662 \n",
      "\n",
      "Epoch: 55 , TRAIN loss: 75.90893 , Temperature: 0.6300223399419125\n",
      "Eval Loss: 78.79699 \n",
      "\n",
      "Epoch: 56 , TRAIN loss: 76.883224 , Temperature: 0.6195263775786136\n",
      "Eval Loss: 78.47663 \n",
      "\n",
      "Epoch: 57 , TRAIN loss: 78.19016 , Temperature: 0.609022540281914\n",
      "Eval Loss: 78.77027 \n",
      "\n",
      "Epoch: 58 , TRAIN loss: 76.861946 , Temperature: 0.5985172095069092\n",
      "Eval Loss: 78.12014 \n",
      "\n",
      "Epoch: 59 , TRAIN loss: 73.619446 , Temperature: 0.5880166589130854\n",
      "Eval Loss: 78.28504 \n",
      "\n",
      "Epoch: 60 , TRAIN loss: 78.89215 , Temperature: 0.5775270487619548\n",
      "Eval Loss: 78.12171 \n",
      "\n",
      "Epoch: 61 , TRAIN loss: 76.98508 , Temperature: 0.5670544205807092\n",
      "Eval Loss: 78.19303 \n",
      "\n",
      "Epoch: 62 , TRAIN loss: 75.32697 , Temperature: 0.556604692096744\n",
      "Eval Loss: 78.41015 \n",
      "\n",
      "Epoch: 63 , TRAIN loss: 76.93637 , Temperature: 0.5461836524472542\n",
      "Eval Loss: 78.22734 \n",
      "\n",
      "Epoch: 64 , TRAIN loss: 77.347115 , Temperature: 0.5357969576674562\n",
      "Eval Loss: 77.85489 \n",
      "\n",
      "Epoch: 65 , TRAIN loss: 79.19493 , Temperature: 0.5254501264603462\n",
      "Eval Loss: 78.03633 \n",
      "\n",
      "Epoch: 66 , TRAIN loss: 76.47484 , Temperature: 0.5151485362502642\n",
      "Eval Loss: 78.01187 \n",
      "\n",
      "Epoch: 67 , TRAIN loss: 84.01845 , Temperature: 0.5048974195219025\n",
      "Eval Loss: 77.76 \n",
      "\n",
      "Epoch: 68 , TRAIN loss: 77.28819 , Temperature: 0.5\n",
      "Eval Loss: 77.700066 \n",
      "\n",
      "Epoch: 69 , TRAIN loss: 77.137474 , Temperature: 0.5\n",
      "Eval Loss: 77.75698 \n",
      "\n",
      "Epoch: 70 , TRAIN loss: 78.205444 , Temperature: 0.5\n",
      "Eval Loss: 78.07223 \n",
      "\n",
      "Epoch: 71 , TRAIN loss: 80.52385 , Temperature: 0.5\n",
      "Eval Loss: 77.37288 \n",
      "\n",
      "Epoch: 72 , TRAIN loss: 76.68891 , Temperature: 0.5\n",
      "Eval Loss: 77.48441 \n",
      "\n",
      "Epoch: 73 , TRAIN loss: 78.11201 , Temperature: 0.5\n",
      "Eval Loss: 77.382744 \n",
      "\n",
      "Epoch: 74 , TRAIN loss: 77.16775 , Temperature: 0.5\n",
      "Eval Loss: 77.38472 \n",
      "\n",
      "Epoch: 75 , TRAIN loss: 80.645424 , Temperature: 0.5\n",
      "Eval Loss: 77.77601 \n",
      "\n",
      "Epoch: 76 , TRAIN loss: 76.03426 , Temperature: 0.5\n",
      "Eval Loss: 77.879234 \n",
      "\n",
      "Epoch: 77 , TRAIN loss: 75.20672 , Temperature: 0.5\n",
      "Eval Loss: 77.278206 \n",
      "\n",
      "Epoch: 78 , TRAIN loss: 79.165 , Temperature: 0.5\n",
      "Eval Loss: 77.424095 \n",
      "\n",
      "Epoch: 79 , TRAIN loss: 79.02695 , Temperature: 0.5\n",
      "Eval Loss: 77.26222 \n",
      "\n",
      "Epoch: 80 , TRAIN loss: 76.47981 , Temperature: 0.5\n",
      "Eval Loss: 77.23355 \n",
      "\n",
      "Epoch: 81 , TRAIN loss: 75.71287 , Temperature: 0.5\n",
      "Eval Loss: 77.09841 \n",
      "\n",
      "Epoch: 82 , TRAIN loss: 81.66651 , Temperature: 0.5\n",
      "Eval Loss: 77.174225 \n",
      "\n",
      "Epoch: 83 , TRAIN loss: 78.54219 , Temperature: 0.5\n",
      "Eval Loss: 77.56864 \n",
      "\n",
      "Epoch: 84 , TRAIN loss: 74.44122 , Temperature: 0.5\n",
      "Eval Loss: 77.258446 \n",
      "\n",
      "Epoch: 85 , TRAIN loss: 75.351944 , Temperature: 0.5\n",
      "Eval Loss: 77.3978 \n",
      "\n",
      "Epoch: 86 , TRAIN loss: 77.25392 , Temperature: 0.5\n",
      "Eval Loss: 77.14561 \n",
      "\n",
      "Epoch: 87 , TRAIN loss: 76.92785 , Temperature: 0.5\n",
      "Eval Loss: 77.03067 \n",
      "\n",
      "Epoch: 88 , TRAIN loss: 77.959564 , Temperature: 0.5\n",
      "Eval Loss: 77.39931 \n",
      "\n",
      "Epoch: 89 , TRAIN loss: 76.33485 , Temperature: 0.5\n",
      "Eval Loss: 76.83479 \n",
      "\n",
      "Epoch: 90 , TRAIN loss: 72.927864 , Temperature: 0.5\n",
      "Eval Loss: 76.98566 \n",
      "\n",
      "Epoch: 91 , TRAIN loss: 75.37268 , Temperature: 0.5\n",
      "Eval Loss: 76.82079 \n",
      "\n",
      "Epoch: 92 , TRAIN loss: 76.071815 , Temperature: 0.5\n",
      "Eval Loss: 76.80871 \n",
      "\n",
      "Epoch: 93 , TRAIN loss: 77.55635 , Temperature: 0.5\n",
      "Eval Loss: 76.834816 \n",
      "\n",
      "Epoch: 94 , TRAIN loss: 79.668594 , Temperature: 0.5\n",
      "Eval Loss: 76.62504 \n",
      "\n",
      "Epoch: 95 , TRAIN loss: 80.15595 , Temperature: 0.5\n",
      "Eval Loss: 76.71642 \n",
      "\n",
      "Epoch: 96 , TRAIN loss: 78.03499 , Temperature: 0.5\n",
      "Eval Loss: 76.797806 \n",
      "\n",
      "Epoch: 97 , TRAIN loss: 74.291115 , Temperature: 0.5\n",
      "Eval Loss: 76.51579 \n",
      "\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 98 , TRAIN loss: 74.88138 , Temperature: 0.5\n",
      "Eval Loss: 76.603874 \n",
      "\n",
      "Epoch: 99 , TRAIN loss: 76.99597 , Temperature: 0.5\n",
      "Eval Loss: 76.43024 \n",
      "\n",
      "Epoch: 100 , TRAIN loss: 74.36385 , Temperature: 0.5\n",
      "Eval Loss: 76.523636 \n",
      "\n",
      "CPU times: user 1h 25min 2s, sys: 5min 14s, total: 1h 30min 17s\n",
      "Wall time: 22min 38s\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": 15,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[107.25453,\n",
       " 97.03229,\n",
       " 93.88198,\n",
       " 90.70981,\n",
       " 89.02783,\n",
       " 87.591515,\n",
       " 86.7343,\n",
       " 86.02666,\n",
       " 85.17785,\n",
       " 84.94581,\n",
       " 84.31941,\n",
       " 83.894806,\n",
       " 83.79714,\n",
       " 83.11628,\n",
       " 83.25606,\n",
       " 82.68739,\n",
       " 82.622406,\n",
       " 82.37978,\n",
       " 82.11848,\n",
       " 82.00074,\n",
       " 81.36838,\n",
       " 81.79444,\n",
       " 81.28827,\n",
       " 81.571144,\n",
       " 81.17473,\n",
       " 80.84536,\n",
       " 80.74687,\n",
       " 80.80222,\n",
       " 80.33481,\n",
       " 80.33829,\n",
       " 80.59088,\n",
       " 80.55267,\n",
       " 80.321335,\n",
       " 80.191475,\n",
       " 80.06816,\n",
       " 79.663635,\n",
       " 79.58615,\n",
       " 80.06235,\n",
       " 79.66266,\n",
       " 79.571686,\n",
       " 79.71212,\n",
       " 79.55637,\n",
       " 79.5078,\n",
       " 79.45644,\n",
       " 79.08239,\n",
       " 78.811165,\n",
       " 79.00496,\n",
       " 79.3904,\n",
       " 79.22826,\n",
       " 78.97445,\n",
       " 78.96268,\n",
       " 79.01817,\n",
       " 78.712555,\n",
       " 78.4662,\n",
       " 78.79699,\n",
       " 78.47663,\n",
       " 78.77027,\n",
       " 78.12014,\n",
       " 78.28504,\n",
       " 78.12171,\n",
       " 78.19303,\n",
       " 78.41015,\n",
       " 78.22734,\n",
       " 77.85489,\n",
       " 78.03633,\n",
       " 78.01187,\n",
       " 77.76,\n",
       " 77.700066,\n",
       " 77.75698,\n",
       " 78.07223,\n",
       " 77.37288,\n",
       " 77.48441,\n",
       " 77.382744,\n",
       " 77.38472,\n",
       " 77.77601,\n",
       " 77.879234,\n",
       " 77.278206,\n",
       " 77.424095,\n",
       " 77.26222,\n",
       " 77.23355,\n",
       " 77.09841,\n",
       " 77.174225,\n",
       " 77.56864,\n",
       " 77.258446,\n",
       " 77.3978,\n",
       " 77.14561,\n",
       " 77.03067,\n",
       " 77.39931,\n",
       " 76.83479,\n",
       " 76.98566,\n",
       " 76.82079,\n",
       " 76.80871,\n",
       " 76.834816,\n",
       " 76.62504,\n",
       " 76.71642,\n",
       " 76.797806,\n",
       " 76.51579,\n",
       " 76.603874,\n",
       " 76.43024,\n",
       " 76.523636]"
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "results"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "sample_y:  (100, 20, 20) , logits_x.shape: (100, 784)\n"
     ]
    },
    {
     "data": {
      "image/png": "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J6eGJCmWl3fxvCTtsvxjlPQasnofAKBykK4CAABRopMDAACiVNJ01aeffmpxpewgq/nOU045xTu21157WTx16tSC7ah4jGqk6bfmzZtbHM5/0TkS4VyKxs7RCZeG6/n165522mleO62Kq9WP77nnHq9dzFVxQ4WWkIc7eX/77bcNvicP+vtVyxTovA3n/PupqeKbbrrJa/fee+9lfYkVT3fvDu+XPrfhbvNh6YDFFX4tfc60Qvodd9zhtdPnVqtrX3TRRV67cO5fXhjJAQAAUWLicY3LalJxljuAJ5Xkf6FpKyUzyRgAql9JOzlafVTTP84517p1a4s7duzoHWvs5p1agdE5544++miL+/TpY/GGG25Y8Bz777+/xa+99lqjrgeoNLqUOExXHXDAARa/88473rF///vfFmsaJNxIUOkGfiussIJ3bLPNNrP42GOPtVgrHDvnV3597rnnLL7lllu8drWUStafyauvvmrxlltu6bU79NBDLdbfyc45N3r06AbPV2wVplYrXnnllb12u+66q8Unnniixbrs3zm/yvawYcMs/vOf/1ywXcz0P2JasmTHHXf02q211loW77HHHt6x2267zWJNXYX3UlPGuiHqSiut5LXr37+/xWeeeabF4d9rrWx84403WvzUU0957UpV0oF0FQAAiBKdHAAAEKW6YkNGdXV1uY0n6SZxzjnXu3dvi9966y3v2LRp0xr1tTbZZBPvdYcOHRpsF6bFHnjgAYuPP/54i8ONxrJWX1+fy4SQhu5nloX+El7DL54ry4KEWUoy76jAdWZ+YVk8mzpMve+++1p8/fXXe+00HRGu2NDNO6dMmVKwnabAunbtanH4LOpqEf3ZhmmKl19+2eLBgwdbHFZRzXrDyVI+m42x4oorWhymCbp162ZxuMp1+PDhFmtl5PBz3rdvX4s33XRTi7XSsnP+Z0c/b2F13uuuu85iXUmnKdA85HE/s76XmtINq+137tzZ4nC10oMPPmjxyJEjC7bbeOONLdaq/7oxqHP+Kj1daRXey7vvvttinRqS92qqQveSkRwAABAlOjkAACBKdHIAAECUyjYnZ8CAAd7rs846y+L1118/ry/rnPPz9LNnz7b4sssu89pdeOGFuV5HIczJYU7OL8n62WzRooXFOj/COed23313i3WJ6f+/jgbPl+SeNtROd8weN26cxbff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rncs1vupz1K/fv28YyeccELB9+mqqeHDh1u8aNGigufXNKfeB+eca9WqlcWawg4rT++8884W633V1LZzzt13330W62fNuWyrMDOSAwAAokQnBwAARIlODgAAiFLFLCHXvOARRxzhHRs0aJDFI0aMsDicW6GViLWColYkDt+nedzwfGPHjrVYlysOGTLEaxdWawSqjS7ZnDt3rndMc/Prrbeed+yWW26xePLkyRbPmTPHa6f5fJ3Dof/unF8Fdfnll7c4fDZ1+fcbb7xh8YUXXui1e/HFFy2OdYnx/+jvMl3+PWnSJK+dzndo27atd+z444+3WH/vhlVndVm6zv8J53HonCu912Hl7I8//tjiK6+80uJ7773Xa6efzVjn4IT0fn366afeMf0ZaPVj5/z5VUcddZTFYVX+VVdd1WKd39auXTuvnVabbtasWYPX55w/Z1Xn3VxzzTVeu3HjxhU8R5YqppOD/KXdFTzJ+xpTBC+rYoB57qie9zUAALJHugoAAESprtj/POvq6kr239Ltt9/e4jC9pHRZty4Fd84fhtNKnpp2cs5PL+lwWkiXg5eyOmp9fX36oYVfOHX4D1mN5FTq+5IowShU5vczz2czfK50Seiaa67pHdNUhQ51h2korXCrPyNdsuqcc/Pnz7dYU15aidc558aPH2/xsGHDLA5TbVlXd83r2cz6furnUqsJO+dcr169LA7vtaah1l13XYt1KXh4Tk0DhilBTUl8/vnnFo8ePdpr99lnn1msv7vzHhnN437m+Wxqmsg55371q19ZrM+pc85tueWWFnfu3Nni8JkrtFFtuNS80P176aWXvHaaItY0ZPjZyPreFrqXjOQAAIAo0ckBAABRYuJxDcky5ZOk2nBjJvT+0tdrSNK0U56ThdmFHAAqR8XMycFPypn3L8cKpTyvIe9OTsKOY1Xl/VFYtczJQTI8m/FgTg4AAKgpdHIAAECU6OQAAIAoMfG4huQ53yZpbZu0E4jTynIicJ61egAA2WMkBwAARIlODgAAiBKdHAAAECXm5NSQtHNKSr2XVKklnbeT9ucAACgPRnIAAECU6OQAAIAo0ckBAABRopMDAACiVHSDTgAAgGrFSA4AAIgSnRwAABAlOjkAACBKdHIAAECU6OQAAIAo0ckBAABRopMDAACiRCcHAABEiU4OAACIEp0cAAAQJTo5AAAgSnRyAABAlOjkAACAKNHJAQAAUaKTAwAAokQnBwAARIlODgAAiBKdHAAAEKWlih2sq6urL9WF4Cf19fV1eZ06/Ie6ul/+UvX1v/wxaOg8Db0vq6/XGEmuIanwWgv8HDK/nzyb5ZHXs8n9LA+ezXgUupeM5AAAgCjRyQEAAFGikwMAAKJEJwcAAESp6MRjwLnkk4pL/fXCdmmvqTHfS1bXAADIHiM5AAAgSnRyAABAlOjkAACAKDEnp4ZkObcmSUG9Un+9LK8hy4KBAIDyYCQHAABEiU4OAACIEp0cAAAQJTo5AAAgSkw8rnFJitmlnYSbdJJxnjuhp5XlBGkKBAJAeTCSAwAAokQnBwAARIlODgAAiBKdHAAAECUmHteQtJN30+4AnuWE5bSTk/OuzMwu5ABQuWqik9OtWzfv9XHHHWfxRhttZPExxxzjtRszZky+FwbUoKWXXtrinj17esdWWWUVi7fddluLe/To4bV78MEHLb799tstnjdvnteOTmdpLbHEEgVfL7PMMha3bt3aa/ftt99aPGfOHIu///77jK8QIb1H+vxdc801Xrt11lnH4h9//NHid99912v3xz/+0eK3337b4h9++KHxF5sC6SoAABAlOjkAACBKdcWGc+vq6qpqrFeHvo899liLDzzwQK9dOFT6P1OnTvVe77LLLhZ37drV4kmTJnnt3nvvvcW/2CLq6+tz2QK7ofuZ5TydJO9LIuncmqyuoTHziRKmQzK/n5X+bIZpi969e1v8l7/8xeL+/ft77Zo3b97g+cKh7i+//NLiiy++2OKrr77aa/fdd98lvOJkSvlsVpKllvJnNqy88soWH3300RaH97N9+/YWt2jRwuJmzZp57fR5++ijjyz+7W9/67V76623LM4iFZnH/az0e7nkkkt6r7faaiuLb775ZotXWGEFr53eI/3ZL1q0yGs3evRoi/WzoffOOT/llYVC95KRHAAAECU6OQAAIEp0cgAAQJSqbk6O5vpXX31179iTTz5pcadOnRr9tXQ5aqtWrSx+5ZVXvHabb765xVnkGZmTw5ycXzxhBT6bujRc8/zOOffXv/7V4l69elncpEmTgufTn234XOmS4zfeeMPiQYMGee107k4WamlOjs6POuKII7xjp556qsXLLrusxeHcnWL3UOk8EV02ftVVV3ntfv/731ucxZLkWpmTo/dF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5557zuK11lrLa6dDo6uttprFixYtyu/iFkMs6aqk8uzk5P391Fq6qhj93sO01oorrmixVsvVJcbO+UvD9bkNl72WSy2lq4rR+/bqq69arMuWQ7q8PJwaUK7J+bXybKrwd5TuMPDss89arH8bnfNLOmiaOUxrVdq9JF0FAACiRCcHAABEKao5OQ8++KDFYYpK6eqOSklRAdWu2DD1ySefbLFWTg3fM2zYMIs//fTT7C4OjbLEEv7/h/V+du/e3eIwFaIVsk888USLqR1VPrrizTl/M+swRaV0Y12ttFzp9zKqTg4aL8u5L0nPlWReUJYPUpbFB2PchRwAYkG6CgAARIlODgAAiFJVLyFfZZVVvNe6Q7nuQv7www977XbbbTeLw93QK0E5l5CnPnGJqxs3RhnSVTW3THWdddbxXr/22msWN2vWzOJ58+Z57VZffXWLp06dmtPVpVerS8g33nhj7/ULL7xgcdOmTS0O5zjq79pwx/lKUCtLyJs0aWLxKaec4h3TOapa/VjnUznn3CabbGJxpe007hxLyAEAQI1h4nENyXO376TnSnINWe6DleT7STtqAwCobFXXyenSpYvFTz/9tHdMU1Q6nHbMMcd47SoxRQVUO01bhFVQNUWlnco77rjDazd9+vScrg6Lq1WrVhbffffd3jG91yrcDPmpp57K/sKQiFYd33777S3+4x//6LXTFJVWNT722GO9dmH6qlqQrgIAAFGikwMAAKJUdemqPn36WNytWzfvmM6Z0M3gJk+enP+FVYG0G2FmuYFmkrk1Wc6/STuPJssd1GvFpptuavG6665bsN3cuXMtvvTSS71jpJLLSz/3WtW4a9euBd/zySefWHzaaad5x7777rsMrw7FhL+zOnXqZLFWNdbUsXPO/fjjjxbrSuQnnnjCa1etzyYjOQAAIEp0cgAAQJTo5AAAgChVxZwcrbapu5+Gvv32W4vDKscAstehQweLr7rqKovDJcaaz3/jjTcsrtZlqbHq1auXxbpreLgLuf6u1bk7Ot8K+dN5OLobvHP+vNRwdwA1fvx4iy+88EKLFyxYkMUlll1VdHKQn7STfsN2jZmAG7437aTfckwyZhdyAKhcpKsAAECUKnIkp0WLFt7rc845x+K2bdsWfN+XX35p8fz58zO/LqDWtWvXznutlY011RGOcM2cObPB93zzzTdZXyIWQ5hW1HSFVpAPPfbYYxbrUmNGMkurefPmFp9xxhneMS3poOlGTTU652/YqZtc69LyasZIDgAAiBKdHAAAEKWKTFchH5Wwi3aWO6FnVQU56WTrtNWgAQDlUZGdnCOOOMJ7rTuoqnDH4l//+tcWf/jhh9lfGFCDdDdjfcacc65v374Wa95/4cKFXrtHHnnE4pEjR1pMJ7G8VlttNe/1FltsYbF26ufMmeO1+8tf/mJxOMcDpaNbNGywwQbeMX0edXuNa665xmv3zDPPWBzjvSRdBQAAokQnBwAARKki01XhbqdaRfPyyy+3eOjQoV67adOm5XthQA3Q9JRzznXp0sXi/v37e8e0Kuqnn35q8T333OO1u/766y2eNWuWxaSrSk+Xje+3337eMV2SrKmLiy++2GunS425h6UTVp5u1aqVxZ999pl3rHPnzhbfddddFp977rleu9jLrVRkJwf5SDLBNunk5LSVftP+QsyymnFWFZYBAJWNdBUAAIhSXbH/ndbV1fFf1zKor6/Pa633z+5nVsvK8x7lyHL5exlGcjK/n+V6NsPh8kI/uzDlHIu8ns1S3k+9Z+HGjZrKmDJlisU6TcA552bMmGFxNY9w5nE/+btZHoXuJSM5AAAgSszJqSFpR0PSzuWphKJ+SVAMEADixEgOAACIEnNyKlCl5f2TjnQkeV9Dyj2Sk/aaFuNrkvePRKU9m2gc5uTEgzk5AACgptDJAQAAUSqargIAAKhWjOQAAIAo0ckBAABRopMDAACiRCcHAABEiU4OAACIEp0cAAAQJTo5AAAgSnRyAABAlOjkAACAKNHJAQAAUaKTAwAAokQnBwAARIlODgAAiBKdHAAAECU6OQAAIEp0cgAAQJTo5AAAgCjRyQEAAFFaqtjBurq6+lJdCH5SX19fl8d5G7qf9fX1YZtE50r7vrTnCtsklfZcGb8v8/vJs1kepXw2kT+ezXgUupeM5AAAgCjRyQEAAFGikwMAAKJUdE4Oak/SuS+NmYOTRtqvl+f8m7RfDwBQGozkAACAKNHJAQAAUaKTAwAAosScnBqXZK5LQ/NMspzrErZL0qYhlTCfKMv5PQCAxqGTA6AiaYeRjmJ10nu41FI//bn54YcfvHY//vhjya4JPr1HSy65pMVLLOEnelq1atVguzlz5njtvvvuO4sr4bklXQUAAKJEJwcAAEQpqnSVDqHtsssuFp988sleu7/97W8W69BaMc8884zFX331VdpLBKKlw9vNmjXzjrVp08ZiHfbeYostvHZHH320xV27drV49uzZXrt7773X4iFDhlg8f/78xb1sJBCmLvR106ZNLV577bW9dhdccIHFa665psUTJ0702v31r3+1+IEHHrB40aJFKa8YSu9Xx44dvWMDBw60eOutt7Z4gw028Nrp+zQN9f7773vt/vSnP1n87LPPWhymKEslqk4Oiks7WTjLyclJJuam3SS0IWknGaedQFwJOWgAwH+RrgIAAFGikwMAAKJUV2x4va6urqrG3pdZZhmLs87NDx061OIjjzwy03OH6uvr8yrk8ov3M889ogqdvxLTVWm/XqHLyOwi/nfCMj2b4c9T52N06dLF4hVXXNFr16dPH4v79etnca9evbx2K6ywgsXNmzcveB3ffPONxTqf4+KLL/baLVy4sOA50sjr2ayU37WFlhNr7Jxzbdu2tXi77baz+LjjjvPa6Rwd/ax8//33XrsJEyZYrL9fR44c6bUL39dYedzPSrmXSy+9tMU9e/a0eMCAAV673Xff3eLu3btbHM6r03k9+jkJ57XqfKtLL73U4jvvvNNrl/Xf6EL3kpEcAAAQJSYe15C0E4jTTsLNsuJx0vMnkXZEJu0oFJORAaA8KjJd1aJFC++1LmXTJYWvvvqq1y7PdJVW5AyXkO+0004FrymNcg6J08lZvK+XsJNT1UPimqro3Lmzd2zvvfe2eOedd7ZYl4mHdCmpPrPOOdepUyeL27Vr1+A1OOf/3CdNmtTgNTjn3NixYy3OoqpuDOkq/Vk2adLEO6b3o1u3bhbr8m/nnFtjjTUs7t+/f8F2mnLUrxs+N/p7/aWXXrL40EMP9dpNmTLF4kq9n6W8l5pC6tChg3dM01InnHCCxeGzqc+jpp7CsgHt27e3WP9Gh8+mnkPTkH/+85+9do8++qjF8+bNc41FugoAANQUOjkAACBKFZmu0hnZzvlDbZqGCocy77//fotvuOEGi/fff/+Mr9C35ZZbWhyuBkij0obE89w5PO9rKLUY0lW6kaJzzm2yySYWn3feed6x9dZbz2JdPROuatLnVtNLYepXh9I19dG6deuC16grbvR3gHPOnXTSSRZPnjzZNValPZtFzmex3hfn/JVvm2++uXds0003tXidddaxWFdTOeffD01d6IqekN6z8DnR15q6euKJJ7x2f/jDHyz+6KOPvGNp0lfV9myGKaS11lrL4nBl4UYbbWSx3pcwNTRt2jSLP/zwQ4vDatM9evSweP3117c4XIWl6Su9J59++qnXTlcs6y4Ezjm3YMECt7hIVwEAgJpCJwcAAESJTg4AAIhS2erkhMvEzz33XIuPOeaYgu9r2bKlxVtttZV3THcm1uqKOm/AOT/Xn8W8kWuvvdbicBdeoBroc7DKKqt4x7SicLgzsc6z0DlR4Y7Dmt/XHcXD3cU1hz9jxoyC16tzQvTaN9tsM6/dvvvua3GY99eqybHRORjh7z/9/Rr+DtWfq56j2Hw3PRbOxfr2228t1rka4TwOnTekX1eXpzvn3BVXXGHxySef7B3LulxApdDPt1YVd865yy+/3OK+fft6x3T+jj6Pek+cc+61116z+N1337U4nMul8+V0jl04T0jvn97zrl27eu0OPvhgi0eNGuUde/755y0OKyovLooB1rgkdV6ynECc9hqSyLIQH7uQA0D1I10FAACiVLaRnAsvvNB7ffTRRy/2OTbccEPvtQ6zazVFjZ1z7uabb7b4gAMOsPjKK68seP5wGFytvPLKFu+yyy7esQcffLDg+4BKoUPRZ599tndMl4uGy8uVpgjCIXFNPY0fP97i0aNHe+06duxo8QMPPGCxLmd2zl/qrEtbwyXMWpl3ueWW847pUvYYRuA0NdC7d2+Lw0qzWhIgTEloBWQdzQzTj5qW+uKLLyx+8cUXvXZ6bO7cuRb/3//9n9dON2pdfvnlC16fpt523HFH79gnn3xicfj5q2aa2jvllFO8Y/pzDJ9N/UxrykdTUs7590wrFOu9c8656dOnW/zmm29arBu0Oucva2/Tpo3F4ei4Tj351a9+5R3T9NWXX37pGoORHAAAECU6OQAAIEpMPK4hWQ7JV8Lwfp6TptNuEloJPxcAwH/lvq2D7mqruX4tt+7cz3cyTWOHHXaw+MknnyzYTvOEq622msVhrlJz+CNGjLA4nAukbrrpJu/1YYcdVuSKG5ZX6Xjn3C/ez6SdgrS7kCc5V5a7nqeVcSenIkvH69JP3ZpESzE49/OS/kpz/Trv5u233/bavfDCCxY/99xzFus8Cuf8uR86xyeca6NL2U888USLdf6Qc34J++uuu847plu/hNtLFFLJ2zrofTr//PMtPvDAA712Osej2PYKul1GOC9Ct1TQ8vzh/MdCwh2zdTd7nSe50korFTyHzh9xzt++R+eMFFOp2zros7n11ltbrKVRnHOuXbt2Foe/e3Te1FtvvWXxZZdd5rV7/fXXLZ45c6bFev/Da9K5W+G91L/tWsJB/+4655eVePrpp71jp512msXh9h2FsK0DAACoKXRyAABAlHKfk7P99ttbfOqpp2Z67jFjxnivp0yZkuh9upSx2LDm1KlTLd5vv/0sHjduXMH36FJI5/zl5eEurEA5adVxfTbDHb+LLSXWNIbuYKzpKef8FJW2C6sOF6pUG7Z75ZVXLNZnTp835/zlyAcddJB3TJfOJk1vVDKt5K6lLHTKgHP+/QxTHLr0WtOPYTpB0/KaCgkrHhe6n2H665prrrFYP3+//e1vvXa67Lh79+7eMa3kfMghhzT4dauFlnTQ8irhs6nCn7XevzfeeMPi999/32unKapw5/FC59dU1meffea1u/TSSy3WUg9azsE5/3Ooy86d89OUSdNVhTDxuIaUuopvQ+9LMjk46fuSfL2G5FlROcuqywCAxiFdBQAAopT7SE64KqKxtArjXnvt5R3TYfCsJf0+wkqeWqmVdBUqybrrrmuxDisXW+kYDmd//PHHFmvaYvLkyV67//znPxZr6inpRorhaNiCBQss1uHscJWUVkNu3769d0w3DIwhXaXVgHXVTbiBov4swzSgrrS55557LH7ppZe8droqLkxRJRGmPXWj1ieeeMLi3XbbzWun6ZrwM6H3utppClarAYf3UoU/U52W8fnnn1usm2s61/gNMEOa/po1a5bF4eek2KrNYpXVFxcjOQAAIErMyYEn7/kwaevdlLpOTkPS/mwAAOXBSA4AAIhS7iM5WqExi1UmWsk4zzk4obvuuqtkXwsohT333NNiXZob0nkzmm93zrlHHnnEYq2UHO5grEuGk87DKUYrrnbu3NniFVdc0WunuX2t9OucPyeg2LLqShWOImq5jvB7VTp3I1yee9ZZZ1n83nvvWRzu6h3O/2gs/V50/mOxz2V4n3RX+Wq8n0rnIhX6nDpXeKdx5/x5Tg899JDFOj8nPEcW9Hp1KXxYykDnF4WfL93xvLEYyQEAAFGikwMAAKJUFROP77vvPouPO+64Ml5J/MpRzC7J5N0k18XE4OLC5aerr766xcWWjWt101dffdU7ppVvNZUVpqQa+xkKr12r+2pVXK1wHL6vadOm3jFNoVVjeiP8mWi152L3U1MD4UbGmqIKl5dnKbx2Xf595JFHWqxL4Z0r/jx/8MEHGV1d6Wn61Tl/E+hi5Uv0s/r11197x3TjVK3Sn3WqMVzurZtraiXj5s2bFzxH+HnV3zmNfTYZyQEAAFGikwMAAKJUkemqcAOxvffe22IdxsqCbhoWDt3qUGnSise33nqr9/rhhx9uxNUB+Qk3OPyfcEhYV2lcfvnl3jGtaJrFMLgOTesweLdu3bx255xzjsUbb7xxg+9xzv9ewk0hdWPBLFZ8ldsKK6yQqJ1WhdZVN84V36AxDb2fuuIr/Oxdf/31Fuv9DNM4ej/D6tZacbtaUo7/E35uV1lllQbbhd+XpqhuueUW79hjjz1mcdZ/N/Xv4eDBg71jBx98sMVt2rSxOEw16mdtzpw53jG93kanuhv1bgAAgApVkSM5yEeSiblZ7uSddCJwkp28015T2u856bmyunYAQPYYyQEAAFGqyJGc8H/Hjc0n6hJT55zr06ePxZtvvrnFnTp1SnV+zYuGc3KyXq4HZCWsQPo/4fOn827mzZvnHdORK10WnHREMJyLoEvDDzroIIt33313r13Hjh0tDpeGK10G/eijj3rHpkyZkugaq4XOKyq27FZ/Xrrs3Dl/XoueL+n9DO+F7qB92GGHWbzjjjt67XTuhn4mis1Bue2227xjYXmDahIuoW7RokWD7cK/JzqvTJf/O+eXCkizDDscle7Zs6fFw4YNs1ifWef8+Tp6jnDem5acCOcTTZ06NdE1JsFIDgAAiFJFjuQgH1kW+iv16oW0hf6y/J6ZbwMA1SX3Tk6aYbJwGF2rPxZz9tlnW7zeeutZrBuGOVe88mIa++23n8XPPvtspucGshI+f7qEU4+Fw8r6POqmns45d88991i8YMECi8Ph9g4dOli80korWawbETrnXN++fS3WFEY4nF+owxkugdZhb11S69zPlyBXm/A+6aaoms4Lf1atW7e2WH9nOuf/nHUD5DCtqLTK9D777OMd69evn8X6e7jY/dTPon6mnHPunXfesfhf//qXdyys+FvNCpUsCdNVmo7VaRjO+ek73fAy/DzoZ2WzzTazOFwartWLNS0Znq/QvQwraGupmLFjx3rHwg07G4N0FQAAiBKdHAAAEKXc01U6pLjvvvsmeo9u1uacc6+99lqm19RYI0eO9F7rECpQLXRDwy5dulgcprU01bTNNtt4x3STz86dO1vcsmVLr52mO/R8OlQetiuW6tZh+++++87i0aNHe+0uueQSi19++WXvWNbVfcvtpZdeslh/h4apJq0iHFYe1orWmv4Jf1ZaoVYrw+vnyLnCK21Cej81RRWm/4cMGWJxmOKo5pWsrVq18l4n/WxqJfAtttii4Dk1faxV/sNzaLuw2nTSOYl6H/T7CP9O6uq4SZMmeceyrEDOxOMakmVhvLRfL+370hYRzLIAYpbnAgDkj3QVAACIEp0cAAAQpdzTVffee6/FSefkVApdkqn537322strp8vzgGrxyCOPWKz5/LBqre4eveqqq3rHdO6Hvi9cIqz5fZ2noVWSi9F5N8459/nnn1v89ttvW3zFFVd47bQK7Pz58xN9rWoRpkZvv/12iwcMGGBxWEKj0E7vYVud0xHOkejatavFeq/D+17oesP7qb9DdR7OZZdd5rUbP368xQsXLix4/moT3gedo7LssstaHM6T0XIA4Vyb3r17N/i+8B7p6zRTE8IdCXS+1uuvv27xNddc47XT6tp5lnNgJAcAAESJicfwJO3J51kpOe/Kwml3Ds9y8jMAIH+5d3I++eQTi7XCYbipVyUI004HHnigxU8//XSpLwfIVNjZevLJJy0+7rjjLA6XFetwdvv27b1julxUzx929gqlpcI0iL7WJczjxo3z2mmq7aGHHrJYq/Q69/OKuTHTUhsPPvigxYMGDfLaacX38L4Uuk/F0op6rNj91PSSpp2cc27EiBEW33333RZPnDjRa1eoSne1mzFjhvf68ccft1j/VobV+jUNFaayCkn7H1m9l1qRePbs2V67F154wWItIfPKK6947XSz37T/8UyCdBUAAIgSnRwAABCl3NNVY8aMsXj33Xe3OEz/rLDCCnlfitGZ/Tpktv/++3vtanGzzSyLAWZZRDDLa0gyzJ10KLyaiwHqCo677rrL4uOPP95rp6tsiqU3dDg7XHGhx/T5Czfi06Fv/d0Rbq6pVcc1zRymp/R+VNO9SUM3QDzvvPMs1g00nfM3QS22WXHSn12h6tPO+SvaJk+ebLGmpJzz7++0adMsDiv/xnoPw+9TN77VjTe33XZbr12hjTyd85+5Yj83/R1W6Dl1zk836srjsMr4E088YfHHH39scbiCqtg1ZXmfGckBAABRopMDAACiRCcHAABEqe4XcnW5JUB79erlvdYcZNbLy5977jnvtS451V13K0V9fX1ehWJ+dj/TbtCZZ72bLDfoTHL+xtS2STgnJ/P7mfWzqfM2TjjhBO/YIYccYnGbNm28Y4WqpRbbtXru3LkWh0uJhw8fbrEuidYKx875c3mKLRPPeg5HXs9m1vdT50ppxVzn/Mrz++yzj3esU6dOFutO8uHy5EJLucOl0I8++qjFOsdRq1Q7538mtHRA3nNw8rifWdzLZZZZxmIt6RDeL32t73HOn1+jFZXD31k610aPTZ061Wun82gnTJhgcTgnR+dU6Xy5sLxAqZ5NigHWkDx38m5I2g9xlruJN3SuPCcLUwwQACoH6SoAABClsqWrQpqi6tevn3fsyiuvbPA9WvXUOeeGDh3aYLtRo0Z5r8NhuEpTyiHxWhzJSXLupBKevyKHxIPzWdyhQwfv2AEHHGDxRhtt5B3r1q2bxVoNWUszOOdXO3/zzTctDktJ6LJ2TYnoMuVyqpZ0VXBu77XeJ91o0znnVl99dYu33357i8N01cyZMy3We60pxvC1tgtLB5RrtLMank1NPWoK0Tl/w9ywHICmqNZaay2LtSSEc859+eWXFmuqcMqUKV47rSaupQG0dIFz5SvbUOheMpIDAACiRCcHAABEiYnH8CQdXkyS5qqEHc0rYddzAEB5VMycHPyknHNyGnHu1O/Nckl3Gnl3cqoh758F/TnGuqKsGufkoLBaeTZrAXNyAABATaGTAwAAosScHACZiDVFBaB60cmBJ8v5MGm3RkjbJu01pK2vk/T8AIDyIF0FAACiRCcHAABEiU4OAACIEnNyakie80caM68lybmSaExBwiTXkHbuEPN2AKA8GMkBAABRopMDAACiRCcHAABEiU4OAACIUtENOgEAAKoVIzkAACBKdHIAAECU6OQAAIAo0ckBAABRopMDAACiRCcHAABEiU4OAACIEp0cAAAQJTo5AAAgSnRyAABAlOjkAACAKNHJAQAAUaKTAwAAokQnBwAARIlODgAAiBKdHAAAECU6OQAAIEp0cgAAQJSWKnawrq6uvlQXgp/U19fX5XXq8B/q6vwvVV+f7JYneV/YJun5G3pflpJ+j6Ek30+BNpl/Qzyb5ZHXs8n9LA+ezXgUupeM5AAAgCjRyQEAAFGikwMAAKJEJwcAAESp6MRjxCXvCb2hLCcjp/16DUk72RoAUF0YyQEAAFGikwMAAKJEJwcAAESJOTn4RaUu6pd0Lk9aWc7BKfU8JwBAcozkAACAKNHJAQAAUaKTAwAAokQnBwAARImJxzUkz8nCSd+X5TUk3AH8F8/fmInISa4BAFAejOQAAIAo0ckBAABRIl0FoChNwbVu3do7tvzyy1vcuXNni9u2beu1++qrryyeOnVqg7Fzzn3zzTcWs6dYPvR+tmrVyju23HLLWbzqqqs2+O/OOff5559bPGbMmAb/3Tnnvv/++8ZdLFLLcgpBNWMkBwAARImRHKSS5eTdJP/jSHL+tNfQmMnC7GgOAJWrbJ2cZZdd1nu9//77W7zbbrt5xzbffHOL9Y9IsT8wemzEiBFeu9tvv93i++67bzGuGojfMsss473efvvtLR40aJB3bJ111rFYU1dLL720127evHkWf/HFFxY//PDDXrt//vOfFn/22WcW//DDD4muHT8X3gu9ZzvuuKN3rF+/fhb37t3b4jCtpWmoadOmWXzhhRd67e69916LSUWm06RJE4s7dOjgHdtzzz0t7t+/v3dstdVWs1jTzC1btvTaLVq0yOIvv/zS4scee8xrN2zYMIs1Rblw4UKv3Y8//tjAd1E+pKsAAECU6OQAAIAo1RUbNqyrq8ttTPHRRx/1Xm+33XYWFyuwliZdFZ5vwYIFFm+00UYWf/jhh4muPW/19fV5VZT72f1MOx+m1LuCJyn0V47VBAnn5GR+P7N+Nps1a2ZxmML461//arGuoHLOuSWW+On/Sfq9f/fdd147TTfpe8Kh7TfeeMPiwYMHW/zRRx957co1JJ7Xs5n1/dQUR9++fb1jl1xyicW6gso5P7W15JJLFjy/3mu9FzNmzPDaXXXVVRbfeeedFk+fPt1rV65VWHnczyzupT4jml489thjvXY777yzxeGKRr1/xX436v3T5/Tbb7/12k2cONHiv/3tbxYPHz7cazd//nyLS5mWLHQvGckBAABRopMDAACiRCcHAABEqWxLyDt27Oi91hxkmNd96623LNYl34cffnjB83fr1s3icNldixYtLNa8/1FHHfVLlw1EqX379hbvtdde3rHw+VG6zFvn02j+3jl/OevWW29tcVhB+Ve/+pXFuiT9448/LngN+Dkt0XHiiSd6x3r27GnxUkv5fwK0MvV//vMfi3VpsXPOderUyeKuXbtaHFZG1jkkWg35gQceKPh1WV7u/wx0vpLOJ3XOX5YflgqYO3euxfqczpkzx2un87fWXHNNi8PnXudv6RzaJ5980munc3IqAcUAa0jSXbpL/b6sJj/nPUE6ybnS/qwAANkjXQUAAKJUtpGcIUOGeK91+Gvo0KHeMU1XqX/84x/ea63Qef3111u82WabFbyOSlk2DpSTViQeNWqUd0xTEO+//7537K677rJYU1SafnbOX5YeLmlWOjSvKSoqHi8eHVH8+uuvvWOffvqpxWEa8J577rFYPwfhEu/111/f4vPPP99inSbgnD81QJcqa5Vd/JyO/o4bN87iSy+91Gv3xBNPWBwu+deqxJqiCp/NlVde2WItF6HpYuf8dJiWiAjTX5U2cs1IDgAAiBKdHAAAECUmHqOkSl1hOe0E6aTnYhdyAKhcZevkhLt/p9kNXPO9zvk73q6++uoWh394dI7PlVdeudhfF4iNztu44YYbvGO33XabxeGzpHN5lOb5nXNu4403tlhz+7oE1jl/zkFY+h/J6ZLsiy66yDum82vC+RSzZs2yWOfQFFtOrOUHwu023nnnHYufeeYZi8Odq/nPQWE6/2XSpEnescmTJ1sclgPQeWw6X0fLCzjn3MCBAy3ecMMNC55Pf0fccccdFofPcKUhXQUAAKJEJwcAAESp6ubknHnmmRbvu+++3rFevXpZrMOf4VBouHwdqHWaZpg9e7Z3TOcdhXOQ9LUOg+++++5eOy0R0bRpU4vfe+89r92tt95qMSmM9LTqbPgzTjoHTavSn3DCCd6xQw45xGJNV4XL1XWH6jA1hsUXPhP6WtNazvlLxVdYYQWLw2Xo22yzjcXNmjWzOEwpnn766RZr6jFMUVaaquvkIL20E2fzriQcnj/tROAsqw0nfV+e1wAAaBzSVQAAIEpVMZKzwQYbWHzuuedaXGwUQo+FlZFHjhyZ9SUC0SqW+tUVGCuuuKLF4SafOlyu5+jevbvXbtCgQRbfeOONFodpEEbHiit2z4rRzRrXWmstiw899FCvXbjB8v9oKtI55/bYYw+LdQPXDz74wGsXpkaw+ML7rPfi+OOPt3innXYq2E5TTx999JHX7rnnnrO4miqQM5IDAACiVBUjOSidvOeUVEJBvSznEwEAKhcjOQAAIEp1xf7XXFdXVxGJ7+bNm1v82muvWbzGGmt47QrNyZk5c6bXbsGCBRbrfJ1wR/I0VZizUF9fn9dQw8/uZyVss5DVSE7SUai033MjriHz+1kpz6bm83/zm99YfO2113rt2rVrZ7H+jMKfsy6DnTBhgsV7772312706NEFz5GnvJ7NSrmfOsdq++23t/jOO+/02oXV5v8nXE6sczd0Cfk///lPr53uZF7KCroxPZvh756ePXta/NBDD1ms1apDWg37yy+/9I7p30e9X7qc3LmfL2UvlUL3kpEcAAAQJTo5AAAgSlWRripEqx8759xhhx1mcbdu3SwuVrCtUIrLOed23HFHix9//PHGXexiKOWQeKmL2SU5f9pryHtCcdpriGlIPKTpjTXXXNPiffbZx2u36aabWty1a1eLl19+ea+dVlzViq2aYnbOucMPP9xi3Sww7+qrsaer9PPbunVri9dff32vnW7kqMd0I1bn/PurKa5wybj+Lv/73/9usaZP8hDTsxn+7unRo4fFV199tcVrr722105Tivo8ayVr5/xNPr/99luLr7vuOq/dGWec0WC7vJGuAgAANYVODgAAiFJVp6tCWoVzpZVWsnjAgAFeu4EDB1qsm3qGw30vvviixVtttVVWl/mLSFeRrkrwdSru2dThbE01OecPg2vaYpVVVvHaXXjhhRb369ev4NfSTUQ322wzi8MVklmLPV1VSPi51vurVZI1xeWcvzGrpkyWWWYZr93HH39scf/+/S3+/PPPU15xMjE/m0svvbTFunmupoSd81dD6d/No446ymunlZJbtWplcViNXNNVusoy7yrJpKsAAEBNoeJxDcmyRk2p6+RkWdsmz++ZPZUAoHIwkgMAAKIU1ZycpDQ/eemll1q8//77e+30Z6P5yXBX86yVs+IxIznJ37cY1xBt3r+xwrk7yy23nMVjx461uG3btl47XYJ80EEHWTx8+HCvXdZLymt1Tk5S4TOiczceffRRi/v06eO1mzVrlsVaumPMmDFZX6In5jk5qliV8ULt2rRp4x0bMmSIxQcffLDF4Vwbra6spR7CStalejYZyQEAAFFiTk6NSzuHJMmKqIakXV2V9hqyXJXViNVVic4PAMhWzXdyVl99dYvDP0b6+oMPPijZNQG1Ihyy1gq38+bNszgcOtdKquPGjbOYDmVl0SXlWkE3vE+6eecXX3yR+3XVmqTPhbYLq4zrM6ex3jvnnHvvvfcs1uXpi/Nsahq7sWkt0lUAACBKdHIAAECUajJddfzxx1uss/zD+RQHHnigxSNHjsz/woAao1VZnXNu9913t7hdu3YWh0PW7777rsUTJkywmHRV6RXa1NM55/74xz9arKtaw1TIsGHDLJ45c2bWl4iENE0UVvnXiseahtKdAZxz7qabbmqwXdq5l41Vk52cWpV2CXSey8Ubc11p2jTUrjGThdMuRwcA5I90FQAAiBKdHAAAEKVo01Wa/z399NO9Y4MHD7ZY0wtaddO5n+caATRe8+bNLb7gggu8Y4ceeqjFulN1WC313HPPLXgMpaW/a6+77jrv2Pbbb2+xVsZ96qmnvHbXXHONxVpGAPnTlPumm25q8a233uq10zIOkyZNsvjKK6/02umcqqTLv4uVb2ksRnIAAECUoh3JQTJZTehtTNXgStg/K8m5qXgMANUlqk5O7969LR4xYoTFvXr18trpH6IPP/zQ4jXXXDPHqwMqly4dbdKkiXdMnxetdOpc4Q5cuPFmly5dLL788sst3nXXXQu+Tysen3POOV47LekQbhAIX9jx1p9x0p/dkksu6b1eb731LL7++ustXmeddQqe480337T46KOP9o599dVXia4DvmL/0Sv0bIb3UpeK33HHHRZrhWrn/MrGZ511lsVa4di55J8pvb48V6mSrgIAAFGikwMAAKJU1emq2267zXu92267WawrOMKhr/vuu89irWocu0qYD1OJu5Dn/T1Xg2bNmlm88sore8c01TRmzBjv2Ny5cy3WZ+63v/2t1+6kk06yuEOHDhaHP9Pp06dbfNhhh1n89NNPe+0WLVr0828CRlNSel+cc65Tp04WhyuZNG2k7wvTS/q6VatWFherTL3vvvta/Pnnn3vtmLeWTJgGbtmypcXhvdRnRJ/vo446ymunKxWbNm1q8ddff+21O/XUUy3+97//bbFWNU4rz/vPSA4AAIgSnRwAABAlOjkAACBKFTMnp0WLFhaHlRaVzrsJ85OaD54yZYrFJ554otdO5+QAtUrnw+iztPbaa3vtNIcfzrn44osvLNblwyuuuKLXTncbnz9/vsXPPvus1+6YY46xWOfnJK2civ/SOQ46z8I55/bYYw+Lt956a++Y3qdVV13VYp1H5ZxzSy31058OrTj95JNPeu2OPPJIi7USLnNw0tH745xzffr0sVjnzjnnLwE/+OCDLQ6fby0ZoWUbDjnkEK+d/t2spuexYjo5yF+SibONKeqX9lxJzp1k9/Kk76OoHwDUBtJVAAAgShUzknPaaadZHFZB1f9V6/+mwyEzrXKsQ+zhxpsA/GdJKxm///77XjtNSa211lresfXXX99iHUpfsGCB106XEp933nkWP//88147NtvMht7b8F7o0uDOnTt7x7R8gKYxwiq2ukHjkCFDLL7zzju9dpqaROOFqcftttvO4oEDB3rHunbtarEuIQ9HpCdPnmyxpjLfeOMNr121jmQzkgMAAKJEJwcAAESpYtJVKI+0Q5BJJv1mOYE4y8nCSSZEp31f0nMBAPJXMZ2c5ZZbzuLwj4nm6XXXcM0FO8fScCAtLc3+ySefeMd0noXOAXDOuR49elisu0yHS4nfeusti2fPnm0xHcD8LVy40Hv9+uuvW7zFFlt4x3SbAF3yHS71Hzp0qMXjx4+3mB3hs6d/D3XpvnPO9erVy+Jwmb++T8sxPPjgg167c845x+Jp06Y17mIrEOkqAAAQJTo5AAAgShWTrnrxxRct1iE455x7/PHHLb7gggtKdk1ALQrTG/fff7/FuhTcOb9SsqYtwh2RSUuVT1hqQ1OHN910k3dMlw3fddddFk+dOtVrl8XO00hGn53wOZo4caLFa665pndsxowZFmv14nfeecdrp7uVx6hiOjmoDGkn1zZGVpOf0369JJOhG3pfltcEAMge6SoAABClumL/O62rq2OMuQzq6+tzGQ5o6H4mGdVIIuloTJZ7QqVdCp7mPI05Vx73k2ezPEr5bCJ/PJvxKHQvGckBAABRYk5ODcmywF2e15B0ZCerUaik2JkcAKoLIzkAACBKdHIAAECU6OQAAIAo0ckBAABRKrqEHAAAoFoxkgMAAKJEJwcAAESJTg4AAIgSnRwAABAlOjkAACBKdHIAAECU/h/6Kw1Sjb+wbQAAAABJRU5ErkJggg==\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
}
