{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import sys\n",
    "sys.path.append('../..'); sys.path.append('../'); \n",
    "import numpy as np\n",
    "from setGame import *\n",
    "import tensorflow as tf\n",
    "from tensorflow.keras import layers, Model\n",
    "from sklearn.model_selection import train_test_split\n",
    "import utils\n",
    "from transformer_modules import Encoder, Decoder, AddPositionalEmbedding\n",
    "from abstracters import SymbolicAbstracter, RelationalAbstracter"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "setgame = SetGame(verbose=0)\n",
    "X, y, triples = setgame.generate_grouped_data()\n",
    "inds = (y==True)\n",
    "posi = np.arange(len(y))[inds]\n",
    "negi = np.arange(len(y))[~inds][0:len(posi)]\n",
    "inds = list(posi)+list(negi)\n",
    "X_balanced = X[inds,:]\n",
    "y_balanced = y[inds]\n",
    "triples_balanced = np.array(triples)[inds]\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 216x144 with 3 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "[1, 1, 1, 0, 0, 0, 0, 1, 0, 0, 1, 0]"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "i = np.random.choice(range(len(y_balanced)))\n",
    "setgame.show_triple(triples_balanced[i]), y_balanced[i]\n",
    "attr = setgame.tabulate_attributes_for_triple(triples_balanced[i])\n",
    "iattr = [int(attr[i]) for i in range(len(attr))]\n",
    "print(y_balanced[i])\n",
    "iattr\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "X = np.zeros((len(y_balanced), 12), dtype=int)\n",
    "y = np.zeros(len(y_balanced), dtype=int)\n",
    "\n",
    "for i in np.arange(len(y_balanced)):\n",
    "    attr = setgame.tabulate_attributes_for_triple(triples_balanced[i])\n",
    "    X[i] = [int(attr[i]) for i in range(len(attr))]\n",
    "    y[i] = y_balanced[i]\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "((42, 12), (8, 12), (648, 12))"
      ]
     },
     "execution_count": 27,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "train_size = 50\n",
    "test_size = 0.3\n",
    "val_size = 0.1\n",
    "\n",
    "X_train, X_test, y_train, y_test = train_test_split(X, y, train_size=train_size, test_size=test_size)\n",
    "X_train, X_val, y_train, y_val = train_test_split(X_train, y_train, test_size=val_size/(1-test_size))\n",
    "\n",
    "X_train.shape, X_val.shape, X_test.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "class TransformerClassifier(tf.keras.Model):\n",
    "    def __init__(self, num_layers, num_heads, dff,\n",
    "            input_vocab, embedding_dim, \n",
    "            dropout_rate=0.1, name='transformer_classifier'):\n",
    "\n",
    "        super().__init__(name=name)\n",
    "\n",
    "        if isinstance(input_vocab, int):\n",
    "            self.source_embedder = layers.Embedding(input_vocab, embedding_dim, name='source_embedder')\n",
    "        elif input_vocab == 'vector':\n",
    "            self.source_embedder = layers.TimeDistributed(layers.Dense(embedding_dim), name='source_embedder')\n",
    "        else:\n",
    "            raise ValueError(\n",
    "                \"`input_vocab` must be an integer if the input sequence is token-valued or \"\n",
    "                \"'vector' if the input sequence is vector-valued.\")\n",
    "\n",
    "        self.pos_embedding_adder_input = AddPositionalEmbedding(name='add_pos_embedding_input')\n",
    "        self.encoder = Encoder(num_layers=num_layers, num_heads=num_heads, dff=dff, dropout_rate=dropout_rate, name='encoder')\n",
    "        self.flattener = layers.Flatten()\n",
    "        self.final_layer = layers.Dense(2, name='final_layer')\n",
    "\n",
    "\n",
    "    def call(self, inputs):\n",
    "        source = inputs\n",
    "\n",
    "        x = self.source_embedder(source)\n",
    "        x = self.pos_embedding_adder_input(x)\n",
    "        encoder_context = self.encoder(x)\n",
    "        output = self.flattener(encoder_context)\n",
    "        logits = self.final_layer(output)\n",
    "\n",
    "        try:\n",
    "          # Drop the keras mask, so it doesn't scale the losses/metrics.\n",
    "          # b/250038731\n",
    "          del logits._keras_mask\n",
    "        except AttributeError:\n",
    "          pass\n",
    "\n",
    "        return logits\n",
    "        \n",
    "class AbstractorClassifier(tf.keras.Model):\n",
    "    def __init__(self, num_layers, num_heads, dff,\n",
    "            input_vocab, embedding_dim, dropout_rate=0.1, name='abstractor_classifier'):\n",
    "\n",
    "        super().__init__(name=name)\n",
    "\n",
    "        if isinstance(input_vocab, int):\n",
    "            self.source_embedder = layers.Embedding(input_vocab, embedding_dim, name='source_embedder')\n",
    "        elif input_vocab == 'vector':\n",
    "            self.source_embedder = layers.TimeDistributed(layers.Dense(embedding_dim), name='source_embedder')\n",
    "        else:\n",
    "            raise ValueError(\n",
    "                \"`input_vocab` must be an integer if the input sequence is token-valued or \"\n",
    "                \"'vector' if the input sequence is vector-valued.\")\n",
    "\n",
    "        self.pos_embedding_adder_input = AddPositionalEmbedding(name='add_pos_embedding_input')\n",
    "        self.encoder = Encoder(num_layers=num_layers, num_heads=num_heads, dff=dff, dropout_rate=dropout_rate, name='encoder')\n",
    "        self.abstractor = RelationalAbstracter(num_layers=num_layers, num_heads=num_heads, dff=dff, dropout_rate=dropout_rate, name='abstractor')\n",
    "        self.flattener = layers.Flatten()\n",
    "        self.final_layer = layers.Dense(2, name='final_layer')\n",
    "\n",
    "\n",
    "    def call(self, inputs):\n",
    "        source = inputs\n",
    "\n",
    "        x = self.source_embedder(source)\n",
    "        x = self.pos_embedding_adder_input(x)\n",
    "        encoder_context = self.encoder(x)\n",
    "        abstracted_context = self.abstractor(encoder_context)\n",
    "        outputs = self.flattener(abstracted_context)\n",
    "        logits = self.final_layer(outputs)\n",
    "\n",
    "        try:\n",
    "          # Drop the keras mask, so it doesn't scale the losses/metrics.\n",
    "          # b/250038731\n",
    "          del logits._keras_mask\n",
    "        except AttributeError:\n",
    "          pass\n",
    "\n",
    "        return logits\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [],
   "source": [
    "create_opt = lambda : tf.keras.optimizers.Adam()\n",
    "loss = tf.keras.losses.SparseCategoricalCrossentropy(from_logits=True, name='binary_crossentropy')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model: \"transformer_classifier\"\n",
      "_________________________________________________________________\n",
      " Layer (type)                Output Shape              Param #   \n",
      "=================================================================\n",
      " source_embedder (Embedding)  multiple                 128       \n",
      "                                                                 \n",
      " add_pos_embedding_input (Ad  multiple                 0         \n",
      " dPositionalEmbedding)                                           \n",
      "                                                                 \n",
      " encoder (Encoder)           multiple                  83584     \n",
      "                                                                 \n",
      " flatten_4 (Flatten)         multiple                  0         \n",
      "                                                                 \n",
      " final_layer (Dense)         multiple                  1538      \n",
      "                                                                 \n",
      "=================================================================\n",
      "Total params: 85,250\n",
      "Trainable params: 85,250\n",
      "Non-trainable params: 0\n",
      "_________________________________________________________________\n"
     ]
    }
   ],
   "source": [
    "transformer_model = TransformerClassifier(num_layers=2, num_heads=2, dff=64, input_vocab=2, embedding_dim=64)\n",
    "transformer_model.compile(loss=loss, optimizer=create_opt(), metrics=['accuracy'])\n",
    "transformer_model(X[:32])\n",
    "transformer_model.summary()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 1/50\n",
      "2/2 [==============================] - 4s 493ms/step - loss: 1.1837 - accuracy: 0.4524 - val_loss: 2.5093 - val_accuracy: 0.3750\n",
      "Epoch 2/50\n",
      "2/2 [==============================] - 0s 56ms/step - loss: 1.3544 - accuracy: 0.6429 - val_loss: 1.3728 - val_accuracy: 0.3750\n",
      "Epoch 3/50\n",
      "2/2 [==============================] - 0s 59ms/step - loss: 0.7060 - accuracy: 0.6667 - val_loss: 0.8401 - val_accuracy: 0.6250\n",
      "Epoch 4/50\n",
      "2/2 [==============================] - 0s 61ms/step - loss: 0.9246 - accuracy: 0.4286 - val_loss: 0.8523 - val_accuracy: 0.5000\n",
      "Epoch 5/50\n",
      "2/2 [==============================] - 0s 62ms/step - loss: 0.8865 - accuracy: 0.4286 - val_loss: 1.0125 - val_accuracy: 0.3750\n",
      "Epoch 6/50\n",
      "2/2 [==============================] - 0s 62ms/step - loss: 0.6293 - accuracy: 0.7143 - val_loss: 1.3108 - val_accuracy: 0.3750\n",
      "Epoch 7/50\n",
      "2/2 [==============================] - 0s 57ms/step - loss: 0.5809 - accuracy: 0.7143 - val_loss: 1.1717 - val_accuracy: 0.3750\n",
      "Epoch 8/50\n",
      "2/2 [==============================] - 0s 62ms/step - loss: 0.6599 - accuracy: 0.6667 - val_loss: 0.9394 - val_accuracy: 0.3750\n",
      "Epoch 9/50\n",
      "2/2 [==============================] - 0s 61ms/step - loss: 0.5392 - accuracy: 0.7381 - val_loss: 0.8901 - val_accuracy: 0.3750\n",
      "Epoch 10/50\n",
      "2/2 [==============================] - 0s 63ms/step - loss: 0.6251 - accuracy: 0.6667 - val_loss: 0.9167 - val_accuracy: 0.3750\n",
      "Epoch 11/50\n",
      "2/2 [==============================] - 0s 73ms/step - loss: 0.6181 - accuracy: 0.6190 - val_loss: 0.9987 - val_accuracy: 0.5000\n",
      "Epoch 12/50\n",
      "2/2 [==============================] - 0s 71ms/step - loss: 0.5948 - accuracy: 0.7857 - val_loss: 1.2294 - val_accuracy: 0.3750\n",
      "Epoch 13/50\n",
      "2/2 [==============================] - 0s 74ms/step - loss: 0.6333 - accuracy: 0.7381 - val_loss: 1.3786 - val_accuracy: 0.2500\n",
      "Epoch 14/50\n",
      "2/2 [==============================] - 0s 64ms/step - loss: 0.6828 - accuracy: 0.6667 - val_loss: 1.2760 - val_accuracy: 0.3750\n",
      "Epoch 15/50\n",
      "2/2 [==============================] - 0s 63ms/step - loss: 0.6144 - accuracy: 0.7381 - val_loss: 1.1637 - val_accuracy: 0.3750\n",
      "Epoch 16/50\n",
      "2/2 [==============================] - 0s 65ms/step - loss: 0.5899 - accuracy: 0.7619 - val_loss: 1.1344 - val_accuracy: 0.3750\n",
      "Epoch 17/50\n",
      "2/2 [==============================] - 0s 67ms/step - loss: 0.6005 - accuracy: 0.7857 - val_loss: 1.1301 - val_accuracy: 0.3750\n",
      "Epoch 18/50\n",
      "2/2 [==============================] - 0s 66ms/step - loss: 0.5916 - accuracy: 0.7143 - val_loss: 1.1578 - val_accuracy: 0.5000\n",
      "Epoch 19/50\n",
      "2/2 [==============================] - 0s 65ms/step - loss: 0.5367 - accuracy: 0.8095 - val_loss: 1.2193 - val_accuracy: 0.3750\n",
      "Epoch 20/50\n",
      "2/2 [==============================] - 0s 69ms/step - loss: 0.5215 - accuracy: 0.7143 - val_loss: 1.3402 - val_accuracy: 0.3750\n",
      "Epoch 21/50\n",
      "2/2 [==============================] - 0s 66ms/step - loss: 0.5604 - accuracy: 0.7619 - val_loss: 1.4559 - val_accuracy: 0.3750\n",
      "Epoch 22/50\n",
      "2/2 [==============================] - 0s 71ms/step - loss: 0.6022 - accuracy: 0.7619 - val_loss: 1.5081 - val_accuracy: 0.3750\n",
      "Epoch 23/50\n",
      "2/2 [==============================] - 0s 80ms/step - loss: 0.5663 - accuracy: 0.7619 - val_loss: 1.4087 - val_accuracy: 0.3750\n",
      "Epoch 24/50\n",
      "2/2 [==============================] - 0s 70ms/step - loss: 0.4992 - accuracy: 0.8095 - val_loss: 1.2430 - val_accuracy: 0.5000\n",
      "Epoch 25/50\n",
      "2/2 [==============================] - 0s 81ms/step - loss: 0.5280 - accuracy: 0.7619 - val_loss: 1.2058 - val_accuracy: 0.2500\n",
      "Epoch 26/50\n",
      "2/2 [==============================] - 0s 71ms/step - loss: 0.4703 - accuracy: 0.7381 - val_loss: 1.2499 - val_accuracy: 0.5000\n",
      "Epoch 27/50\n",
      "2/2 [==============================] - 0s 71ms/step - loss: 0.4466 - accuracy: 0.8095 - val_loss: 1.3930 - val_accuracy: 0.3750\n",
      "Epoch 28/50\n",
      "2/2 [==============================] - 0s 60ms/step - loss: 0.4297 - accuracy: 0.8095 - val_loss: 1.4816 - val_accuracy: 0.3750\n",
      "Epoch 29/50\n",
      "2/2 [==============================] - 0s 64ms/step - loss: 0.4023 - accuracy: 0.8333 - val_loss: 1.3784 - val_accuracy: 0.3750\n",
      "Epoch 30/50\n",
      "2/2 [==============================] - 0s 66ms/step - loss: 0.4683 - accuracy: 0.8095 - val_loss: 1.2053 - val_accuracy: 0.5000\n",
      "Epoch 31/50\n",
      "2/2 [==============================] - 0s 75ms/step - loss: 0.4765 - accuracy: 0.7857 - val_loss: 1.1569 - val_accuracy: 0.3750\n",
      "Epoch 32/50\n",
      "2/2 [==============================] - 0s 76ms/step - loss: 0.5387 - accuracy: 0.7857 - val_loss: 1.1881 - val_accuracy: 0.5000\n",
      "Epoch 33/50\n",
      "2/2 [==============================] - 0s 72ms/step - loss: 0.3873 - accuracy: 0.7857 - val_loss: 1.2387 - val_accuracy: 0.3750\n",
      "Epoch 34/50\n",
      "2/2 [==============================] - 0s 60ms/step - loss: 0.4111 - accuracy: 0.8333 - val_loss: 1.3037 - val_accuracy: 0.3750\n",
      "Epoch 35/50\n",
      "2/2 [==============================] - 0s 68ms/step - loss: 0.4207 - accuracy: 0.8333 - val_loss: 1.3284 - val_accuracy: 0.3750\n",
      "Epoch 36/50\n",
      "2/2 [==============================] - 0s 52ms/step - loss: 0.4131 - accuracy: 0.8333 - val_loss: 1.2623 - val_accuracy: 0.3750\n",
      "Epoch 37/50\n",
      "2/2 [==============================] - 0s 57ms/step - loss: 0.5246 - accuracy: 0.7619 - val_loss: 1.1861 - val_accuracy: 0.5000\n",
      "Epoch 38/50\n",
      "2/2 [==============================] - 0s 62ms/step - loss: 0.4143 - accuracy: 0.7857 - val_loss: 1.1515 - val_accuracy: 0.3750\n",
      "Epoch 39/50\n",
      "2/2 [==============================] - 0s 61ms/step - loss: 0.4367 - accuracy: 0.8333 - val_loss: 1.2162 - val_accuracy: 0.5000\n",
      "Epoch 40/50\n",
      "2/2 [==============================] - 0s 64ms/step - loss: 0.3556 - accuracy: 0.8810 - val_loss: 1.3026 - val_accuracy: 0.3750\n",
      "Epoch 41/50\n",
      "2/2 [==============================] - 0s 66ms/step - loss: 0.4170 - accuracy: 0.8095 - val_loss: 1.2612 - val_accuracy: 0.3750\n",
      "Epoch 42/50\n",
      "2/2 [==============================] - 0s 59ms/step - loss: 0.3940 - accuracy: 0.7857 - val_loss: 1.1520 - val_accuracy: 0.5000\n",
      "Epoch 43/50\n",
      "2/2 [==============================] - 0s 63ms/step - loss: 0.4318 - accuracy: 0.8333 - val_loss: 1.0866 - val_accuracy: 0.5000\n",
      "Epoch 44/50\n",
      "2/2 [==============================] - 0s 135ms/step - loss: 0.3862 - accuracy: 0.9286 - val_loss: 1.0653 - val_accuracy: 0.5000\n",
      "Epoch 45/50\n",
      "2/2 [==============================] - 0s 56ms/step - loss: 0.4158 - accuracy: 0.8095 - val_loss: 1.1078 - val_accuracy: 0.5000\n",
      "Epoch 46/50\n",
      "2/2 [==============================] - 0s 81ms/step - loss: 0.3802 - accuracy: 0.9048 - val_loss: 1.1663 - val_accuracy: 0.5000\n",
      "Epoch 47/50\n",
      "2/2 [==============================] - 0s 67ms/step - loss: 0.4188 - accuracy: 0.8571 - val_loss: 1.1931 - val_accuracy: 0.5000\n",
      "Epoch 48/50\n",
      "2/2 [==============================] - 0s 76ms/step - loss: 0.3793 - accuracy: 0.8571 - val_loss: 1.0514 - val_accuracy: 0.5000\n",
      "Epoch 49/50\n",
      "2/2 [==============================] - 0s 91ms/step - loss: 0.3830 - accuracy: 0.8333 - val_loss: 0.8942 - val_accuracy: 0.6250\n",
      "Epoch 50/50\n",
      "2/2 [==============================] - 0s 64ms/step - loss: 0.4019 - accuracy: 0.8810 - val_loss: 0.8736 - val_accuracy: 0.5000\n"
     ]
    }
   ],
   "source": [
    "history = transformer_model.fit(X_train, y_train, validation_data=(X_val, y_val), epochs=50, verbose=1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "utils.plot_history(history, ('loss', 'accuracy'));"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "test accuracy: 58.33%\n",
      "\n"
     ]
    }
   ],
   "source": [
    "out = transformer_model.predict(X_test, verbose=0)\n",
    "yhat = np.argmax(out, axis=1)\n",
    "print('test accuracy: %.2f%%\\n' % (100*np.mean(yhat==y_test)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model: \"abstractor_classifier\"\n",
      "_________________________________________________________________\n",
      " Layer (type)                Output Shape              Param #   \n",
      "=================================================================\n",
      " source_embedder (Embedding)  multiple                 128       \n",
      "                                                                 \n",
      " add_pos_embedding_input (Ad  multiple                 0         \n",
      " dPositionalEmbedding)                                           \n",
      "                                                                 \n",
      " encoder (Encoder)           multiple                  83584     \n",
      "                                                                 \n",
      " abstractor (RelationalAbstr  multiple                 151040    \n",
      " acter)                                                          \n",
      "                                                                 \n",
      " flatten_5 (Flatten)         multiple                  0         \n",
      "                                                                 \n",
      " final_layer (Dense)         multiple                  1538      \n",
      "                                                                 \n",
      "=================================================================\n",
      "Total params: 236,290\n",
      "Trainable params: 236,290\n",
      "Non-trainable params: 0\n",
      "_________________________________________________________________\n"
     ]
    }
   ],
   "source": [
    "abstractor_model = AbstractorClassifier(num_layers=2, num_heads=2, dff=64, input_vocab=2, embedding_dim=64)\n",
    "abstractor_model.compile(loss=loss, optimizer=create_opt(), metrics=['accuracy'])\n",
    "abstractor_model(X[:32])\n",
    "abstractor_model.summary()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 1/50\n",
      "3/3 [==============================] - 10s 490ms/step - loss: 1.3898 - accuracy: 0.5176 - val_loss: 0.7040 - val_accuracy: 0.4667\n",
      "Epoch 2/50\n",
      "3/3 [==============================] - 0s 67ms/step - loss: 0.9276 - accuracy: 0.4824 - val_loss: 0.7235 - val_accuracy: 0.4667\n",
      "Epoch 3/50\n",
      "3/3 [==============================] - 0s 59ms/step - loss: 0.9652 - accuracy: 0.4471 - val_loss: 0.8572 - val_accuracy: 0.5333\n",
      "Epoch 4/50\n",
      "3/3 [==============================] - 0s 64ms/step - loss: 0.7825 - accuracy: 0.5294 - val_loss: 0.7872 - val_accuracy: 0.4667\n",
      "Epoch 5/50\n",
      "3/3 [==============================] - 0s 62ms/step - loss: 0.8046 - accuracy: 0.5059 - val_loss: 0.6932 - val_accuracy: 0.5333\n",
      "Epoch 6/50\n",
      "3/3 [==============================] - 0s 60ms/step - loss: 0.7977 - accuracy: 0.4353 - val_loss: 0.7104 - val_accuracy: 0.5333\n",
      "Epoch 7/50\n",
      "3/3 [==============================] - 0s 68ms/step - loss: 0.7768 - accuracy: 0.4706 - val_loss: 0.6872 - val_accuracy: 0.5333\n",
      "Epoch 8/50\n",
      "3/3 [==============================] - 0s 61ms/step - loss: 0.8482 - accuracy: 0.3882 - val_loss: 0.7366 - val_accuracy: 0.4667\n",
      "Epoch 9/50\n",
      "3/3 [==============================] - 0s 62ms/step - loss: 0.7652 - accuracy: 0.5059 - val_loss: 0.6788 - val_accuracy: 0.5333\n",
      "Epoch 10/50\n",
      "3/3 [==============================] - 0s 62ms/step - loss: 0.7541 - accuracy: 0.4353 - val_loss: 0.6529 - val_accuracy: 0.6667\n",
      "Epoch 11/50\n",
      "3/3 [==============================] - 0s 61ms/step - loss: 0.6436 - accuracy: 0.6588 - val_loss: 0.6336 - val_accuracy: 0.5333\n",
      "Epoch 12/50\n",
      "3/3 [==============================] - 0s 58ms/step - loss: 0.6930 - accuracy: 0.6471 - val_loss: 0.5989 - val_accuracy: 0.6000\n",
      "Epoch 13/50\n",
      "3/3 [==============================] - 0s 64ms/step - loss: 0.6749 - accuracy: 0.5765 - val_loss: 0.5489 - val_accuracy: 0.7333\n",
      "Epoch 14/50\n",
      "3/3 [==============================] - 0s 74ms/step - loss: 0.5481 - accuracy: 0.7529 - val_loss: 0.4575 - val_accuracy: 1.0000\n",
      "Epoch 15/50\n",
      "3/3 [==============================] - 0s 65ms/step - loss: 0.4568 - accuracy: 0.8706 - val_loss: 0.3726 - val_accuracy: 0.9333\n",
      "Epoch 16/50\n",
      "3/3 [==============================] - 0s 79ms/step - loss: 0.3652 - accuracy: 0.8706 - val_loss: 0.2430 - val_accuracy: 1.0000\n",
      "Epoch 17/50\n",
      "3/3 [==============================] - 0s 66ms/step - loss: 0.2351 - accuracy: 0.9529 - val_loss: 0.1399 - val_accuracy: 1.0000\n",
      "Epoch 18/50\n",
      "3/3 [==============================] - 0s 63ms/step - loss: 0.1407 - accuracy: 1.0000 - val_loss: 0.0870 - val_accuracy: 1.0000\n",
      "Epoch 19/50\n",
      "3/3 [==============================] - 0s 59ms/step - loss: 0.1202 - accuracy: 0.9765 - val_loss: 0.0575 - val_accuracy: 1.0000\n",
      "Epoch 20/50\n",
      "3/3 [==============================] - 0s 57ms/step - loss: 0.0750 - accuracy: 0.9882 - val_loss: 0.0397 - val_accuracy: 1.0000\n",
      "Epoch 21/50\n",
      "3/3 [==============================] - 0s 62ms/step - loss: 0.0643 - accuracy: 1.0000 - val_loss: 0.0288 - val_accuracy: 1.0000\n",
      "Epoch 22/50\n",
      "3/3 [==============================] - 0s 66ms/step - loss: 0.0533 - accuracy: 0.9882 - val_loss: 0.0272 - val_accuracy: 1.0000\n",
      "Epoch 23/50\n",
      "3/3 [==============================] - 0s 61ms/step - loss: 0.0338 - accuracy: 1.0000 - val_loss: 0.0214 - val_accuracy: 1.0000\n",
      "Epoch 24/50\n",
      "3/3 [==============================] - 0s 62ms/step - loss: 0.0295 - accuracy: 0.9882 - val_loss: 0.0146 - val_accuracy: 1.0000\n",
      "Epoch 25/50\n",
      "3/3 [==============================] - 0s 65ms/step - loss: 0.0194 - accuracy: 1.0000 - val_loss: 0.0099 - val_accuracy: 1.0000\n",
      "Epoch 26/50\n",
      "3/3 [==============================] - 0s 60ms/step - loss: 0.0172 - accuracy: 1.0000 - val_loss: 0.0081 - val_accuracy: 1.0000\n",
      "Epoch 27/50\n",
      "3/3 [==============================] - 0s 61ms/step - loss: 0.0100 - accuracy: 1.0000 - val_loss: 0.0079 - val_accuracy: 1.0000\n",
      "Epoch 28/50\n",
      "3/3 [==============================] - 0s 61ms/step - loss: 0.0176 - accuracy: 1.0000 - val_loss: 0.0060 - val_accuracy: 1.0000\n",
      "Epoch 29/50\n",
      "3/3 [==============================] - 0s 62ms/step - loss: 0.0151 - accuracy: 1.0000 - val_loss: 0.0068 - val_accuracy: 1.0000\n",
      "Epoch 30/50\n",
      "3/3 [==============================] - 0s 59ms/step - loss: 0.0172 - accuracy: 1.0000 - val_loss: 0.0050 - val_accuracy: 1.0000\n",
      "Epoch 31/50\n",
      "3/3 [==============================] - 0s 68ms/step - loss: 0.0095 - accuracy: 1.0000 - val_loss: 0.0034 - val_accuracy: 1.0000\n",
      "Epoch 32/50\n",
      "3/3 [==============================] - 0s 68ms/step - loss: 0.0315 - accuracy: 0.9882 - val_loss: 0.0030 - val_accuracy: 1.0000\n",
      "Epoch 33/50\n",
      "3/3 [==============================] - 0s 73ms/step - loss: 0.0111 - accuracy: 1.0000 - val_loss: 0.0048 - val_accuracy: 1.0000\n",
      "Epoch 34/50\n",
      "3/3 [==============================] - 0s 77ms/step - loss: 0.0138 - accuracy: 1.0000 - val_loss: 0.0029 - val_accuracy: 1.0000\n",
      "Epoch 35/50\n",
      "3/3 [==============================] - 0s 75ms/step - loss: 0.0597 - accuracy: 0.9882 - val_loss: 0.1441 - val_accuracy: 0.8000\n",
      "Epoch 36/50\n",
      "3/3 [==============================] - 0s 67ms/step - loss: 0.0298 - accuracy: 0.9882 - val_loss: 0.0026 - val_accuracy: 1.0000\n",
      "Epoch 37/50\n",
      "3/3 [==============================] - 0s 72ms/step - loss: 0.0168 - accuracy: 1.0000 - val_loss: 0.0029 - val_accuracy: 1.0000\n",
      "Epoch 38/50\n",
      "3/3 [==============================] - 0s 61ms/step - loss: 0.0067 - accuracy: 1.0000 - val_loss: 0.0027 - val_accuracy: 1.0000\n",
      "Epoch 39/50\n",
      "3/3 [==============================] - 0s 64ms/step - loss: 0.0096 - accuracy: 1.0000 - val_loss: 0.0032 - val_accuracy: 1.0000\n",
      "Epoch 40/50\n",
      "3/3 [==============================] - 0s 61ms/step - loss: 0.0204 - accuracy: 0.9882 - val_loss: 0.0028 - val_accuracy: 1.0000\n",
      "Epoch 41/50\n",
      "3/3 [==============================] - 0s 65ms/step - loss: 0.0072 - accuracy: 1.0000 - val_loss: 0.0026 - val_accuracy: 1.0000\n",
      "Epoch 42/50\n",
      "3/3 [==============================] - 0s 68ms/step - loss: 0.0071 - accuracy: 1.0000 - val_loss: 0.0027 - val_accuracy: 1.0000\n",
      "Epoch 43/50\n",
      "3/3 [==============================] - 0s 74ms/step - loss: 0.0088 - accuracy: 1.0000 - val_loss: 0.0030 - val_accuracy: 1.0000\n",
      "Epoch 44/50\n",
      "3/3 [==============================] - 0s 66ms/step - loss: 0.0092 - accuracy: 1.0000 - val_loss: 0.0024 - val_accuracy: 1.0000\n",
      "Epoch 45/50\n",
      "3/3 [==============================] - 0s 68ms/step - loss: 0.0047 - accuracy: 1.0000 - val_loss: 0.0023 - val_accuracy: 1.0000\n",
      "Epoch 46/50\n",
      "3/3 [==============================] - 0s 62ms/step - loss: 0.0060 - accuracy: 1.0000 - val_loss: 0.0022 - val_accuracy: 1.0000\n",
      "Epoch 47/50\n",
      "3/3 [==============================] - 0s 69ms/step - loss: 0.0033 - accuracy: 1.0000 - val_loss: 0.0022 - val_accuracy: 1.0000\n",
      "Epoch 48/50\n",
      "3/3 [==============================] - 0s 66ms/step - loss: 0.0043 - accuracy: 1.0000 - val_loss: 0.0020 - val_accuracy: 1.0000\n",
      "Epoch 49/50\n",
      "3/3 [==============================] - 0s 77ms/step - loss: 0.0031 - accuracy: 1.0000 - val_loss: 0.0017 - val_accuracy: 1.0000\n",
      "Epoch 50/50\n",
      "3/3 [==============================] - 0s 76ms/step - loss: 0.0035 - accuracy: 1.0000 - val_loss: 0.0015 - val_accuracy: 1.0000\n"
     ]
    }
   ],
   "source": [
    "history = abstractor_model.fit(X_train, y_train, validation_data=(X_val, y_val), epochs=50, verbose=1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAYIAAAEGCAYAAABo25JHAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjMuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8vihELAAAACXBIWXMAAAsTAAALEwEAmpwYAABNI0lEQVR4nO2deXxU5dX4v2e2TPadJCRAAEFQVtnUKi7Upe67WJdKtb6utfZtq7a12qpvrW9tq9Wq1Cr61rUi/VG1WhUVN5RFVkH2JUAgZE8mk9me3x/3zmQCCUTIZGbuPN/PZz4zc+8z9547c+eee5bnHFFKodFoNJrUxRZvATQajUYTX7Qi0Gg0mhRHKwKNRqNJcbQi0Gg0mhRHKwKNRqNJcRzxFuCbUlRUpCorK+MthsaiLF68eI9Sqjge+9bntiaW7O/cTjpFUFlZyaJFi+IthsaiiMiWeO1bn9uaWLK/c1u7hjQajSbF0YpAo9FoUhytCDQajSbFSboYgWb/+P1+qqqq8Hq98RYloXG73VRUVOB0OuMtStKjz7nE4mDO7ZgpAhF5GjgL2K2UGrWfcZOABcClSqlXYyVPqlBVVUV2djaVlZWISLzFSUiUUtTW1lJVVcXgwYPjLU7So8+5xOFgz+1YuoZmAafvb4CI2IHfAW/HUI6Uwuv1UlhYqP+Q+0FEKCwsPOg7WBF5WkR2i8jKbtaLiDwiIutFZLmIHHVIAic4+pxLHA723I6ZIlBKzQfqDjDsFmA2sDtWcqQi+g95YA7xO5rF/m9yvgMMMx/XAY8fys6SAX3OJQ4H81vELUYgIuXA+cDJwKQDjL0O4w/FwIEDuxzz1sqdVNW3ce3xQ3pZUo2mM0qp+SJSuZ8h5wLPKaPG+wIRyRORMqXUzr6RMMb4vbDiHzDucrBF3Uv6POBt3Ge41x/E6bBh7+YCFQiFCIQUboe90/L2QBC7TXBE7SOEos0X5FCr59sE0l12hA6Z/KEQ7f5Q5L0jMx93RlbHh5SC1hoIBQkpRZu/azlcDhsue4fMCoXXHyIY6oOS/zYHWYVl3/hj8QwW/wm4XSkVPJAGU0rNBGYCTJw4sctv852vdrNgY61WBAlAVlYWLS0t8RYjnpQD26LeV5nL9lEEPbnJSTg2vAdzb4ai4TBwSsfy5p3Q3tRpqALSFNDNX1wBdsCuQEnHMAW4zM9Ff1yAjN66nvo6y+EAHOa2RaCxoQ2b6zBcDvOi7vdA0/YDy+HrfCwA7j5q++ITF5BcimAi8JKpBIqAM0QkoJT658FsLMNlx+ML9KJ4Gs1B09Vlr8tLQU9uchIOn8d4rt/cWREEfeDOhYKOm7EWr59Ne1qxIYwsy8Fu6/hqlFJsrfPQ1BbA5bARCIY4rCQLmwjrdrVgtwu+YIisNAeVhRnUe/xU1XsoyXFTkuM+pEPY0dDGnpZ2KgszyXI72FjTSrs/yGH9skhz2gntXoPNH2JbnYchxZmGu0WFCAQCNGUNocpjpyI/g4JMV6ftBoIh1u1uwSbCYf2y8PqDbKxpJSfdwcCCjJi70NIO8nNxm0eglBqslKpUSlUCrwI3HqwSgLAiCPaWeJpeQCnFT3/6U0aNGsXo0aN5+eWXAdi5cydTp05l3LhxjBo1io8++ohgMMjVV18dGfvHP/4xztIfElXAgKj3FcCOOMnS+wTajOf6zR3LlIJAO9g7X4q8fuM/GVKKxjZ/p3V1rT4a2/yU5qYxuCgDBLbVedha5yGkFIMKM+if66bZ62dHQxs7GtrITHPQL7vry915553HhAkTOPLII5k5cyYAb731FkcddRRjx45l2rRpALS0tPDz227k4lO+xcQJ43jq2Rfx+AJMObycNKfhnnrt9Xe46cd30uoLMP3yq/jxj3/MSaeewe33P8L7n3zB9y84nWnHTeHYY4/l66+/BiAYDHLH7T/jolOO5eyTjua+Bx/ilf/3Jrf94ArK89MREd555x0uuOCCQ/r6Y0Es00dfBE4EikSkCrgbcAIopZ7o7f2lu+y0Bww/XPRdRyrz63+t4qsdTQce+A04on8Od599ZI/GvvbaayxdupRly5axZ88eJk2axNSpU3nhhRc47bTT+MUvfkEwGMTj8bB06VK2b9/OypVGIk5DQ0Ovyt3HzAVuFpGXgClAo2XiA2Bc8AEaokrXhAKA4tfv7eSrmq2RxeH/pGC4W9zmhTbsY7eLRJYFQyqiONKcdhzm/3hAQTpXHVOJw2bb7131008/TUFBAW1tbUyaNIlzzz2XH/zgB8yfP5/BgwdTV2fkrtx7773k5eayfMVy1u9qob6+fp87e0SwCRRkuPD6gyxduZoXXnqJMrWb5a15fPLxR6S5nLz77rv8/Oc/Z/bs2cycOZNNmzaxbOlSaj0B1m7ZSW5+Pg/c9TPqa2spLi7mmWeeYcaMGYf4A/Q+MVMESqnLvsHYqw91fxku42Rq8wfJStPz5BKBjz/+mMsuuwy73U5JSQknnHACCxcuZNKkSXz/+9/H7/dz3nnnMW7cOIYMGcLGjRu55ZZbOPPMMzn11FPjLX639OAm503gDGA94AES759/KPi7sAjCykHsQMedf0gpbAJ2m+ALhFBKISK0+0MIErkDB2NM2B/viLqZy0pzkJfupCArDae9eyfGI488wpw5cwDYtm0bM2fOZOrUqZF8+oKCAgDeffddXnrpJdIcdgYUZJCT7qR/bnrnjYmAUvTPS8flsHHaWediE0BBOgGmX3oJ69atQ0Tw+/2R7V5//fU4HA76ZdsJVJSQ4XJw1VVX8ve//50ZM2bw2Wef8dxzz/X0m+4zLHPFTHcZh+LxBbQiMOnpnXusUN2kdkydOpX58+fzxhtvcOWVV/LTn/6Uq666imXLlvH222/z2GOP8corr/D000/3scQ940A3OWa20E19JE7fE77o10dZBEEj8nr3WSPBafjvQyHFqh1NFGe7KMh0saa6mZIcN+2BEI0eH4OLs3rtv/rBBx/w7rvv8tlnn5GRkcGJJ57I2LFjI26baMLKCCAn3UlOujEDN9rS8Lb7AYXNJmSmORjav4iSLCc0wf333ctJJ53EnDlz2Lx5MyeeeOI+2xURyvMzAJgxYwZnn302brebiy++GIcj8a5Plqk1lGHeWbTpOEHCMHXqVF5++WWCwSA1NTXMnz+fyZMns2XLFvr168cPfvADrrnmGpYsWcKePXsIhUJceOGF3HvvvSxZsiTe4mu6I2BOVmra3qEUguazvcPF0h4IolC4nXZcDjuZaQ5qmttp8Pjol+Pu1Ru2xsZG8vPzycjIYM2aNSxYsID29nY+/PBDNm3aBBBxDZ166qk8+uijkc/W19cDUFJSwurVqwmFQsx54z/77kSFzH01UV5eDsCsWbMiq0899VSeeOIJAoFAp/3179+f/v37c99993H11Vf32jH3JtZRBKZrSAeME4fzzz+fMWPGMHbsWE4++WQefPBBSktL+eCDDxg3bhzjx49n9uzZ3HrrrWzfvp0TTzyRcePGcfXVV/Pb3/423uJruiOsCFDQWGUu84HN2WleQZuZk59u3qTlZ7gIKUXWfgK+B8vpp59OIBBgzJgx3HXXXRx99NEUFxczc+ZMLrjgAsaOHcull14KwC9/+Uvq6+sZNWoUY8eO5f333wfggQce4KyzzuLkk0+mrLSEfSYJmIrgZz/7GXfeeSff+ta3CAY7rjfXXnstAwcOjJzzL7zwQmTd5ZdfzoABAzjiiCN69bh7C+nOfE9UJk6cqLpq3vHB17u5+pmFzL7hWCYMyo+DZInB6tWrGTlyZLzFSAq6+q5EZLFSamI85Onu3E443vhvWPiU8fqK2az2lzOy2AEoY26ByY6GNupafRzZPwcRIaQUtS0+8jKc+/X1JwTN1ca8iLKxIKasjVXgqTWWfUNuvvlmxo8fzzXXXNPLgnbNNz23E/zX6DkZZoxAu4Y0mhjj9xp3/9ARJ+gidbTNH8TttEf85jYRirP3H/BNGMIXf9Ux0xgV6lj+DZgwYQLLly/niiuu6CXhep/Ei1ocJB2uIT2pTKOJKQEv5FZA0w4jcyjzGAj5O8UHlDJSQfPSk7TMd1eKIHRwimDx4sW9JFTsSALV3DPCuchtfm0RaDQxJeAFZwbkDTTmEoTMmy9HhyLwB435A26nvZuNJDjhC35ob4sgSY/nAFjQItCKQKOJKQEvONIgp79hEYTM/1yUaygcKE56RdDJNRQ8KIsgGbDMUWlFoNH0EYF2cLghf5ARIwiZE8iiLILwDGFrKYKDcw0lA5Y5qvTwzGIdI9BoYou/zZg0ll8J3gYI+gHpCCBjKII0hz15y73YTAW2tyKwWeaS2QnLHJXLbsNuE20RaDSxJmwR5A0y33uNQHH0zFx/CLcziS8v2iJITkSEDKeuQJpsZGVldbtu8+bNjBrVbbtrTbwIeE3XUKXxPhQwYgYmSin8wVDCponu75yLEFEEUdcTrQiSg3SXXc8j0GhiTUQRDOpYFpU6GgwpQkolrCLoEd1aBL0X8wiXokgELJM1BGZPAp0+2sG/74DqFb27zdLR8J0Hul19++23M2jQIG688UYA7rnnHkSE+fPnU19fj9/v57777uPcc8/9Rrv1er3ccMMNLFq0CIfDwR/+8AdOOukkVq1axYwZM/D5fIRCIWbPnk3//v255JJLqKqqIhgMctddd0XKC2h6gXDWkDsX0s1Z/OFA8b/vwLZzOUN8QcM11Bs+9T4851paWjj33HONz7U1c9/dv+Tc6d8DpXjulbn8/q8vInYnY8aM4f/+7//YtWsX119/PRs3bgTg8ccfp3///px11lmRkuq///3vaWlp4Z577uHEE0/k2GOP5ZNPPuGcc85h+PDh3Hffffh8PgoLC3n++ecpKSmhpaWFW265hUWLFiEi3H333TQ0NLBy5cpIr46//vWvrF69mj/84Q+H+g1bSxGkuxw6WBxnpk+fzo9+9KPIn/KVV17hrbfe4rbbbiMnJ4c9e/Zw9NFHc84553yjbk2PPfYYACtWrGDNmjWceuqprF27lieeeIJbb72Vyy+/HJ/PRzAY5M0336R///688cYbgFGQTNOL+L3gNMs2h+MEUamjIbNsTV81tO/Nc87tdjNnzhxysrPZs+p9jj7nGs659Cq+WrmC+x/5G5+89yZFlUdECsr98Ic/5IQTTmDOnDkEg0FaWloiRey6o6GhgQ8//BAwCt4tWLAAEeGpp57iwQcf5KGHHuLee+8lNzeXFStWRMa5XC7GjBnDgw8+iNPp5JlnnuHJJ5881K8PsJgi0F3K9mI/d1GxYvz48ezevZsdO3ZQU1NDfn4+ZWVl3HbbbcyfPx+bzcb27dvZtWsXpaWlPd7uxx9/zC233ALAiBEjGDRoEGvXruWYY47h/vvvp6qqigsuuIBhw4YxevRofvKTn3D77bdz1llncfzxx8fqcFOTsEUAHXGCsGvoOw/Q1Oqjqt7DiNJs7I7Yp4/25jmnlOLnP/+58blgO9t37mTXrl3MmzePi86cRlFRMdDR22DevHmR/gJ2u53c3NwDKoJo67SqqopLL72UnTt34vP5Ir0Twj0TwuTnG5bXySefzOuvv87IkSPx+/2MHj36G35bXZPETrx90YogMbjooot49dVXefnll5k+fTrPP/88NTU1LF68mKVLl1JSUoLX6z3whqLorjjid7/7XebOnUt6ejqnnXYa8+bNY/jw4SxevJjRo0dz55138pvf/KY3DksDxuSxkN+IEUCHIthrVjGAow9jBL11znX63HuvUlJcjNfrRYVChjXRA1eXw+EgFDUjee/9ZmZmRl7fcsst3HzzzaxYsYInn3wyMja6t0E01157LbNmzer1TmeWUgTpTh0sTgSmT5/OSy+9xKuvvspFF11EY2Mj/fr1w+l08v7777Nly5YDb2Qvpk6dyvPPPw/A2rVr2bp1K4cffjgbN25kyJAh/PCHP+Scc85h+fLl7Nixg4yMDK644gp+8pOf6N4GvUm4BHVYEUy+DjKKwNbhXPAHQzhsNmx95BqC3jvnOn3uk0Vs2WaU2Z520gm88q93qK1rADp6DUybNo3HH38cMHoWNzU1UVJSwu7du6mtraW9vZ3XX399v/sL9zZ49tlnI8u765kwZcoUtm3bxgsvvMBll/W4CeQBsZQiMILFOkYQb4488kiam5spLy+nrKyMyy+/nEWLFjFx4kSef/55RowY8Y23eeONNxIMBhk9ejSXXnops2bNIi0tjZdffplRo0Yxbtw41qxZw1VXXcWKFSuYPHky48aN4/777+eXv/xlDI4yRQk3ogkrgtxycGV0HhJUOO19O5Gst865Tp977Q1GDBtqbP+IEfzih9dwwmlnM3bsWH784x8D8PDDD/P+++8zevRoJkyYwKpVq3A6nfzqV79iypQpnHXWWfvd9z333MPFF1/M8ccfT1FRUWR5dz0TAC655BK+9a1vRdxFvYJSKiYP4GlgN7Cym/WXA8vNx6fA2J5sd8KECao77pi9XE249z/drk8Fvvrqq3iLkDR09V0Bi1SM/hMHeuzv3E4YGrcrdXeOUoueiSza+3tcW92kNtW09LFgMWD3GqX2rDNee5uU2r7EeI4zZ555pnr33Xf3O+abntuxtAhmAafvZ/0m4ASl1BjgXmDmoe5Qxwg0mhgTblwftgi6GhJUOPrYIogJYuuYRxD2+cdxQllDQwPDhw8nPT2dadOm9eq2Y5Y1pJSaLyKV+1n/adTbBUDFoe4zw2WnzR/sNtCiSUxWrFjBlVde2WlZWloan3/+eZwk0nRLxDXUdavJkFIEQok7qzhMj845sZl1lOiYYRzHMtR5eXmsXbs2JttOlPTRa4B/d7dSRK4DrgMYOHBgtxtJd9lRyqhzEi5Cl4okmyIcPXo0S5cu7dN9qiRr0ZowBMIWQXqnxeFzLmBmDPV1jOCb0qNzLtoiUPG3CHrKwZzbcT8qETkJQxHc3t0YpdRMpdREpdTE4uLibreV4dRdytxuN7W1tfpCtx+UUtTW1uJ2d+/e0HRDFxZB9DnnDxrnXaJbBD3ClnyK4GDP7bhaBCIyBngK+I5SqvZQtxfuW+zxBSk81I0lKRUVFVRVVVFTUxNvURIat9tNRcXBeSNF5HTgYcAOPKWUemCv9fkYyRJDAS/wfaXUykOTOEEIp486OyyC6HPO4wtS1+qDhiTpTbw/2urB1wr1dvA2Go+GtZ2qrCYiB3Nux00RiMhA4DXgSqVUrzi+Ij0JUrjekNPpjMxO1PQ+ImIHHgNOAaqAhSIyVyn1VdSwnwNLlVLni8gIc3zvRvfihT88j6DDIog+5576aCP3vbGFZb86ldyMJO1XHObdX8Onj8Cvao3XnzxsvE5wRXAwxEwRiMiLwIlAkYhUAXcDTgCl1BPAr4BC4C+mPzuglJp4KPvUXco0fcBkYL1SaiOAiLwEnAtEK4IjgN8CKKXWiEiliJQopXb1ubS9zd4TyvaiutGL22kjJz1Rwo+HgCvDKLEd8IHfA65MSyoBiG3W0H6nvSmlrgWu7c19prt0jEATc8qBbVHvq4Ape41ZBlwAfCwik4FBGFlx1lcETV5Kc9xJlazQLU6zFIS/1XARuTL3Pz6JSXInXmfCMQJvCruGNDGnqyvc3pH5B4B8EVkK3AJ8CXR5dyIi14nIIhFZlBRxnR5YBCU5FgnCh2dM+zyGReDM2P/4JMZiikC7hjQxpwoYEPW+AtgRPUAp1aSUmqGUGgdcBRRjTKDch55mxCUM4awhZ/cWQVmuRRRBxCLwGMrApRVBUpDu1IpAE3MWAsNEZLCIuIDpwNzoASKSZ64Dw/05XynV1Mdyxob9zCxWSrG7qZ0SqyiCsCvI12I8nNZ1DVkgotNB2CLQFUg1sUIpFRCRm4G3MdJHn1ZKrRKR6831TwAjgedEJIgRRL4mbgL3NmGLwL7vzOK6Vh++YIhSq7qG3LnxlSeGWEwRdMwj0GhihVLqTeDNvZY9EfX6M2BYX8vVJwTaDCXQRV3+6iYjfmAZRbC3ayinf3zliSGWcg25nTZE0O0qNZpYEWjfb6AYsJBrKGwRtBqZQxZ2DVlKEYgI6U5dgVSjiRnRbSr3wnoWgakIdLA4+TCa02hFoNHEBL+324yhXY1ebALF2V0riqQjEiw25xFYOH3UUjECMCaV6WCxRhMjAt5OrqGV2xv587x1BEOwprqJoiwL1BgKE1YE7c1GbMTCE8ospwgynA49s1ijiRV7KYI3V+zkP1/tYmRpDjluJycengRzIXpKuNS2x6yHqS2C5CFddynTaGLHXoqg3uOnMNPFm7ceH0ehYoTNZlz8W/cY7y1sEVjEhusgQ7uGNJrYEWjvFCxu8PjIy3Dt5wNJjjMDWs3SH1oRJA+6b7FGE0P8bZ16EdR7fOQne7np/eHKgNbdxmsLu4YspwjSXY6U7keg0cSUfSwCv7UtAlcWtGiLIOnIcNp1sFijiRUBb6d+xZa3CJwZ4NnT8dqiWE4R6GCxRhNDoiaUKaWob/WTb2mLwGxOA9oiSCZ0sFijiSFRWUMeXxBfMGRt11B0WQmtCJKHDJedQEjhC4TiLYpGYz2iZhbXe3wA1nYNRZeV0K6h5CHdrECqrQKNppdRqpNF0ODxA1jcIoi6+OtaQ8lDpEuZXweMNZpeJegHVEQRpIZFkNXxWlcf/eaIyNMisltEVnazXkTkERFZLyLLReSo3tivblep0cSIQOfuZPWmRZCfaWGLIGwF2JzgsO5xxtIimAWcvp/138Fo3jEMuA54vDd2Gm5XqV1DGk0vE+5OZmYNNZgWQZ6VLYKwa8jCbiGIoSJQSs0H6vYz5FzgOWWwAMgTkbJD3a/uUqbRxIhwv2JzZnF9q2kRWDlGEM4UsrBbCOIbIygHtkW9rzKX7YOIXCcii0RkUU1NzX43mh5xDekYgUbTq0Qsgo4YQXaawzplp7tCWwQxR7pYproaqJSaqZSaqJSaWFy8/zK3uoG9RhMjAkYHsmjXUF6mhd1CEGURaEUQK6qAAVHvK4Adh7pRHSzWaGJERBGYriGPxWcVQ4ciiM4esiDxVARzgavM7KGjgUal1M5D3WjENWQWnlOqSyNDo9F8U7qyCKyuCLRr6NAQkReBz4DDRaRKRK4RketF5HpzyJvARmA98Ffgxt7Yb0ZkQlmAJq+f4x98nxe/2Nobm9ZoUhu/qQic0RaBdg1ZgZh1KFNKXXaA9Qq4qbf3G04f9fiCPDV/I1X1bSzb1sBlkwf29q40mtRiL4vAqDyaKhaBzhpKKuw2Ic1hY1tdG099vAmAnY3eOEulsRIicrqIfG1Ohryji/W5IvIvEVkmIqtEZEY85Ox1orKGAsEQzd6AtecQQIdLyOIWgeUUARgB438u3Y7XH+TwkmyqtSLQ9BIiYgcew5gQeQRwmYgcsdewm4CvlFJjgROBh0Qk+W+do2YWN7SlwBwC6Jg/oGMEyUeGy0EwpLhoQgVThhRQ3aQVgabXmAysV0ptVEr5gJcwJkdGo4BsEREgC2NiZfJPbImyCFJiVjHorKFkxu204bLbuPXbwynNddPY5tcTzDS9RU8mQj4KjMRIh14B3KqU6rIu+jeZLBl3IjOL3R11hqxuEbgy4Izfw5hL4i1JTIlZsDieXHBUBRkuO+V56ZTmGLMgqxu9DCm2tlbX9Ak9mQh5GrAUOBkYCrwjIh8ppZr2+aBSM4GZABMnTkzsXOcoi6C+tRVIAUUAMPkH8ZYg5lhSEdx00mGR16W5WhFoepWeTIScATxgZsatF5FNwAjgi74RMUYEvCA2sDmiehFY3DWUIljSNRRNWa6R86zjBJpeYiEwTEQGmwHg6RiTI6PZCkwDEJES4HCMOTPJTbhxvUhHLwIrl6BOISxpEUQTdg3pFFJNb6CUCojIzcDbgB14Wim1KjxRUin1BHAvMEtEVmC4km5XSu2Jm9C9RVTj+nqPH6ddyDRn8muSG8srgnSXnbwMp04h1fQaSqk3MWbGRy97Iur1DuDUvpYr5vi9kVnF4fISRmKUJtmxvGsIDKtAWwQazSHSySLwWb+8RAqRGoog180uHSPQaA6NqMb19R6/9QvOpRApoQjKcrVFoNEcMlGKoEFbBJYiJRRBaU46e1ra8QW6nNOj0Wh6QqC9k0WQEnMIUgTrK4KPHmIkGwC0e0ijORTMGIFSKjV6EaQQ1lYEbQ3w3m8Yt/VZ4JspglU7GtnZ2BYjwTSaJMTMGmr1BfEHlXYNWQhrK4I6wxIorP4EO8EexwmCIcUVT33Or+d+FUvpNJrkwrQI6lvNyWTaIrAM1lYEtYYisPsaGSsbejyXYHlVA/UeP59vqtWtLjWaMObMYl1ewnpYXBGsB7GhxMYpruU9tgjmrzUmgdZ7/Kzf3RJLCTVx5MILL+SNN94gFNJJBD0ibBHo8hKWw/qKIG8gUjGZk+zLqW7qmc9//roaSnKMiTNfbK6LpYSaOHLDDTfwwgsvMGzYMO644w7WrFkTb5ESGzNrqMlrWAS56doisAoxVQRxb+lXux4KD4PDvs2I0Hra6qsP+JFGj58vt9ZzycQBFGensXCTVgRW5dvf/jbPP/88S5YsobKyklNOOQVghIjMEBF9ldsbfxs43XjagwBkplm+Qk3KEDNFEPeWfkoZMYLCw2DYtwEY3LDAXKXY2djWpf//kw17CCmYOryYyZUFLNxc3yviaBKT2tpaZs2axVNPPcX48eMBdgFHAe/EV7IEIxSEkB8cblrajSZPuuCcdYilRRDfln4tu8DXYiiC0rG0OgsY276YYEjxu7e+5pjfzmPaHz7kj++sZUtta+Rj89fWkJ3mYNyAPCZV5rO9oY2qek+viKRJLC644AKOP/54PB4P//rXv5g7dy5AvVLqFozzURMmYMbXHG5aw4pAWwSWIZa/ZFct/absNeZRjFruO4Bs4NKuWvqJyHXAdQADBw7s2d5r1xvPhUPBZqO6+FiO3/4Bv3/rK56Yv5lTjyihyevnkXnreOLDDcyaMZmjhxQwf20Nxx5WiNNuY9LgAgAWbq6jIt/azatTkZtvvpmTTz65y3VKqYl9LE5iE9WdrKU5gMthw2m3dogxlYjlL/lNWvr1B8YBj4pIzj4fUmqmUmqiUmpicXFxz/YeUQRGtzLPwJMpkBY+++hdTj+ylMevmMBL1x3Dx7efzKDCDK59diGvLq5iR6OXqcONfYwozSE7zcEXm7R7yIqsXr2ahoaGyPv6+nqAHp5gKUbEIkjD0x7UbiGLEUtF0NOWfq8pg/VAuKXfoVO7HuxpkFMBgOuwEwE4v2grf5o+DrvN0FPleen8/ZopFGWn8dNXlwMwdZhxLbDbhAmV+SzUmUOW5K9//St5eXmR9/n5+aAVQddEGtdn0Noe0G4hixFLRRDfln61GyJuIYDhQ4fQmt6fy8prcDs73830y3Hz92umUJbrZkRpNgMKOtxAkwcXsH53C3XmbEqNdQiFQp0SBoLBIHRtyWraGoxndy4t7QGytCKwFDH7NePe0q92PRQNj7wVETIHT4YdS7ocPqAgg7dvm0q7v3OIYnKlESf4z6pqpk/uYXxCkxScdtppXHLJJVx//fWICE888QRAY7zlSki8pns0PQ+PL0iGdg1Zipiq9bi19AsGoG4THH5G5+XlE+Crf0JLDWTt6wHIcTvB3XnZ6IpchhRncsdrK/jg6xru+M4IKosye11kTd/zu9/9jieffJLHH38cpRSnnnoq//jHP6riLVdC4jX1ozuXlvY6st3aIrAS1gz7N241cp7NQHGE8gnGczdWQVekOey8ccvx/Pcpw5m/roZT/zif9bube1FYTbyw2WzccMMNvPrqq8yePZv/+q//irdIiUvENZRHq3YNWQ5rKgKz2Nw+iqBsLIgNti/+RptLd9m5Zdow5t58HL5giE831PaSoJp4sm7dOi666CKOOOIIhgwZwpAhQwBGx1uuhMTbYDxHXENaEVgJiyqCzqmjEdKyoHhkzxTBhnmwu3PtmaHFmeRlOFm9U1sEVmDGjBnccMMNOBwO3n//fa666ioAreW7oq3ByMJzppvBYh0jsBI9UgQicquI5IjB30RkiYj0vm+/t6hdD2m5kFm077ryowxFsL/y0m318OJl8PptnRaLCCNKs1lT3dTLAmviQVtbG9OmTUMpxaBBg7jnnnvAmNio2RtvI6TnoZTS6aMWpKcWwfeVUk0Ygd1ijPz/B2Im1aGgFOz6ykgdlS4yAcsnGBf6+k3db2PZy8YEmq2fQeP2TqtGlObwdXUzoZDuU5DsuN1uQqEQw4YN49FHH2XOnDkAByw214Niij8VkaXmY6WIBEWkIBbH0Gd4G8Cdiy8YIhBSWhFYjJ4qgvAV9QzgGaXUMhIx31opeO/XsPVTGHZK12PCAePt3QSMlYLFz0DeQEAZWUZRjCzLxuMLsk3XH0p6/vSnP+HxeHjkkUdYvHgxf//738GY1NgtPSmmqJT6X6XUOKXUOOBO4EOlVHLPSmxrMAPFZuVRnT5qKXqqCBaLyH8wFMHbIpINJFY3D6Xgnbvg4z/ChBlwwj43agb9RoIjvSNOsPRF+OdNHTMnt30ONWtg6k+N4PLK2Z0+PqLUqICh4wTJTTAY5JVXXiErK4uKigqeeeYZZs+eDdB6gI/2pJhiNJcBL/aO1HHE2wDpebrgnEXpqSK4BrgDmKSU8mCYz73bO+BQeecu+PTPMOkHcNYfIzOK98HuNC7wVYvgvXvhn9fD0r/DnOshFIJFz0BaDoy60HhsX2zMSTAZXpKNCDpOkOTY7XYWL158MK1IuyqmWN7VQBHJAE4HZne13hxznYgsEpFFNTU131SWviNsEfi0IrAiPf01jwGWKqVaReQKjHrtD8dOrIOgbBwcfROcdn/XsYFoyifAgseg6gs46irIr4T3fgNv9YNVc+CoK8GVCUeeD+/8Cla9Bsf/N2Ckkg4uzGRNlEXg8QWobfF1Kk2hSXzGjx/Pueeey8UXX0xmZmSSYN4BPtaTYophzgY+2Z9bSCk1E5gJMHHixMQNPJnBYm0RWJOe/pqPA2NFZCzwM+BvwHPACbES7Bsz+iLj0ROGnmQogmm/guN+bCyr3wxfzDReT7jaeM4bCAOmwMoORQAwoiybr3Z0WAS/nLOSuct28PuLx3Le+C5vDjUJSF1dHYWFhcybNy96cd4BPtaTYophpmMFt1AoZCgCdy4tZoxAp49ai54qgoBSSonIucDDSqm/icj3YilYTBl2CtyxFdy5HcvO/AO01kIoAKVRc4pGXQj//pkxp6CfURh1RGkO/15ZTWt7gPZAiNeX78TlsPGjl5dS2+rjmuMG9/EBaQ6GZ555Zp9ls2bN2nyAj0WKKQLbMS723917kIjkYtwoXXHIgsab9iZAgTsPj2kR6All1qKnv2aziNwJXAkcb2ZOJHdP12glAEbs4LIX9p1fcMR58Nad8MFv4eJZYM4lUArW7mpm4eY6fMEQr914HI/OW8+9r39FMBTiuqlD++pINAfJjBkzkH3diJX7+0wPiykCnA/8Ryl1oOBz4hM1qzjcplKXmLAWPQ0WXwq0Y8wnqMYIjv1vzKSKJ3tfGLJLYNpdRhqp6ToaWdaROfTiF9uYVJnPqPJcHrv8KCZXFvDqYl23LBk466yzOPPMMznzzDOZNm0aTU1NAMEDfU4p9aZSarhSaqhS6n5z2RN7FVScpZSaHjvp+5BIwTkdI7AqPfo1lVLVIvI8MElEzgK+UEo9F1vREohjb4Wtn8Pbv4D+R1FePpGsNAfPfbaZTXtaueVko5SF3SaMH5THMx9vJhAM4dCt/BKaCy+8sNP7yy67DLvdnh4ncRKXqF4ErT5DT+oy1NaipyUmLgG+AC4GLgE+F5EeRmYtgM0G5z8OOf3hH1dja93N4aXZrKluJjfdyRmjyyJDhxZn4QuGqKpvi6PAmoNh3bp1AK54y5FwRLmGWtsDOGxCmkPf5FiJntp3v8CYQ7AbQESKgXeBV2MlWMKRng+XPAfPfAdmncnkkgdZvAUuPKqiU8ezocVZAGyoadF9CxKc7OzsTjGC0tJSMLKCNNF0KkHdSGaao6vYiiaJ6alat4WVgEntN/isdeg/Dq6YDc07uWnzD6mw1fLdKZ27lh0WpQg0iU1zczNNTU2Rx9q1awEa4ixW4hGOEaTn0erTjeutSE8v5m+JyNsicrWIXA28wV6dx1KGQcfClf8kM9jIB0UPclh2oNPq3AwnRVlpbNid/MkiVmfOnDk0NnZ0pmxoaIADzyOwJmveMGbad4W3AcQOrixdedSi9EgRKKV+ijH7cQwwFpiplLo9loIlNAMmIVfMxtFUBfP3TZ4aWpypLYIk4Ne//jW5uR1pxHl5eQD94yVPXFk1Bxb+tet1bQ1GurUILVoRWJIe/6JKqdnsp2ZKyjFgMoy/Aj5/EiZ+3yh7bTK0Xxb/XrEzjsJpekIolFh1E+NKWwN4myAUoqbVj8Mm5GeacXOz4ByAxxckU88qthz7tQhEpFlEmrp4NIvIAauuHahuuznmRLNu+yoR+fBgDyQunHwXONLgP7/stHhocRb1Hj91rb44CabpCRMnTuTHP/4xGzZsYOPGjdx2220AqVlf3NsAKGhv4taXvuTcxz6hyes31pkF5wDDNaRnFVuO/SoCpVS2Uiqni0e2Uipnf5/tSd12EckD/gKco5Q6EiM9NXnILjFqEH39Jmx4P7J4aLGRLaTdQ4nNn//8Z1wuF5deeimXXHIJ6enpAFvjLVdcCAeEvQ3UtvjYWufhztdWGNVZzYJzgNmmUisCqxHLzJ+e1G3/LvCaUmorwF6ZScnB0TdC3iD4z12R8hSRFNLdWhEkMpmZmTzwwAMsWrSIRYsW8T//8z+QaH02+opwimhbA23+IBkuO28s38mLX2wzu5PlAYZrKEO7hixHLBVBT+q2DwfyReQDEVksIld1taGErtnudMPUn8CuFbB1AQDleemkOWzaIkhwTjnllHCmEAD19fUAw+IlT9xQqmPSmNdQBGeP6c/xw4r49b9WEWitj9Tm0sFiaxJLRdCTuu0OYAJwJnAacJeIDN/nQ0rNVEpNVEpNLC4u7n1JD5VRF4IrGxbPAsBmE4YUZ7GhRqeQJjJ79uwJZwoBkJ+fD8leTPFg8LdB0IxntTXgNe/6/3jpOGwCYgaL/cEQvkCILB0jsByxVAQ9qdteBbyllGpVSu0B5mOkpyYXrkwYc7FRmK6tHtAppMmAzWZj69aOkMDmzZuh+yYz1sXb2Ol1mz9IutNOUVYahxc6sBM0S1CbdYa0RWA5YqkIInXbRcSFUbd97l5j/h9GWWuH2dZvCrA6hjLFjglXQ8ALy18BjDjBtjoPXv8Bi1lq4sT999/Pcccdx5VXXsmVV17JCSecAEaPgdQi7BYCgp56AiFFulk2ZVh4wmR6Hi2+cAlqHSOwGjFTBEqpABCu274aeCVctz2qdvtq4C1gOUZRu6eUUitjJVNMKRsL/cfD4mdBKYb2yyKkYEttamYjJgOnn346ixYt4vDDD+fSSy/loYceglQMFocDxUDAY1i06WYZicFZRgqp0iWoLU1Mf1Gl1JvsVYoiuma7+f5/sUpvg6O+B6//CKoWMbTYCHVsqGnh8NLs+Mql6ZKnnnqKhx9+mKqqKsaNG8eCBQsgFWcWR1sErYYiCBdSHJjeDkCrZHYoAh0jsBypVzguloy+CJyZsORZhhQZKaTrdQppwvLwww+zcOFCBg0axPvvv8+XX34JEDjQ5yxH2CKwOVDm67BrqL/bUATVPjetZoxAWwTWQ/+ivUlattEPeeMHpLvsVBZmdGpyr0ks3G43brcbgPb2dkaMGAHgjqtQ8SAcLM4dgDKtg7BrqJ/TC8B2r5s2R9g1pGMEVkMrgt5mwGQje6i5mnED8vhsY228JdJ0Q0VFBQ0NDZx33nmccsop4fTR1KsLEnYN5Q1Emo3XYYug0G7EuLZ4nGS5tGvIquhftLcpn2g8Vy1i/MBR/HPpDnY2tlGWqzsgJhpz5swB4J577uGkk06isbGRuXPnboizWH1PWwOk5UBGIbY9W4COGIE70AzApmYbQ9J1sNiq6F+0tykbAzYnVC1k3IjjAPhyawNlo7UiSGTM1FFIyXkEDUYJifQ87D7DTRR2DYm3kRYyqWr00S/XiBHoWkPWQweLextnOpSOhu2LGVmWg8th48ut9fGWSqPpnnC/AXcuDl8z0DGPAG8jbY5squrb8PgC2ATcTn3ZsBr6F40FFRNh+xJcNsWo/jks3dYQb4k0mu4JVxd152FTATJoj1IEDfidOVTVe4w6Qy7dr9iKaEUQCyomgb8Vdq9m/MB8llc14g+m3jwlq2K5PhveBsMiMEtN59KK22VeGtoaUO5cmr0Bqhu9Oj5gUbQiiAXlE4znqoWMH5hHeyDE19XN8ZVJ0ytYss9GW0PEIgDIldZOFoHdVBBf72rWJagtilYEsaBgCKQXwPZFjBuQB6DjBNbBen02ooLFADm0RrKG8DbizCoAYPOeVh0otihaEcQCESNOULWI8rx0irPT+HJrQ7yl0vQOvdZnAxKg10bAB36PoQjMngMFdg9Ou83oU+CpIyO3CICQ0nMIrIpWBLGiYhLUfI20NzFuQJ4OGFuHXuuzAQnQayM8qzjKNVTkaDOWteyCYDvu4sERV5GeVWxNtCKIFRUTAQXblzB+YB4b97TS4Em9SasWxFp9NsKziqNcQ4U2s2Ju/WYAJL+S8nxjHowOFlsTrQhiRf+jjOeqhR1xAm0VWAFr9dkIF5xz50JaLiGEAntYERizjMmvpMJUBBnaNWRJtCKIFel5UDwCtn3B2Io8bAJLdZwg6bFcn42wRZCeBzYbbbZM8qSzRUDugIgi0E1prIlW77Fk4NGwcg6ZTmF4SbaOE1gES/XZCMcIzPiARzLJFbPXdsMWyO4PTjfleRmAdg1ZFW0RxJKBx0B7ozmxzAgYK5V6pWw0CYzZYzscH2iWLHIwFUH9ZsgfBBCxCHTWkDXRiiCWDJhiPG9bwLgBeTS2+dm0pzW+Mmk00USCxUbqaDOZZEcUwRbIrwSiFIG2CCyJVgSxJL8Sskph6wLGDcgH0PMJNIlFWwM40sGRBkAjmWSpFgi0Q9N2yDMsghGlOZw8oh+TB+fHUVhNrIipIuhJTRZz3CQRCYrIRbGUp88RMeIEWxdwWL8sstIcOk6gSSy8DRG3EECDyiQj1AIN2wAVsQjSXXaevnoSh/XT/betSMwUQU9qskSN+x1GFob1GHgMNG7D3rydMRW5fLlNl5rQJBDexkigGKA+lElGsBkaNhsLzBiBxtrE0iLoSU0WgFuA2UBi12M5WAaacYKtCxg/MI81O5tp8wXjK5NGEybci8CkLpiOU/mg5mtjgWkRaKxNLBXBAWuyiEg5cD7QKfVub+Jej+VQKBkNzsxInCAQUqzc0RhvqTQagyjXkFKK2qDZSW/nMrCnGTEujeWJpSLoSU2WPwG3K6X2e4sc93osh4LdAQMmmYogD9ATyzQJRFuHa6g9EKJRZRrLdy6DvIFg0/kkqUAsf+We1GSZCLwkIpuBi4C/iMh5MZQpPgw4GnatpNjppSI/XccJNIlDlEXQ5gvSiKkI9qzV8YEUIpZJwZGaLMB2jJos340eoJQaHH4tIrOA15VS/4yhTPFh4NGAgm0LGTeggCVbtCLQJAChILQ3RWIEbf5gh0WgQjo+kELEzCLoSU2WlKHcLEBXvZzxA/PZ0ehlV5M3vjJpNHuVl2jzB2kKWwSgFUEKEdNpgj2pyRK1/OpYyhJX3LmQVQK16xk91rj7+mpnEyU57jgLpklpogvOYbqGVJQiyNOuoVRBR4L6isJhsGcdQ4qNP9rGGl1qQhNnIiWo8wDw+oM0kdGxXlsEKYNWBH1F0TCoXUdhposct4ONNS3xlkiT6uxVZ6jNHySAg6DDtAp0sDhl0BWk+oqiYdBWj3jqGFKcpS0CTfyJblMJkYmOwbRc7E5Xp4lmGmujFUFfUTjMeN6zliHFmXyyfk985dEkNw1b4ZOH4bTfgsPVsXzpi7DiH/uOtzth2t1QElXlZS/XUJvfUATKnQuufrGRW5OQaNdQX1F0mPFcu46hxVnsamqnpT0QX5k0ycvq12HhU7B7FdWNXj4N31h88SRsX2zc7Uc/1r4Fq//VeRvN1YBAZhFgxAgA2kZfAZOu6cOD0cQbbRH0FXmDwO4yAsZlpwCwqaaV0RXa/NYcBA1mP+H6LcxcnMasTzex4M5p9KvfAkeeD2f/qfP4h0Z2fCZ6Gzn9IyWow64hNek6yHShSR20RdBX2OxQMBRq1zOkOAuAjXt0wFhzkIT7CddvZnuDh5CCNxZ9DW11XWf75Fd2fCZ6G1Fj2/whwCg5rUkttCLoS4oOgz3rGFSYgQhs0AFjzcFSH7YINlPd1A7AF0uWGsu6VASDulAEWzrNFQjHCNIc+rKQauhfvC8pHAb1m3DbQlTkp+sUUs0BafL6+dmry3jnq10dC5XquKg3bGFXo5cMl51Q3SZjWVdpn/mV0LTD6DwG4PdC845OSsPrD5LutCPSVb1IjZXRiqAvKRoGoQDUb2ZIkU4h1RyYTJeDt1ZW8260ImjZDYE2AFT9Zmpa2rloQgWVdjNg3JVFkDcIUGbnMaDRfI5SGm2+oHYLpShaEfQlkRRSI3No055WQqG9K3NrNB3YbcLkwYV8vqm2Y2E46FsyChq2oUJBhpdkc0xBE81k4HN2kYAQVg7hzmNhi6JTjMCwCDSph1YEfUlUCumQ4kza/EGqdfE5zQE4ekgBm2s9VDea50r4Ij74BCTkp5Q6SnPcjMpoYEuoHx983UWzv/Cdf1SQ2VheGRnS5g/idupLQiqif/W+JD0fMot1zaEkR0ROF5GvRWS9iNzRxfoTRaRRRJaaj18dyv6mDC4E6LAKwoHiwccDMEBqKM11U+jbyW57CXO+3L7vRrJKjY5jUUHmkD2N617bhj9oZAt5tWsoZdGKoK8pHAa16xmqU0iTEhGxA48B3wGOAC4TkSO6GPqRUmqc+fjNoezziP45ZKc5WLCxzlhQv9m4sBcfDsBA2y5Ksl1I41ZCeZV8tbNp343YbEbHsSiLoCGtP/9ZvZutdR5Au4ZSGa0I+hozhbRfdhqZLru2CJKPycB6pdRGpZQPeAk4N5Y7tNuESYMLOiyChi2GSyd3ACFsVNpqKFT1EPASzB3Ijoa2rmNP0XMJGrZQ6ywDYHu9EXg2XENaEaQiWhH0NYXDwLMHaatnSHEWG3QKabJRDmyLel9lLtubY0RkmYj8W0SO7G5jInKdiCwSkUU1NTXd7vToIQVsrGlld7O3YyKY3UmDsx+HOWuxmQFkR+Fg/EHF7ub2fTeSP8hQIkpB/RZ22YzG9FVhReDTFkGqohVBX1M03Hg2i89piyDp6CrJfu/b7yXAIKXUWODPwD+725hSaqZSaqJSamJxcXG3Ow3HCb5YXw1N2yPB351SwiB7TSSTKKvUSEioqvfsu5H8SqPuUP0maG9iO/06jfX6dYwgVdGKoK8pHWU8V69gSFEWOxrb8Ph08bkkogoYEPW+AtgRPUAp1aSUajFfvwk4RaToUHZ6ZP8cstIcrF27ulM/4S2hYspCu0yXj1BYMRSA7Q1t+24kPIt444cAbAoVdRqrYwSpi1YEfU1OOaQXwM5ljCrPQSlYtaOL4J4mUVkIDBORwSLiAqYDc6MHiEipmNNzRWQyxv+sdp8tfQMcdhsTK/PZveVrY0HeIJRSrPMVkhusg92rIac/5YX5QIe7pxPhVNFNhiJY7yvuNLbNp2MEqUpMFUEP0uwuF5Hl5uNTERkbS3kSAhEoGwvVyxlTkQfAsm0NcRVJ03OUUgHgZuBtYDXwilJqlYhcLyLXm8MuAlaKyDLgEWC6UuqQZw5OGVyIrXGr8Sa/kub2ABsCpqGx+SPIG0S6y05Rlqsb15BpEWyaD8Bqb1hphF1DIe0aSlFiVoY6Ks3uFAxzeqGIzFVKfRU1bBNwglKqXkS+A8wEpsRKpoShbAwseJzidKE8L52lWhEkFaa75829lj0R9fpR4NHe3u/RQwoQ2U3Q5sSeXcaumla2KbOBjKc2csdfnpfetUXgzjXmsnhqURmF7Kx34LQLu5raafMF8QVD2jWUosTSIjhgmp1S6lOlVL35dgGGv9X6lI6BoA9q1jB2QC7LqxrjLZEmCRhVnkulvYZ6ZynYbFQ3eTsUAUTu+CvyMyIpoftgxgmCOQMJKRhekg0QyV7TiiA1iaUi6GmaXZhrgH/HUJ7Eocz0gFUvZ2xFHlvrPNS1+uIrkybhcdptDHfVsilo+ParG73sIYeQI90YYFoEFfnpVO1vLgHgzTLi3aPLjbpE63cbisCtXUMpSSwVQU/S7IyBIidhKILbu1nfo1zrpKFgKDgzYWdUnKCqIa4iaZKD/mo3a7wF1LX6zNpD0uH7N+/2y/PT8QVC7GntmEswb80umrz+yNjmdOOe7EhTEazb3QxoiyBViaUiOGCaHYCIjAGeAs5VSnWZWdHTXOukwWaD0tFQvZzRFbmI6ICxphuUgl2rjEfVItyBRraqfnyxqZbqJi/5GU5s+YONsVEWAXRkA23a08r3Zy3iH4uqImMa0voDcERZNnabRCwCrQhSk1j2LI6k2QHbMdLsvhs9QEQGAq8BVyql1sZQlsSjbAwsfYEsp41h/bJ0nEDTNaEgPH5sp0VVtv74N9axq8lLSY4biofDlk8hqwQwYgRgKIKjBuazYKNxf7W9vg1GG/WJql2GZVCUlUZpjpt1YUXg0hnlqUjMFIFSKiAi4TQ7O/B0OM3OXP8E8CugEPiLmXYdUEpNjJVMCUXpGPDNhLqNjK3IY96a3SildHcoTWfEBpc81/HekU7rh1l8vqkOuw1Kc91w/E9g/JWGpYmRNQQdNYQ+NxXBriYvDDoWfjCPDRvygDXkZbgoz09n8RYjZ0PPI0hNYmkR9CTN7lrg2ljKkLCUjTGeq5cxdsBE/rG4iqr6NgYUZMRXLk1iYbPBEZ1r2k3ato4/vLuWTJeDUf1zwZ1jPEwy0xzkZzipqveglIpULa1u8hrzWMonUL9yDXabkON2UJGfzhebjDHaNZSaaDswXhSPBJsTdhqZQ6ADxpqecfTQQpSClvaA4Rrqgor8DKrq29ha56G6yYvTLh2NbYB6j5+8dCciEnElAXpCWYqiFUG8cLig3wioXs7hpdm4HLbkjhMEAx2N0TUxZUxFLmkO469bltudIkhne0NbJD5wwvB+7G72RlJKGzw+8jKcxljTlQTaIkhVtCKIJ2VjYecyXHbhyP45fLGpjl6oRBAf3vkVPHNGvKVICdIcdo4aaJSHKOlGERiziz0s2FhHUZaLbx1WiD+oqPMY81XqW/3kZ7iAjiwj0IogVdGKIJ5UTDZKA+xcyhmjyli6rYH/eXN1ciqD7Ytg51LDMtDEnKOHGGWpS7t1DaXj9Yd4d/UuJg8uiFgOYfdQvcdHXkQRdLiG9ISy1EQrgnhyxDlGH9mlL3Lt8YO56phB/PWjTfzhnSTMpK3bCKEANG478FjNITN98gBuOHFopETE3oQv7s3eAFMGF0ZiCWFF0ODxk2+6hkpz3djMZDVtEaQmWhHEk/R8GHEmrHgFCfq45+wjmT5pAH+et56nP94Ub+l6jrcJWs0Z33Ub4ytLilCS4+b200dgt3Wdblwe5e45ekghZbnG++qmDosgP9OwCFwOGyU5bhw2wWnXl4RURP/q8Wbc5dBWD2vfxmYT7j9/NFOHF/Pwe+uSp2FNfZTS0oogIQgrgvwMJ8P6ZVGU5cImxlyCNl+Q9kAoEiwGw5WkrYHURSuCeDP0JMgug6UvAEaj8ltOPozGNj+vLdkeZ+F6SPTFP9wcXRNXctxO8jOcRg8Dm+Cw2yjOTqO60Uu9GTAOB4sBBhRkkOWO6bQiTQKjf/l4Y7PDmEvh0z9Dy27I6sfEQfmMLs/lmU828d3JA7F1Y/4nDGFFkDdIWwQJxBNXTKB/VGpoaY6b6qZoRdBhEdz27eFcNtm7zzY0qYG2CBKBcd8FFYTlrwAgIlxz3GA21LTy4bokqLZat9Goc1M6WiuCBGLKkMJOM9VLctzsavLS4PEDRLKGwLAIJlUW9LmMmsRAK4JEoPhwKJ8Ii56OTMo6Y3QZ/bLTkiNoXLcJCoYYj7pNEArFWyJNF5TmutnZjWtIk9poRZAonHgH1G2AD38HGJkc3zu2ko/W7WFNdYI3t49WBMF2aN6n2rgmASjNddPsDbCjwShGF+0a0qQ2WhEkCsNOgfFXwMd/hO2LAbhs8kAyXXYufuIz/vLBetp8wTgL2QU+j3HhLxhsKALQ7qEEJTz5bE210YQmT1sEGhOtCBKJ0/7HyCCacwP4vRRkuphz07eYMriAB9/6mhN//z6vLNxGsKsWhPHCzBIK5Q/Bn2t2ytKKICGJKIKdzWS67Lgc+u+vMdBnQiLhzoVz/gx7voZ//wxCIYaXZPPU9ybxyn8dQ1luOj+bvZwzH/mIj9ftibe0BuZF/y/LQ5zxzAaU3aUVQYISrku0fneLtgY0ndCKINE4bBocdxsseRZeuzYSPJ48uIA5Nx7Lny8bT0t7gCv+9jlPfZQAF1zzov+3VbBuj5caR5kRM9AkHGGLwBcMkZ+p4wOaDrQiSESm3Q3f/jWsnA1/v9CYeYyRVnr22P68998ncMboUu57YzUz52+Ir6x1G/E48mhQGZw5uozlngJaq9fFV6YYIyKni8jXIrJeRO7Yz7hJIhIUkYv6Ur7uyExzkG1OGtMZQ5potCJIRETguB/B+U/C1s/gzxPg8ychYKT9pTnsPDx9PGeNKeN/3lzDA/9ew9fVzTGNHbQHug5UB2s3sj7Qj2kjSvj9xWOpd1cg9Zto9fpjJks8ERE78BjwHeAI4DIROaKbcb/DaNWaMIStAu0a0kSjFUEiM3Y6/GAelBxpxAwemwSvfh/e/BnOzx7mT8crLhxXxhMfbuC0P81n7K//w03PL2H1zt5NN/143R7G/vo/XZbI9u5ax/pgP77/rUrSXXYmHzWRDLzc9fz71Lf6elWOBGEysF4ptVEp5QNeAs7tYtwtwGxgd18KdyBKzTiBTh3VRKNLTCQ6ZWPhqrmw/l349BHY8SW01kJ7Iw7g95n9uHv8NJbnnszbnmH8c9lu3lixkzNHl3H22DLKctMpy3VTnJ2GSEepCqUUta0+8jNcnSpY+gIh7DaJLFu5vZH/+r9FOG02Zs7fiMtu4yenHW5sw+8lvW0nLRkncsxQoz7+oGGj4HPYvnElp/xR+M25ozhjdFm3h7d4Sx1rqpu5ZOKAZKl8WQ5E19quAqZEDxCRcuB84GRg0v42JiLXAdcBDBw4sFcF7YoSbRFouiCmikBETgceBuzAU0qpB/ZaL+b6MwAPcLVSakksZUpKRIx5BsNO6VjWugfWv4usfZucda9znO9Fjsso4q4xpzKv7TAe/rqG61fsAIwL+vCSLM4fX8HU4UV8uLaGfyyqYtOeVpx2oX9eOhkuB7uavNS1+ijKcnHhhAqmDivm1peWkpfh4tUbjuGR99bx6PvrCSrFiNJsdq5fxvUoho0Y3aFkzLkEfzwlh+tWuLnx+SUcP6yI204ZHumqBbChpoUH31rD26t2AfD8gq38/uKxHNG/owl7gtJV4ae9fXJ/Am5XSgWjlW9XKKVmAjMBJk6cGPO84LBrSFsEmmhipgiifKmnYNw1LRSRuUqpr6KGfQcYZj6mAI+z192VphsyiwzX0djp4PfC+ndg5Wu41r3J6d4GTrdBKMOFL62AFkc+NR6haV6I+vfsVJDDf2eXUjKyFL/Pi9fTgj+oUP2KsGX1Y02zm3kfr+GN+Vlku7P421VHU5bm5/4TcyhpauTL+a/xicpiuK0KnDB+3IQOuXIHgs1B/+AO/nn9VfzfJxt4cv4mLv7LRxw9pAi3PUR1bQM761twudK4fdphDCrO41f/WsM5j37MqUeW4LLbEKDF66Op1UNjq5cWv+AN2UBsjCzLZsKgfA4vycbjC9LYZsQjBhVmMKgwk8w0Oy3eAK2+IA6bkJvuJNvtQCnwh0IEgorCLBdpjoMqu1wFDIh6XwHsPZV6IvCSqQSKgDNEJKCU+ufB7LA3KYm4hrRFoOkglhZBxJcKICJhX2q0IjgXeE4ZjucFIpInImVKqZ0xlMt6ON0w8mzjEQoZ8xC2foatfgvu1hrcrTUUBbx4fX5aWj3kqmqcnuWwqRXEBs5Mo+hdkweAU4Efhm8YFfC08dIG/Ahgr2uIu2RYxxu7A/IGwkcP4fjoIWYAMwDcdL5cppnPnxhPZwAhl6DWCXa6r1XkFxfBbYJ/q828DZfI7bgyb9YVggshDxtB89GkbITM1yFsNF34N0aMPaYHX+4+LASGichgYDswHfhu9ACl1ODwaxGZBbyeCEoAoCziGtIWgaaDWCqCA/pSuxlTDnRSBH3tR01qbDboN9J47IXbfEQI+sHmMFxPAO0t0LrbiEF49hj9lP1txrhQwOiollViTHzz1EJLNaTlQMZeVSvP/ANsXWBs22YDpUCFjIfNAQ638RwKQNBnPIeC2MJjxGbIZHMYZbptDkOGQDvOgBenCmELBGhq8+GyCWkOG0opmtsDNLf5CYZCuGwKp02hQkH8/gDBgB8hhB2FkyBZBXkH9fUqpQIicjNGNpAdeFoptUpErjfXP3FQG+4jjhlayA+OH8zkwbrSqKaDWCqCnvhSezKmz/2oKYN9r7vCtCzjEa4ZdLAMPcl4xBAXhs8lmnSgX0z3aqCUehN4c69lXSoApdTVfSBSj8lMc/CLM/fJdtWkOLFM0+iJL7UnYzQajUYTQ2KpCCK+VBFxYfhS5+41Zi5wlRgcDTTq+IBGo9H0LTFzDfXQl/omRpxwPUb66IxYyaPRaDSaronpPIID+VLNbKGbYimDRqPRaPZPUkzl1Gg0Gk3s0IpAo9FoUhytCDQajSbF0YpAo9FoUhzZu6xwoiMiNcCWblYXAQnSwzFmWP0Y4318g5RSxfHYcYqf21Y/Poj/MXZ7biedItgfIrJIKTUx3nLEEqsfo9WP72Cx+vdi9eODxD5G7RrSaDSaFEcrAo1Go0lxrKYIZsZbgD7A6sdo9eM7WKz+vVj9+CCBj9FSMQKNRqPRfHOsZhFoNBqN5huiFYFGo9GkOJZRBCJyuoh8LSLrReSOeMtzqIjIABF5X0RWi8gqEbnVXF4gIu+IyDrzOf9A20pkRMQuIl+KyOvme0sd36FitfMa9LmdiMdnCUUgInbgMeA7wBHAZSKS7G2YAsB/K6VGAkcDN5nHdAfwnlJqGPCe+T6ZuRVYHfXeasd30Fj0vAZ9bifc8VlCEQCTgfVKqY1KKR/wEnBunGU6JJRSO5VSS8zXzRgnVDnGcT1rDnsWOC8uAvYCIlIBnAk8FbXYMsfXC1juvAZ9bpuvE+r4rKIIyoFtUe+rzGWWQEQqgfHA50BJuIub+dwXbXpjxZ+AnwGhqGVWOr5DxdLnNehzOw5ydYlVFIF0scwSebEikgXMBn6klGqKtzy9hYicBexWSi2OtywJjGXPa9DndiIR0w5lfUgVMCDqfQWwI06y9Boi4sT4ozyvlHrNXLxLRMqUUjtFpAzYHT8JD4lvAeeIyBmAG8gRkb9jnePrDSx5XoM+txPt+KxiESwEhonIYBFxAdOBuXGW6ZAQEQH+BqxWSv0hatVc4Hvm6+8B/6+vZesNlFJ3KqUqlFKVGL/XPKXUFVjk+HoJy53XoM9tc1hCHZ8lLAKlVEBEbgbeBuzA00qpVXEW61D5FnAlsEJElprLfg48ALwiItcAW4GL4yNezLD68fUYi57XoM/thDs+XWJCo9FoUhyruIY0Go1Gc5BoRaDRaDQpjlYEGo1Gk+JoRaDRaDQpjlYEGo1Gk+JoRaBBRE4MV0jUaKyCPq97jlYEGo1Gk+JoRZBEiMgVIvKFiCwVkSfNeuctIvKQiCwRkfdEpNgcO05EFojIchGZE659LiKHici7IrLM/MxQc/NZIvKqiKwRkefN2Z8aTczR53X80YogSRCRkcClwLeUUuOAIHA5kAksUUodBXwI3G1+5DngdqXUGGBF1PLngceUUmOBY4Gd5vLxwI8w6t4PwZj9qdHEFH1eJwaWKDGRIkwDJgALzZuadIyiVSHgZXPM34HXRCQXyFNKfWgufxb4h4hkA+VKqTkASikvgLm9L5RSVeb7pUAl8HHMj0qT6ujzOgHQiiB5EOBZpdSdnRaK3LXXuP3VDNmfWdwe9TqIPjc0fYM+rxMA7RpKHt4DLhKRfhDpfzoI4ze8yBzzXeBjpVQjUC8ix5vLrwQ+NGu+V4nIeeY20kQkoy8PQqPZC31eJwBaOyYJSqmvROSXwH9ExAb4gZuAVuBIEVkMNGL4W8Eoc/uE+YfYCMwwl18JPCkivzG3kTAVEDWphz6vEwNdfTTJEZEWpVRWvOXQaHoTfV73Ldo1pNFoNCmOtgg0Go0mxdEWgUaj0aQ4WhFoNBpNiqMVgUaj0aQ4WhFoNBpNiqMVgUaj0aQ4/x/nuNnCPyK0/wAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "utils.plot_history(history, ('loss', 'accuracy'));"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "test accuracy: 92.44%\n",
      "\n"
     ]
    }
   ],
   "source": [
    "out = abstractor_model.predict(X_test, verbose=0)\n",
    "yhat = np.argmax(out, axis=1)\n",
    "print('test accuracy: %.2f%%\\n' % (100*np.mean(yhat==y_test)))\n"
   ]
  }
 ],
 "metadata": {
  "interpreter": {
   "hash": "075427e9d7b4e88d8be5fc9c4269e9099fc3e822a226ecef7da6e611953d0392"
  },
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.8.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
