{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2023-04-22 19:06:59.939085: I tensorflow/core/platform/cpu_feature_guard.cc:193] This TensorFlow binary is optimized with oneAPI Deep Neural Network Library (oneDNN) to use the following CPU instructions in performance-critical operations:  AVX2 AVX512F AVX512_VNNI FMA\n",
      "To enable them in other operations, rebuild TensorFlow with the appropriate compiler flags.\n",
      "2023-04-22 19:07:00.084527: I tensorflow/core/util/util.cc:169] oneDNN custom operations are on. You may see slightly different numerical results due to floating-point round-off errors from different computation orders. To turn them off, set the environment variable `TF_ENABLE_ONEDNN_OPTS=0`.\n",
      "2023-04-22 19:07:00.126039: E tensorflow/stream_executor/cuda/cuda_blas.cc:2981] Unable to register cuBLAS factory: Attempting to register factory for plugin cuBLAS when one has already been registered\n",
      "2023-04-22 19:07:00.801449: W tensorflow/stream_executor/platform/default/dso_loader.cc:64] Could not load dynamic library 'libnvinfer.so.7'; dlerror: libnvinfer.so.7: cannot open shared object file: No such file or directory; LD_LIBRARY_PATH: /gpfs/loomis/apps/avx/software/cuDNN/8.1.1.33-CUDA-11.2.2/lib:/gpfs/loomis/apps/avx/software/CUDAcore/11.2.2/nvvm/lib64:/gpfs/loomis/apps/avx/software/CUDAcore/11.2.2/extras/CUPTI/lib64:/gpfs/loomis/apps/avx/software/CUDAcore/11.2.2/lib:/vast/palmer/apps/avx.grace/software/Code-Server/4.7.0/lib:/gpfs/loomis/apps/avx/software/Perl/5.32.0-GCCcore-10.2.0/lib:/gpfs/loomis/apps/avx/software/DB/18.1.40-GCCcore-10.2.0/lib:/gpfs/loomis/apps/avx/software/libreadline/8.0-GCCcore-10.2.0/lib:/gpfs/loomis/apps/avx/software/gettext/0.21-GCCcore-10.2.0/lib:/gpfs/loomis/apps/avx/software/ncurses/6.2-GCCcore-10.2.0/lib:/gpfs/loomis/apps/avx/software/libxml2/2.9.10-GCCcore-10.2.0/lib:/gpfs/loomis/apps/avx/software/XZ/5.2.5-GCCcore-10.2.0/lib:/gpfs/loomis/apps/avx/software/expat/2.2.9-GCCcore-10.2.0/lib:/gpfs/loomis/apps/avx/software/cURL/7.72.0-GCCcore-10.2.0/lib:/gpfs/loomis/apps/avx/software/zlib/1.2.11-GCCcore-10.2.0/lib:/gpfs/loomis/apps/avx/software/GCCcore/10.2.0/lib64:/gpfs/loomis/apps/avx/software/GCCcore/10.2.0/lib\n",
      "2023-04-22 19:07:00.801589: W tensorflow/stream_executor/platform/default/dso_loader.cc:64] Could not load dynamic library 'libnvinfer_plugin.so.7'; dlerror: libnvinfer_plugin.so.7: cannot open shared object file: No such file or directory; LD_LIBRARY_PATH: /gpfs/loomis/apps/avx/software/cuDNN/8.1.1.33-CUDA-11.2.2/lib:/gpfs/loomis/apps/avx/software/CUDAcore/11.2.2/nvvm/lib64:/gpfs/loomis/apps/avx/software/CUDAcore/11.2.2/extras/CUPTI/lib64:/gpfs/loomis/apps/avx/software/CUDAcore/11.2.2/lib:/vast/palmer/apps/avx.grace/software/Code-Server/4.7.0/lib:/gpfs/loomis/apps/avx/software/Perl/5.32.0-GCCcore-10.2.0/lib:/gpfs/loomis/apps/avx/software/DB/18.1.40-GCCcore-10.2.0/lib:/gpfs/loomis/apps/avx/software/libreadline/8.0-GCCcore-10.2.0/lib:/gpfs/loomis/apps/avx/software/gettext/0.21-GCCcore-10.2.0/lib:/gpfs/loomis/apps/avx/software/ncurses/6.2-GCCcore-10.2.0/lib:/gpfs/loomis/apps/avx/software/libxml2/2.9.10-GCCcore-10.2.0/lib:/gpfs/loomis/apps/avx/software/XZ/5.2.5-GCCcore-10.2.0/lib:/gpfs/loomis/apps/avx/software/expat/2.2.9-GCCcore-10.2.0/lib:/gpfs/loomis/apps/avx/software/cURL/7.72.0-GCCcore-10.2.0/lib:/gpfs/loomis/apps/avx/software/zlib/1.2.11-GCCcore-10.2.0/lib:/gpfs/loomis/apps/avx/software/GCCcore/10.2.0/lib64:/gpfs/loomis/apps/avx/software/GCCcore/10.2.0/lib\n",
      "2023-04-22 19:07:00.801600: W tensorflow/compiler/tf2tensorrt/utils/py_utils.cc:38] TF-TRT Warning: Cannot dlopen some TensorRT libraries. If you would like to use Nvidia GPU with TensorRT, please make sure the missing libraries mentioned above are installed properly.\n",
      "/tmp/tmp.CGVi2eb6Te/ipykernel_31158/1689438009.py:6: MatplotlibDeprecationWarning: The seaborn styles shipped by Matplotlib are deprecated since 3.6, as they no longer correspond to the styles shipped by seaborn. However, they will remain available as 'seaborn-v0_8-<style>'. Alternatively, directly use the seaborn API instead.\n",
      "  plt.style.use('seaborn-colorblind')\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import tensorflow as tf\n",
    "from tensorflow.keras import layers\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "plt.style.use('seaborn-colorblind')\n",
    "\n",
    "import itertools\n",
    "\n",
    "seed = 2718\n",
    "np.random.seed(seed)\n",
    "\n",
    "from models import AbstractorOrderRelation, reload_rel_model, rel_model_kwargs\n",
    "\n",
    "from sklearn.model_selection import train_test_split"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Relation-Learning: Task 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "# load data\n",
    "eval_task_data = np.load('object_sorting_datasets/task1_object_sort_dataset.npy', allow_pickle=True).item()\n",
    "\n",
    "objects, seqs, sorted_seqs, object_seqs, target, labels, start_token = (eval_task_data['objects'], eval_task_data['seqs'], eval_task_data['sorted_seqs'], eval_task_data['object_seqs'], \\\n",
    "    eval_task_data['target'], eval_task_data['labels'], eval_task_data['start_token'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_objects, object_dim = objects.shape\n",
    "index_pairs = np.array(list(itertools.permutations(range(num_objects), r=2)))\n",
    "object_order_relations = (index_pairs[:,0] < index_pairs[:, 1]).astype(int)\n",
    "\n",
    "object_pairs = np.array([objects[index_pair] for index_pair in index_pairs])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "object pair: (54, 42); relation: 0\n",
      "object pair: (49, 10); relation: 0\n",
      "object pair: (2, 23); relation: 1\n",
      "object pair: (24, 5); relation: 0\n",
      "object pair: (6, 12); relation: 1\n",
      "object pair: (11, 58); relation: 1\n",
      "object pair: (31, 45); relation: 1\n",
      "object pair: (34, 53); relation: 1\n",
      "object pair: (50, 21); relation: 0\n",
      "object pair: (56, 22); relation: 0\n"
     ]
    }
   ],
   "source": [
    "sample = np.random.choice(len(object_pairs), 10)\n",
    "for index_pair, relation in zip(index_pairs[sample], object_order_relations[sample]):\n",
    "    print(f'object pair: {tuple(index_pair)}; relation: {relation}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2023-04-22 19:07:02.782959: I tensorflow/core/platform/cpu_feature_guard.cc:193] This TensorFlow binary is optimized with oneAPI Deep Neural Network Library (oneDNN) to use the following CPU instructions in performance-critical operations:  AVX2 AVX512F AVX512_VNNI FMA\n",
      "To enable them in other operations, rebuild TensorFlow with the appropriate compiler flags.\n",
      "2023-04-22 19:07:03.397665: I tensorflow/core/common_runtime/gpu/gpu_device.cc:1616] Created device /job:localhost/replica:0/task:0/device:GPU:0 with 30975 MB memory:  -> device: 0, name: Tesla V100-SXM2-32GB, pci bus id: 0000:18:00.0, compute capability: 7.0\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model: \"abstractor_order_rel\"\n",
      "_________________________________________________________________\n",
      " Layer (type)                Output Shape              Param #   \n",
      "=================================================================\n",
      " time_distributed (TimeDistr  multiple                 2112      \n",
      " ibuted)                                                         \n",
      "                                                                 \n",
      " relational_abstracter (Simp  multiple                 54368     \n",
      " leAbstractor)                                                   \n",
      "                                                                 \n",
      " dense_1 (Dense)             multiple                  65        \n",
      "                                                                 \n",
      "=================================================================\n",
      "Total params: 56,545\n",
      "Trainable params: 56,545\n",
      "Non-trainable params: 0\n",
      "_________________________________________________________________\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2023-04-22 19:07:04.548411: I tensorflow/stream_executor/cuda/cuda_dnn.cc:384] Loaded cuDNN version 8101\n"
     ]
    }
   ],
   "source": [
    "model = AbstractorOrderRelation(sigmoid_reg_lamda=10., **rel_model_kwargs)\n",
    "model(object_pairs[:32])\n",
    "\n",
    "opt = tf.keras.optimizers.Adam(learning_rate=0.001)\n",
    "model.compile(loss='binary_crossentropy', optimizer=opt, metrics='binary_accuracy')\n",
    "model.summary()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<keras.callbacks.History at 0x2ba38ee37ca0>"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model.fit(object_pairs, object_order_relations, epochs=50, batch_size=len(object_pairs), verbose=False)\n",
    "model.dense.trainable = False\n",
    "model.fit(object_pairs, object_order_relations, epochs=1000, batch_size=len(object_pairs), verbose=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "126/126 [==============================] - 0s 1ms/step - loss: 0.2575 - binary_accuracy: 0.9901\n"
     ]
    }
   ],
   "source": [
    "model.evaluate(object_pairs, object_order_relations);"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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7NmzccIJJyAzMxO9e/fGueeeiw0bNrSK6QzrnD9/PkaOHNny476xY8fimWeeaXm8M6yxLbNnz0YQBJg2bVqL1hnWWVhYiCAIWv3l5eW1PN4Z1rif7du34zvf+Q569OiB9PR0HHvssVi1alXL45Gs1XVQFi1a5FJSUtw999zj1q9f76677jqXkZHhtmzZEvXUPjNPP/20mzlzpnvkkUccAPfYY4+1evy2225zmZmZ7pFHHnFr1651F154oevTp4+rqKiIZsKfga985Svuvvvuc2+//bZbs2aN+9rXvub69+/v9u3b1xLTGdb55JNPur/85S9uw4YNbsOGDe7GG290KSkp7u2333bOdY41/iOvvvqqGzhwoBs5cqS77rrrWvTOsM6bbrrJDR8+3O3YsaPlr6SkpOXxzrBG55zbu3evGzBggLvsssvcK6+84jZv3uyef/559/7777fERLHWDpuETjzxRHfllVe20oYMGeJ+8pOfRDSj9qVtEkokEi4vL8/ddtttLVptba3Lzs52v/71ryOYYftQUlLiALhly5Y55zrvOp1zrlu3bu63v/1tp1tjZWWlKygocEuWLHHjx49vSUKdZZ033XSTO+aYY+hjnWWNzjl3ww03uFNOOcV8PKq1dsiv48J2Z+0MbN68GcXFxa3WHI/HMX78+C/0msvLywEA3bt3B9A519nU1IRFixahqqoKY8eO7XRrvPrqq/G1r30NZ5xxRiu9M61z48aNyM/Px6BBg/Ctb30LmzZtAtC51vjkk09i9OjROP/889G7d2+MGjUK99xzT8vjUa21QyahsN1ZOwP719WZ1uycw/XXX49TTjkFI0aMANC51rl27Vp07doV8XgcV155JR577DEMGzasU61x0aJFeOONNzB79mzvsc6yzjFjxuCBBx7Ac889h3vuuQfFxcUYN24c9uzZ02nWCACbNm3C/PnzUVBQgOeeew5XXnklrr32WjzwwAMAojueB62AaXvwabuzdiY605qvueYavPXWW3j55Ze9xzrDOo866iisWbMGZWVleOSRR3DppZdi2bJlLY9/0de4bds2XHfddVi8eDG6dOlixn3R1zl58uSWfx999NEYO3YsjjjiCCxYsAAnnXQSgC/+GgEgkUhg9OjRKCoqAgCMGjUK69atw/z58/Ev//IvLXH/12vtkFdCYbuzdgb2u3E6y5p/+MMf4sknn8SLL77Y0poD6FzrTE1NxZFHHonRo0dj9uzZOOaYY3DHHXd0mjWuWrUKJSUlOP7445GcnIzk5GQsW7YMv/zlL5GcnNyyli/6OtuSkZGBo48+Ghs3buw0xxIA+vTpg2HDhrXShg4diq1btwKI7rXZIZPQP3Zn/UeWLFmCcePGRTSrg8ugQYOQl5fXas319fVYtmzZF2rNzjlcc801ePTRR/HCCy9g0KBBrR7vLOtkOOdQV1fXadZ4+umnY+3atVizZk3L3+jRo3HxxRdjzZo1OPzwwzvFOttSV1eHd955B3369Ok0xxIATj75ZO/nEu+9915Ln7fI1nrQLA+fk/0W7XvvvdetX7/eTZs2zWVkZLgPP/ww6ql9ZiorK93q1avd6tWrHQA3d+5ct3r16hbb+W233eays7Pdo48+6tauXeu+/e1vf+GsoFdddZXLzs52S5cubWV5ra6ubonpDOucMWOGW758udu8ebN766233I033uhisZhbvHixc65zrJHxj+445zrHOn/84x+7pUuXuk2bNrmVK1e6s846y2VmZra813SGNTrXbLNPTk52t956q9u4caP7/e9/79LT092DDz7YEhPFWjtsEnLOubvuussNGDDApaamuuOOO67F5vtF5cUXX3QAvL9LL73UOddskbzppptcXl6ei8fj7tRTT3Vr166NdtIhYesD4O67776WmM6wzssvv7zl3OzVq5c7/fTTWxKQc51jjYy2SagzrHP/b2FSUlJcfn6+mzJlilu3bl3L451hjft56qmn3IgRI1w8HndDhgxxd999d6vHo1irWjkIIYSIjA55T0gIIcShgZKQEEKIyFASEkIIERlKQkIIISJDSUgIIURkKAkJIYSIDCUhIYQQkaEkJIQQIjKUhIQQQkSGkpAQQojIUBISQggRGUpCQgghIuP/A9t0lI8IWvQ3AAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "relational_crossattention = model.abstractor.abstracter_layers[0].relational_crossattention.mha\n",
    "\n",
    "embedded_objects = model.embedder(np.expand_dims(objects, axis=0))\n",
    "\n",
    "_, attention_scores = relational_crossattention(\n",
    "    query=embedded_objects, key=embedded_objects, value=embedded_objects, return_attention_scores=True)\n",
    "\n",
    "rel_img = np.transpose(attention_scores[0])\n",
    "\n",
    "fig, ax = plt.subplots()\n",
    "im = ax.imshow(rel_img)\n",
    "ax.invert_yaxis()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "model.save_weights('prelearning_models/task1_rel_model.h5')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "126/126 [==============================] - 1s 1ms/step - loss: 0.2575 - binary_accuracy: 0.9901\n"
     ]
    }
   ],
   "source": [
    "# reload model and check it was saved correctly\n",
    "model = reload_rel_model(weights_path='prelearning_models/task1_rel_model.h5', object_dim=object_dim, kwargs=rel_model_kwargs)\n",
    "model.compile(loss='binary_crossentropy', optimizer=opt, metrics='binary_accuracy')\n",
    "model.evaluate(object_pairs, object_order_relations);"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Relation-Learning: Task 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "# load data\n",
    "eval_task_data = np.load('object_sorting_datasets/task2_object_sort_dataset.npy', allow_pickle=True).item()\n",
    "\n",
    "objects, seqs, sorted_seqs, object_seqs, target, labels, start_token = (eval_task_data['objects'], eval_task_data['seqs'], eval_task_data['sorted_seqs'], eval_task_data['object_seqs'], \\\n",
    "    eval_task_data['target'], eval_task_data['labels'], eval_task_data['start_token'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_objects, object_dim = objects.shape\n",
    "index_pairs = np.array(list(itertools.permutations(range(num_objects), r=2)))\n",
    "object_order_relations = (index_pairs[:,0] < index_pairs[:, 1]).astype(int)\n",
    "\n",
    "object_pairs = np.array([objects[index_pair] for index_pair in index_pairs])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "object pair: (2, 4); relation: 1\n",
      "object pair: (56, 55); relation: 0\n",
      "object pair: (29, 44); relation: 1\n",
      "object pair: (39, 46); relation: 1\n",
      "object pair: (26, 19); relation: 0\n",
      "object pair: (41, 14); relation: 0\n",
      "object pair: (37, 14); relation: 0\n",
      "object pair: (58, 30); relation: 0\n",
      "object pair: (26, 2); relation: 0\n",
      "object pair: (41, 60); relation: 1\n"
     ]
    }
   ],
   "source": [
    "sample = np.random.choice(len(object_pairs), 10)\n",
    "for index_pair, relation in zip(index_pairs[sample], object_order_relations[sample]):\n",
    "    print(f'object pair: {tuple(index_pair)}; relation: {relation}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model: \"abstractor_order_rel\"\n",
      "_________________________________________________________________\n",
      " Layer (type)                Output Shape              Param #   \n",
      "=================================================================\n",
      " time_distributed_2 (TimeDis  multiple                 2112      \n",
      " tributed)                                                       \n",
      "                                                                 \n",
      " relational_abstracter (Simp  multiple                 54368     \n",
      " leAbstractor)                                                   \n",
      "                                                                 \n",
      " dense_9 (Dense)             multiple                  65        \n",
      "                                                                 \n",
      "=================================================================\n",
      "Total params: 56,545\n",
      "Trainable params: 56,545\n",
      "Non-trainable params: 0\n",
      "_________________________________________________________________\n"
     ]
    }
   ],
   "source": [
    "model = AbstractorOrderRelation(sigmoid_reg_lamda=10., **rel_model_kwargs)\n",
    "model(object_pairs)\n",
    "\n",
    "opt = tf.keras.optimizers.Adam(learning_rate=0.001)\n",
    "model.compile(loss='binary_crossentropy', optimizer=opt, metrics='binary_accuracy')\n",
    "model.summary()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<keras.callbacks.History at 0x2ba311bf1070>"
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model.fit(object_pairs, object_order_relations, epochs=50, batch_size=len(object_pairs), verbose=False)\n",
    "model.dense.trainable = False\n",
    "model.fit(object_pairs, object_order_relations, epochs=1000, batch_size=len(object_pairs), verbose=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "126/126 [==============================] - 0s 1ms/step - loss: 0.2001 - binary_accuracy: 0.9975\n"
     ]
    }
   ],
   "source": [
    "model.evaluate(object_pairs, object_order_relations);"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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IURkKAkJIYSIjF6dhPpad9ann34aZ511FmpraxEEAR588MGdPnfOob6+HrW1tSguLsakSZOwYsWKaCb7IZk1axaOPvpolJeXY9CgQTj77LOxatWqnWL6wjrnzZuHMWPGdP9x34QJE/DYY491f94X1rgrs2bNQhAEmDZtWrfWF9ZZX1+PIAh2+ldT836jzL6wxvd45513cNFFF2HAgAEoKSnBEUccgRdeeKH780jW6nopCxcudAUFBe7OO+90K1eudFdffbUrLS1169evj3pqH5pHH33UzZgxw913330OgHvggQd2+vzWW2915eXl7r777nPLly935513nhs8eLBLpVLRTPhD8MlPftLdfffd7q9//atbtmyZO+OMM9zQoUNda2trd0xfWOdDDz3kHnnkEbdq1Sq3atUqd8MNN7iCggL317/+1TnXN9b4tzz33HNu+PDhbsyYMe7qq6/u1vvCOm+88UY3atQot3Hjxu5/jY2N3Z/3hTU659y2bdvcsGHD3CWXXOL+/Oc/u3Xr1rknn3zSvfHGG90xUay11yahY445xl1++eU7aQcffLD71re+FdGMepZdk1A+n3c1NTXu1ltv7dY6OztdZWWl++lPfxrBDHuGxsZGB8AtWbLEOdd31+mcc/369XM///nP+9waW1paXF1dnVu8eLGbOHFidxLqK+u88cYb3eGHH04/6ytrdM6566+/3p1wwgnm51GttVf+Oi5sd9a+wLp169DQ0LDTmpPJJCZOnPixXnNz845yzP379wfQN9eZy+WwcOFCtLW1YcKECX1ujVdccQXOOOMMnHrqqTvpfWmdq1evRm1tLUaMGIHzzz8fa9euBdC31vjQQw9h3LhxOPfcczFo0CCMHTsWd955Z/fnUa21VyahsN1Z+wLvrasvrdk5h2uuuQYnnHACRo8eDaBvrXP58uUoKytDMpnE5ZdfjgceeACHHnpon1rjwoUL8eKLL2LWrFneZ31lnePHj8c999yDJ554AnfeeScaGhpw3HHHYevWrX1mjQCwdu1azJs3D3V1dXjiiSdw+eWX46qrrsI999wDILrjuccKmPYEu9udtS/Rl9Z85ZVX4pVXXsEf//hH77O+sM6DDjoIy5YtQ1NTE+677z5cfPHFWLJkSffnH/c1btiwAVdffTUWLVqEoqIiM+7jvs7TTz+9+78PO+wwTJgwAQcccAAWLFiAY489FsDHf40AkM/nMW7cOMycORMAMHbsWKxYsQLz5s3DF7/4xe64/+u19sonobDdWfsC77lx+sqav/71r+Ohhx7CH/7wh+7WHEDfWmdhYSEOPPBAjBs3DrNmzcLhhx+OH/7wh31mjS+88AIaGxtx1FFHIZFIIJFIYMmSJfjRj36ERCLRvZaP+zp3pbS0FIcddhhWr17dZ44lAAwePBiHHnroTtohhxyCt956C0B012avTEJ/2531b1m8eDGOO+64iGa1ZxkxYgRqamp2WnMmk8GSJUs+Vmt2zuHKK6/E/fffj9///vcYMWLETp/3lXUynHNIp9N9Zo2nnHIKli9fjmXLlnX/GzduHC688EIsW7YM+++/f59Y566k02m8+uqrGDx4cJ85lgBw/PHHe38u8frrr3f3eYtsrXvM8vARec+ifdddd7mVK1e6adOmudLSUvfmm29GPbUPTUtLi3vppZfcSy+95AC4OXPmuJdeeqnbdn7rrbe6yspKd//997vly5e7z3/+8x87K+jXvvY1V1lZ6Z566qmdLK/t7e3dMX1hndOnT3dPP/20W7dunXvllVfcDTfc4GKxmFu0aJFzrm+skfG37jjn+sY6r732WvfUU0+5tWvXumeffdadeeaZrry8vPte0xfW6NwOm30ikXC33HKLW716tfvlL3/pSkpK3C9+8YvumCjW2muTkHPO/fjHP3bDhg1zhYWF7sgjj+y2+X5c+cMf/uAAeP8uvvhi59wOi+SNN97oampqXDKZdCeddJJbvnx5tJMOCVsfAHf33Xd3x/SFdV566aXd5+bAgQPdKaec0p2AnOsba2TsmoT6wjrf+1uYgoICV1tb68455xy3YsWK7s/7whrf4+GHH3ajR492yWTSHXzwwe6OO+7Y6fMo1qpWDkIIISKjV74TEkIIsXegJCSEECIylISEEEJEhpKQEEKIyFASEkIIERlKQkIIISJDSUgIIURkKAkJIYSIDCUhIYQQkaEkJIQQIjKUhIQQQkSGkpAQQojI+H8pYStjtF5SfAAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "relational_crossattention = model.abstractor.abstracter_layers[0].relational_crossattention.mha\n",
    "\n",
    "embedded_objects = model.embedder(np.expand_dims(objects, axis=0))\n",
    "\n",
    "_, attention_scores = relational_crossattention(\n",
    "    query=embedded_objects, key=embedded_objects, value=embedded_objects, return_attention_scores=True)\n",
    "\n",
    "rel_img = np.transpose(attention_scores[0])\n",
    "\n",
    "fig, ax = plt.subplots()\n",
    "im = ax.imshow(rel_img)\n",
    "ax.invert_yaxis()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [],
   "source": [
    "model.save_weights('prelearning_models/task2_rel_model.h5')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "126/126 [==============================] - 1s 1ms/step - loss: 0.2001 - binary_accuracy: 0.9975\n"
     ]
    }
   ],
   "source": [
    "# reload model and check it was saved correctly\n",
    "model = reload_rel_model(weights_path='prelearning_models/task2_rel_model.h5', object_dim=object_dim, kwargs=rel_model_kwargs)\n",
    "model.compile(loss='binary_crossentropy', optimizer=opt, metrics='binary_accuracy')\n",
    "model.evaluate(object_pairs, object_order_relations);"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Argsort-Learning: Task 2\n",
    "Use 'relation model' to learn arg-sorting on Task 2. This will be used to initialize the argsort model on Task 1."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [],
   "source": [
    "from models import AutoregressiveSimpleAbstractor, initialize_with_rel_model, decoder_kwargs, reload_argsort_model\n",
    "from transformer_modules import TeacherForcingAccuracy\n",
    "from evaluation_utils import evaluate_seq2seq_model\n",
    "\n",
    "loss = tf.keras.losses.SparseCategoricalCrossentropy(from_logits=True, ignore_class=None, name='sparse_categorical_crossentropy')\n",
    "create_opt = lambda : tf.keras.optimizers.Adam()\n",
    "\n",
    "teacher_forcing_acc_metric = TeacherForcingAccuracy(ignore_class=None)\n",
    "metrics = [teacher_forcing_acc_metric]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [],
   "source": [
    "# load data\n",
    "eval_task_data = np.load('object_sorting_datasets/task2_object_sort_dataset.npy', allow_pickle=True).item()\n",
    "\n",
    "objects, seqs, sorted_seqs, object_seqs, target, labels, start_token = (eval_task_data['objects'], eval_task_data['seqs'], eval_task_data['sorted_seqs'], eval_task_data['object_seqs'], \\\n",
    "    eval_task_data['target'], eval_task_data['labels'], eval_task_data['start_token'])\n",
    "\n",
    "num_objects, object_dim = objects.shape\n",
    "\n",
    "test_size = 0.2\n",
    "val_size = 0.1\n",
    "\n",
    "seqs_train, seqs_test, sorted_seqs_train, sorted_seqs_test, object_seqs_train, object_seqs_test, target_train, target_test, labels_train, labels_test = train_test_split(\n",
    "    seqs, sorted_seqs, object_seqs, target, labels, test_size=test_size, random_state=seed)\n",
    "seqs_train, seqs_val, sorted_seqs_train, sorted_seqs_val, object_seqs_train, object_seqs_val, target_train, target_val, labels_train, labels_val = train_test_split(\n",
    "    seqs_train, sorted_seqs_train, object_seqs_train, target_train, labels_train, test_size=val_size/(1-test_size), random_state=seed)\n",
    "\n",
    "seqs_length = seqs.shape[1]\n",
    "\n",
    "source_train, source_val, source_test = object_seqs_train, object_seqs_val, object_seqs_test\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [],
   "source": [
    "argsort_model_kwargs = dict(embedding_dim=rel_model_kwargs['embedding_dim'], \n",
    "    seqs_length=seqs_length, decoder_kwargs=decoder_kwargs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model: \"autoregressive_simple_abstractor\"\n",
      "_________________________________________________________________\n",
      " Layer (type)                Output Shape              Param #   \n",
      "=================================================================\n",
      " time_distributed_4 (TimeDis  multiple                 2112      \n",
      " tributed)                                                       \n",
      "                                                                 \n",
      " relational_abstracter (Simp  multiple                 54880     \n",
      " leAbstractor)                                                   \n",
      "                                                                 \n",
      " target_embedder (Embedding)  multiple                 704       \n",
      "                                                                 \n",
      " add_pos_embedding_target (A  multiple                 0         \n",
      " ddPositionalEmbedding)                                          \n",
      "                                                                 \n",
      " decoder (Decoder)           multiple                  274624    \n",
      "                                                                 \n",
      " final_layer (Dense)         multiple                  650       \n",
      "                                                                 \n",
      "=================================================================\n",
      "Total params: 332,970\n",
      "Trainable params: 332,970\n",
      "Non-trainable params: 0\n",
      "_________________________________________________________________\n"
     ]
    }
   ],
   "source": [
    "argsort_model = AutoregressiveSimpleAbstractor(**argsort_model_kwargs)\n",
    "\n",
    "argsort_model((source_train[:32], target_train[:32]));\n",
    "\n",
    "rel_model = reload_rel_model(weights_path='prelearning_models/task2_rel_model.h5', object_dim=object_dim, kwargs=rel_model_kwargs)\n",
    "initialize_with_rel_model(argsort_model, rel_model)\n",
    "\n",
    "argsort_model.compile(loss=loss, optimizer=create_opt(), metrics=metrics)\n",
    "argsort_model.summary()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 1/200\n",
      "20/20 [==============================] - 5s 58ms/step - loss: 2.4095 - teacher_forcing_accuracy: 0.1034 - val_loss: 2.2988 - val_teacher_forcing_accuracy: 0.1138\n",
      "Epoch 2/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 2.3190 - teacher_forcing_accuracy: 0.1077 - val_loss: 2.2876 - val_teacher_forcing_accuracy: 0.1268\n",
      "Epoch 3/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 2.2866 - teacher_forcing_accuracy: 0.1230 - val_loss: 2.2209 - val_teacher_forcing_accuracy: 0.1600\n",
      "Epoch 4/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 2.1634 - teacher_forcing_accuracy: 0.1797 - val_loss: 2.0000 - val_teacher_forcing_accuracy: 0.2341\n",
      "Epoch 5/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 1.9755 - teacher_forcing_accuracy: 0.2349 - val_loss: 1.8448 - val_teacher_forcing_accuracy: 0.2628\n",
      "Epoch 6/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 1.8496 - teacher_forcing_accuracy: 0.2594 - val_loss: 1.7518 - val_teacher_forcing_accuracy: 0.2656\n",
      "Epoch 7/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 1.7658 - teacher_forcing_accuracy: 0.2756 - val_loss: 1.6401 - val_teacher_forcing_accuracy: 0.2903\n",
      "Epoch 8/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 1.6732 - teacher_forcing_accuracy: 0.2908 - val_loss: 1.5710 - val_teacher_forcing_accuracy: 0.2922\n",
      "Epoch 9/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 1.6171 - teacher_forcing_accuracy: 0.2936 - val_loss: 1.5452 - val_teacher_forcing_accuracy: 0.2946\n",
      "Epoch 10/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 1.5926 - teacher_forcing_accuracy: 0.2959 - val_loss: 1.5490 - val_teacher_forcing_accuracy: 0.2931\n",
      "Epoch 11/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 1.5842 - teacher_forcing_accuracy: 0.3046 - val_loss: 1.5085 - val_teacher_forcing_accuracy: 0.3344\n",
      "Epoch 12/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 1.4846 - teacher_forcing_accuracy: 0.3777 - val_loss: 1.2908 - val_teacher_forcing_accuracy: 0.4569\n",
      "Epoch 13/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 1.2860 - teacher_forcing_accuracy: 0.4632 - val_loss: 1.0665 - val_teacher_forcing_accuracy: 0.5306\n",
      "Epoch 14/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 1.1225 - teacher_forcing_accuracy: 0.5181 - val_loss: 0.9118 - val_teacher_forcing_accuracy: 0.6016\n",
      "Epoch 15/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.9677 - teacher_forcing_accuracy: 0.5873 - val_loss: 0.6992 - val_teacher_forcing_accuracy: 0.7097\n",
      "Epoch 16/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.8365 - teacher_forcing_accuracy: 0.6491 - val_loss: 0.6093 - val_teacher_forcing_accuracy: 0.7459\n",
      "Epoch 17/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.7734 - teacher_forcing_accuracy: 0.6778 - val_loss: 0.5568 - val_teacher_forcing_accuracy: 0.7736\n",
      "Epoch 18/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.7329 - teacher_forcing_accuracy: 0.6945 - val_loss: 0.5084 - val_teacher_forcing_accuracy: 0.7899\n",
      "Epoch 19/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.6951 - teacher_forcing_accuracy: 0.7103 - val_loss: 0.4813 - val_teacher_forcing_accuracy: 0.8032\n",
      "Epoch 20/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.6614 - teacher_forcing_accuracy: 0.7242 - val_loss: 0.4577 - val_teacher_forcing_accuracy: 0.8096\n",
      "Epoch 21/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.6363 - teacher_forcing_accuracy: 0.7349 - val_loss: 0.4353 - val_teacher_forcing_accuracy: 0.8210\n",
      "Epoch 22/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.6135 - teacher_forcing_accuracy: 0.7426 - val_loss: 0.4029 - val_teacher_forcing_accuracy: 0.8357\n",
      "Epoch 23/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.5811 - teacher_forcing_accuracy: 0.7572 - val_loss: 0.3752 - val_teacher_forcing_accuracy: 0.8451\n",
      "Epoch 24/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.5561 - teacher_forcing_accuracy: 0.7663 - val_loss: 0.3557 - val_teacher_forcing_accuracy: 0.8521\n",
      "Epoch 25/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.5331 - teacher_forcing_accuracy: 0.7760 - val_loss: 0.3431 - val_teacher_forcing_accuracy: 0.8563\n",
      "Epoch 26/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.5222 - teacher_forcing_accuracy: 0.7804 - val_loss: 0.3283 - val_teacher_forcing_accuracy: 0.8685\n",
      "Epoch 27/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.4976 - teacher_forcing_accuracy: 0.7921 - val_loss: 0.3158 - val_teacher_forcing_accuracy: 0.8779\n",
      "Epoch 28/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.4715 - teacher_forcing_accuracy: 0.8093 - val_loss: 0.2733 - val_teacher_forcing_accuracy: 0.9016\n",
      "Epoch 29/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.4365 - teacher_forcing_accuracy: 0.8260 - val_loss: 0.2369 - val_teacher_forcing_accuracy: 0.9173\n",
      "Epoch 30/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.4067 - teacher_forcing_accuracy: 0.8383 - val_loss: 0.2298 - val_teacher_forcing_accuracy: 0.9213\n",
      "Epoch 31/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.3931 - teacher_forcing_accuracy: 0.8462 - val_loss: 0.2083 - val_teacher_forcing_accuracy: 0.9276\n",
      "Epoch 32/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.3780 - teacher_forcing_accuracy: 0.8521 - val_loss: 0.1958 - val_teacher_forcing_accuracy: 0.9330\n",
      "Epoch 33/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.3722 - teacher_forcing_accuracy: 0.8539 - val_loss: 0.1992 - val_teacher_forcing_accuracy: 0.9308\n",
      "Epoch 34/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.3477 - teacher_forcing_accuracy: 0.8642 - val_loss: 0.1867 - val_teacher_forcing_accuracy: 0.9345\n",
      "Epoch 35/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.3427 - teacher_forcing_accuracy: 0.8653 - val_loss: 0.1709 - val_teacher_forcing_accuracy: 0.9401\n",
      "Epoch 36/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.3309 - teacher_forcing_accuracy: 0.8709 - val_loss: 0.1661 - val_teacher_forcing_accuracy: 0.9408\n",
      "Epoch 37/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.3274 - teacher_forcing_accuracy: 0.8722 - val_loss: 0.1649 - val_teacher_forcing_accuracy: 0.9439\n",
      "Epoch 38/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.3178 - teacher_forcing_accuracy: 0.8763 - val_loss: 0.1673 - val_teacher_forcing_accuracy: 0.9409\n",
      "Epoch 39/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.3171 - teacher_forcing_accuracy: 0.8765 - val_loss: 0.1608 - val_teacher_forcing_accuracy: 0.9423\n",
      "Epoch 40/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.3047 - teacher_forcing_accuracy: 0.8822 - val_loss: 0.1528 - val_teacher_forcing_accuracy: 0.9479\n",
      "Epoch 41/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2992 - teacher_forcing_accuracy: 0.8839 - val_loss: 0.1413 - val_teacher_forcing_accuracy: 0.9502\n",
      "Epoch 42/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2947 - teacher_forcing_accuracy: 0.8845 - val_loss: 0.1462 - val_teacher_forcing_accuracy: 0.9479\n",
      "Epoch 43/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2858 - teacher_forcing_accuracy: 0.8904 - val_loss: 0.1326 - val_teacher_forcing_accuracy: 0.9537\n",
      "Epoch 44/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2806 - teacher_forcing_accuracy: 0.8909 - val_loss: 0.1306 - val_teacher_forcing_accuracy: 0.9555\n",
      "Epoch 45/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2701 - teacher_forcing_accuracy: 0.8942 - val_loss: 0.1274 - val_teacher_forcing_accuracy: 0.9556\n",
      "Epoch 46/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2654 - teacher_forcing_accuracy: 0.8965 - val_loss: 0.1256 - val_teacher_forcing_accuracy: 0.9547\n",
      "Epoch 47/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2593 - teacher_forcing_accuracy: 0.8987 - val_loss: 0.1211 - val_teacher_forcing_accuracy: 0.9578\n",
      "Epoch 48/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2517 - teacher_forcing_accuracy: 0.9030 - val_loss: 0.1220 - val_teacher_forcing_accuracy: 0.9562\n",
      "Epoch 49/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2490 - teacher_forcing_accuracy: 0.9034 - val_loss: 0.1172 - val_teacher_forcing_accuracy: 0.9579\n",
      "Epoch 50/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2473 - teacher_forcing_accuracy: 0.9031 - val_loss: 0.1238 - val_teacher_forcing_accuracy: 0.9536\n",
      "Epoch 51/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2448 - teacher_forcing_accuracy: 0.9043 - val_loss: 0.1140 - val_teacher_forcing_accuracy: 0.9603\n",
      "Epoch 52/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2395 - teacher_forcing_accuracy: 0.9080 - val_loss: 0.1127 - val_teacher_forcing_accuracy: 0.9607\n",
      "Epoch 53/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2373 - teacher_forcing_accuracy: 0.9062 - val_loss: 0.1086 - val_teacher_forcing_accuracy: 0.9615\n",
      "Epoch 54/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2395 - teacher_forcing_accuracy: 0.9061 - val_loss: 0.1059 - val_teacher_forcing_accuracy: 0.9630\n",
      "Epoch 55/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2350 - teacher_forcing_accuracy: 0.9086 - val_loss: 0.1135 - val_teacher_forcing_accuracy: 0.9595\n",
      "Epoch 56/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2333 - teacher_forcing_accuracy: 0.9084 - val_loss: 0.1078 - val_teacher_forcing_accuracy: 0.9629\n",
      "Epoch 57/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2293 - teacher_forcing_accuracy: 0.9123 - val_loss: 0.1063 - val_teacher_forcing_accuracy: 0.9618\n",
      "Epoch 58/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2247 - teacher_forcing_accuracy: 0.9132 - val_loss: 0.1045 - val_teacher_forcing_accuracy: 0.9641\n",
      "Epoch 59/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2280 - teacher_forcing_accuracy: 0.9114 - val_loss: 0.1074 - val_teacher_forcing_accuracy: 0.9619\n",
      "Epoch 60/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2256 - teacher_forcing_accuracy: 0.9120 - val_loss: 0.1047 - val_teacher_forcing_accuracy: 0.9616\n",
      "Epoch 61/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2229 - teacher_forcing_accuracy: 0.9152 - val_loss: 0.1073 - val_teacher_forcing_accuracy: 0.9615\n",
      "Epoch 62/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2176 - teacher_forcing_accuracy: 0.9160 - val_loss: 0.1011 - val_teacher_forcing_accuracy: 0.9648\n",
      "Epoch 63/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2164 - teacher_forcing_accuracy: 0.9166 - val_loss: 0.0993 - val_teacher_forcing_accuracy: 0.9649\n",
      "Epoch 64/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2163 - teacher_forcing_accuracy: 0.9174 - val_loss: 0.0976 - val_teacher_forcing_accuracy: 0.9668\n",
      "Epoch 65/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2112 - teacher_forcing_accuracy: 0.9188 - val_loss: 0.0975 - val_teacher_forcing_accuracy: 0.9646\n",
      "Epoch 66/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2129 - teacher_forcing_accuracy: 0.9171 - val_loss: 0.0982 - val_teacher_forcing_accuracy: 0.9645\n",
      "Epoch 67/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2135 - teacher_forcing_accuracy: 0.9178 - val_loss: 0.0952 - val_teacher_forcing_accuracy: 0.9659\n",
      "Epoch 68/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.2057 - teacher_forcing_accuracy: 0.9215 - val_loss: 0.0899 - val_teacher_forcing_accuracy: 0.9693\n",
      "Epoch 69/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1981 - teacher_forcing_accuracy: 0.9247 - val_loss: 0.0940 - val_teacher_forcing_accuracy: 0.9657\n",
      "Epoch 70/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1994 - teacher_forcing_accuracy: 0.9230 - val_loss: 0.0887 - val_teacher_forcing_accuracy: 0.9686\n",
      "Epoch 71/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1956 - teacher_forcing_accuracy: 0.9253 - val_loss: 0.0860 - val_teacher_forcing_accuracy: 0.9699\n",
      "Epoch 72/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1981 - teacher_forcing_accuracy: 0.9246 - val_loss: 0.0878 - val_teacher_forcing_accuracy: 0.9682\n",
      "Epoch 73/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1931 - teacher_forcing_accuracy: 0.9264 - val_loss: 0.0877 - val_teacher_forcing_accuracy: 0.9689\n",
      "Epoch 74/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1946 - teacher_forcing_accuracy: 0.9254 - val_loss: 0.0871 - val_teacher_forcing_accuracy: 0.9683\n",
      "Epoch 75/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1933 - teacher_forcing_accuracy: 0.9257 - val_loss: 0.0887 - val_teacher_forcing_accuracy: 0.9677\n",
      "Epoch 76/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1920 - teacher_forcing_accuracy: 0.9263 - val_loss: 0.0862 - val_teacher_forcing_accuracy: 0.9699\n",
      "Epoch 77/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1919 - teacher_forcing_accuracy: 0.9273 - val_loss: 0.0856 - val_teacher_forcing_accuracy: 0.9687\n",
      "Epoch 78/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1896 - teacher_forcing_accuracy: 0.9282 - val_loss: 0.0839 - val_teacher_forcing_accuracy: 0.9695\n",
      "Epoch 79/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1857 - teacher_forcing_accuracy: 0.9281 - val_loss: 0.0819 - val_teacher_forcing_accuracy: 0.9716\n",
      "Epoch 80/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1854 - teacher_forcing_accuracy: 0.9294 - val_loss: 0.0810 - val_teacher_forcing_accuracy: 0.9717\n",
      "Epoch 81/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1830 - teacher_forcing_accuracy: 0.9303 - val_loss: 0.0860 - val_teacher_forcing_accuracy: 0.9681\n",
      "Epoch 82/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1790 - teacher_forcing_accuracy: 0.9308 - val_loss: 0.0819 - val_teacher_forcing_accuracy: 0.9690\n",
      "Epoch 83/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1820 - teacher_forcing_accuracy: 0.9308 - val_loss: 0.0847 - val_teacher_forcing_accuracy: 0.9696\n",
      "Epoch 84/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1767 - teacher_forcing_accuracy: 0.9322 - val_loss: 0.0807 - val_teacher_forcing_accuracy: 0.9707\n",
      "Epoch 85/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1781 - teacher_forcing_accuracy: 0.9319 - val_loss: 0.0825 - val_teacher_forcing_accuracy: 0.9698\n",
      "Epoch 86/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1774 - teacher_forcing_accuracy: 0.9312 - val_loss: 0.0811 - val_teacher_forcing_accuracy: 0.9708\n",
      "Epoch 87/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1750 - teacher_forcing_accuracy: 0.9316 - val_loss: 0.0784 - val_teacher_forcing_accuracy: 0.9718\n",
      "Epoch 88/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1747 - teacher_forcing_accuracy: 0.9339 - val_loss: 0.0797 - val_teacher_forcing_accuracy: 0.9721\n",
      "Epoch 89/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1737 - teacher_forcing_accuracy: 0.9331 - val_loss: 0.0731 - val_teacher_forcing_accuracy: 0.9751\n",
      "Epoch 90/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1726 - teacher_forcing_accuracy: 0.9345 - val_loss: 0.0792 - val_teacher_forcing_accuracy: 0.9717\n",
      "Epoch 91/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1702 - teacher_forcing_accuracy: 0.9355 - val_loss: 0.0743 - val_teacher_forcing_accuracy: 0.9735\n",
      "Epoch 92/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1692 - teacher_forcing_accuracy: 0.9360 - val_loss: 0.0778 - val_teacher_forcing_accuracy: 0.9721\n",
      "Epoch 93/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1688 - teacher_forcing_accuracy: 0.9343 - val_loss: 0.0750 - val_teacher_forcing_accuracy: 0.9727\n",
      "Epoch 94/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1707 - teacher_forcing_accuracy: 0.9346 - val_loss: 0.0711 - val_teacher_forcing_accuracy: 0.9745\n",
      "Epoch 95/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1675 - teacher_forcing_accuracy: 0.9373 - val_loss: 0.0726 - val_teacher_forcing_accuracy: 0.9739\n",
      "Epoch 96/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1676 - teacher_forcing_accuracy: 0.9357 - val_loss: 0.0759 - val_teacher_forcing_accuracy: 0.9726\n",
      "Epoch 97/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1705 - teacher_forcing_accuracy: 0.9350 - val_loss: 0.0736 - val_teacher_forcing_accuracy: 0.9741\n",
      "Epoch 98/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1723 - teacher_forcing_accuracy: 0.9345 - val_loss: 0.0735 - val_teacher_forcing_accuracy: 0.9744\n",
      "Epoch 99/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1666 - teacher_forcing_accuracy: 0.9357 - val_loss: 0.0740 - val_teacher_forcing_accuracy: 0.9736\n",
      "Epoch 100/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1634 - teacher_forcing_accuracy: 0.9381 - val_loss: 0.0718 - val_teacher_forcing_accuracy: 0.9749\n",
      "Epoch 101/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1647 - teacher_forcing_accuracy: 0.9378 - val_loss: 0.0714 - val_teacher_forcing_accuracy: 0.9740\n",
      "Epoch 102/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1630 - teacher_forcing_accuracy: 0.9388 - val_loss: 0.0680 - val_teacher_forcing_accuracy: 0.9765\n",
      "Epoch 103/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1585 - teacher_forcing_accuracy: 0.9398 - val_loss: 0.0695 - val_teacher_forcing_accuracy: 0.9754\n",
      "Epoch 104/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1622 - teacher_forcing_accuracy: 0.9400 - val_loss: 0.0693 - val_teacher_forcing_accuracy: 0.9753\n",
      "Epoch 105/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1634 - teacher_forcing_accuracy: 0.9375 - val_loss: 0.0700 - val_teacher_forcing_accuracy: 0.9750\n",
      "Epoch 106/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1606 - teacher_forcing_accuracy: 0.9399 - val_loss: 0.0671 - val_teacher_forcing_accuracy: 0.9763\n",
      "Epoch 107/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1548 - teacher_forcing_accuracy: 0.9424 - val_loss: 0.0663 - val_teacher_forcing_accuracy: 0.9754\n",
      "Epoch 108/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1586 - teacher_forcing_accuracy: 0.9411 - val_loss: 0.0661 - val_teacher_forcing_accuracy: 0.9764\n",
      "Epoch 109/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1579 - teacher_forcing_accuracy: 0.9399 - val_loss: 0.0647 - val_teacher_forcing_accuracy: 0.9780\n",
      "Epoch 110/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1566 - teacher_forcing_accuracy: 0.9421 - val_loss: 0.0604 - val_teacher_forcing_accuracy: 0.9798\n",
      "Epoch 111/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1486 - teacher_forcing_accuracy: 0.9447 - val_loss: 0.0606 - val_teacher_forcing_accuracy: 0.9785\n",
      "Epoch 112/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1485 - teacher_forcing_accuracy: 0.9436 - val_loss: 0.0637 - val_teacher_forcing_accuracy: 0.9764\n",
      "Epoch 113/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1486 - teacher_forcing_accuracy: 0.9430 - val_loss: 0.0547 - val_teacher_forcing_accuracy: 0.9819\n",
      "Epoch 114/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1495 - teacher_forcing_accuracy: 0.9433 - val_loss: 0.0596 - val_teacher_forcing_accuracy: 0.9792\n",
      "Epoch 115/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1464 - teacher_forcing_accuracy: 0.9450 - val_loss: 0.0536 - val_teacher_forcing_accuracy: 0.9819\n",
      "Epoch 116/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1397 - teacher_forcing_accuracy: 0.9483 - val_loss: 0.0535 - val_teacher_forcing_accuracy: 0.9820\n",
      "Epoch 117/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1406 - teacher_forcing_accuracy: 0.9463 - val_loss: 0.0634 - val_teacher_forcing_accuracy: 0.9768\n",
      "Epoch 118/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1413 - teacher_forcing_accuracy: 0.9468 - val_loss: 0.0531 - val_teacher_forcing_accuracy: 0.9815\n",
      "Epoch 119/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1423 - teacher_forcing_accuracy: 0.9462 - val_loss: 0.0540 - val_teacher_forcing_accuracy: 0.9813\n",
      "Epoch 120/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1402 - teacher_forcing_accuracy: 0.9472 - val_loss: 0.0529 - val_teacher_forcing_accuracy: 0.9820\n",
      "Epoch 121/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1382 - teacher_forcing_accuracy: 0.9486 - val_loss: 0.0561 - val_teacher_forcing_accuracy: 0.9809\n",
      "Epoch 122/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1351 - teacher_forcing_accuracy: 0.9497 - val_loss: 0.0509 - val_teacher_forcing_accuracy: 0.9826\n",
      "Epoch 123/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1365 - teacher_forcing_accuracy: 0.9490 - val_loss: 0.0515 - val_teacher_forcing_accuracy: 0.9824\n",
      "Epoch 124/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1330 - teacher_forcing_accuracy: 0.9505 - val_loss: 0.0483 - val_teacher_forcing_accuracy: 0.9835\n",
      "Epoch 125/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1314 - teacher_forcing_accuracy: 0.9510 - val_loss: 0.0470 - val_teacher_forcing_accuracy: 0.9844\n",
      "Epoch 126/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1297 - teacher_forcing_accuracy: 0.9517 - val_loss: 0.0560 - val_teacher_forcing_accuracy: 0.9800\n",
      "Epoch 127/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1334 - teacher_forcing_accuracy: 0.9496 - val_loss: 0.0469 - val_teacher_forcing_accuracy: 0.9842\n",
      "Epoch 128/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1328 - teacher_forcing_accuracy: 0.9508 - val_loss: 0.0453 - val_teacher_forcing_accuracy: 0.9852\n",
      "Epoch 129/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1316 - teacher_forcing_accuracy: 0.9511 - val_loss: 0.0432 - val_teacher_forcing_accuracy: 0.9863\n",
      "Epoch 130/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1273 - teacher_forcing_accuracy: 0.9529 - val_loss: 0.0466 - val_teacher_forcing_accuracy: 0.9843\n",
      "Epoch 131/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1269 - teacher_forcing_accuracy: 0.9532 - val_loss: 0.0455 - val_teacher_forcing_accuracy: 0.9846\n",
      "Epoch 132/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1275 - teacher_forcing_accuracy: 0.9516 - val_loss: 0.0450 - val_teacher_forcing_accuracy: 0.9845\n",
      "Epoch 133/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1310 - teacher_forcing_accuracy: 0.9522 - val_loss: 0.0472 - val_teacher_forcing_accuracy: 0.9837\n",
      "Epoch 134/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1249 - teacher_forcing_accuracy: 0.9535 - val_loss: 0.0410 - val_teacher_forcing_accuracy: 0.9866\n",
      "Epoch 135/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1230 - teacher_forcing_accuracy: 0.9546 - val_loss: 0.0388 - val_teacher_forcing_accuracy: 0.9872\n",
      "Epoch 136/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1167 - teacher_forcing_accuracy: 0.9573 - val_loss: 0.0359 - val_teacher_forcing_accuracy: 0.9882\n",
      "Epoch 137/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1171 - teacher_forcing_accuracy: 0.9578 - val_loss: 0.0364 - val_teacher_forcing_accuracy: 0.9883\n",
      "Epoch 138/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1139 - teacher_forcing_accuracy: 0.9577 - val_loss: 0.0351 - val_teacher_forcing_accuracy: 0.9891\n",
      "Epoch 139/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1137 - teacher_forcing_accuracy: 0.9584 - val_loss: 0.0319 - val_teacher_forcing_accuracy: 0.9904\n",
      "Epoch 140/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1089 - teacher_forcing_accuracy: 0.9608 - val_loss: 0.0306 - val_teacher_forcing_accuracy: 0.9906\n",
      "Epoch 141/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1090 - teacher_forcing_accuracy: 0.9604 - val_loss: 0.0319 - val_teacher_forcing_accuracy: 0.9898\n",
      "Epoch 142/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1091 - teacher_forcing_accuracy: 0.9604 - val_loss: 0.0302 - val_teacher_forcing_accuracy: 0.9905\n",
      "Epoch 143/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1056 - teacher_forcing_accuracy: 0.9614 - val_loss: 0.0294 - val_teacher_forcing_accuracy: 0.9909\n",
      "Epoch 144/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1090 - teacher_forcing_accuracy: 0.9602 - val_loss: 0.0331 - val_teacher_forcing_accuracy: 0.9890\n",
      "Epoch 145/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1027 - teacher_forcing_accuracy: 0.9620 - val_loss: 0.0261 - val_teacher_forcing_accuracy: 0.9920\n",
      "Epoch 146/200\n",
      "20/20 [==============================] - 0s 21ms/step - loss: 0.1013 - teacher_forcing_accuracy: 0.9641 - val_loss: 0.0288 - val_teacher_forcing_accuracy: 0.9907\n",
      "Epoch 147/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1032 - teacher_forcing_accuracy: 0.9626 - val_loss: 0.0267 - val_teacher_forcing_accuracy: 0.9918\n",
      "Epoch 148/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.1032 - teacher_forcing_accuracy: 0.9621 - val_loss: 0.0260 - val_teacher_forcing_accuracy: 0.9919\n",
      "Epoch 149/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0973 - teacher_forcing_accuracy: 0.9648 - val_loss: 0.0234 - val_teacher_forcing_accuracy: 0.9928\n",
      "Epoch 150/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0947 - teacher_forcing_accuracy: 0.9664 - val_loss: 0.0266 - val_teacher_forcing_accuracy: 0.9917\n",
      "Epoch 151/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0955 - teacher_forcing_accuracy: 0.9662 - val_loss: 0.0233 - val_teacher_forcing_accuracy: 0.9928\n",
      "Epoch 152/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0938 - teacher_forcing_accuracy: 0.9662 - val_loss: 0.0226 - val_teacher_forcing_accuracy: 0.9930\n",
      "Epoch 153/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0923 - teacher_forcing_accuracy: 0.9672 - val_loss: 0.0211 - val_teacher_forcing_accuracy: 0.9936\n",
      "Epoch 154/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0942 - teacher_forcing_accuracy: 0.9661 - val_loss: 0.0215 - val_teacher_forcing_accuracy: 0.9934\n",
      "Epoch 155/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0929 - teacher_forcing_accuracy: 0.9668 - val_loss: 0.0226 - val_teacher_forcing_accuracy: 0.9932\n",
      "Epoch 156/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0893 - teacher_forcing_accuracy: 0.9677 - val_loss: 0.0221 - val_teacher_forcing_accuracy: 0.9933\n",
      "Epoch 157/200\n",
      "20/20 [==============================] - 0s 21ms/step - loss: 0.0903 - teacher_forcing_accuracy: 0.9673 - val_loss: 0.0219 - val_teacher_forcing_accuracy: 0.9934\n",
      "Epoch 158/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0887 - teacher_forcing_accuracy: 0.9682 - val_loss: 0.0223 - val_teacher_forcing_accuracy: 0.9930\n",
      "Epoch 159/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0900 - teacher_forcing_accuracy: 0.9681 - val_loss: 0.0204 - val_teacher_forcing_accuracy: 0.9935\n",
      "Epoch 160/200\n",
      "20/20 [==============================] - 0s 21ms/step - loss: 0.0859 - teacher_forcing_accuracy: 0.9692 - val_loss: 0.0186 - val_teacher_forcing_accuracy: 0.9944\n",
      "Epoch 161/200\n",
      "20/20 [==============================] - 0s 21ms/step - loss: 0.0849 - teacher_forcing_accuracy: 0.9697 - val_loss: 0.0197 - val_teacher_forcing_accuracy: 0.9937\n",
      "Epoch 162/200\n",
      "20/20 [==============================] - 0s 21ms/step - loss: 0.0867 - teacher_forcing_accuracy: 0.9693 - val_loss: 0.0197 - val_teacher_forcing_accuracy: 0.9943\n",
      "Epoch 163/200\n",
      "20/20 [==============================] - 0s 21ms/step - loss: 0.0854 - teacher_forcing_accuracy: 0.9691 - val_loss: 0.0178 - val_teacher_forcing_accuracy: 0.9945\n",
      "Epoch 164/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0866 - teacher_forcing_accuracy: 0.9696 - val_loss: 0.0195 - val_teacher_forcing_accuracy: 0.9942\n",
      "Epoch 165/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0870 - teacher_forcing_accuracy: 0.9686 - val_loss: 0.0198 - val_teacher_forcing_accuracy: 0.9941\n",
      "Epoch 166/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0847 - teacher_forcing_accuracy: 0.9699 - val_loss: 0.0184 - val_teacher_forcing_accuracy: 0.9944\n",
      "Epoch 167/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0859 - teacher_forcing_accuracy: 0.9701 - val_loss: 0.0195 - val_teacher_forcing_accuracy: 0.9939\n",
      "Epoch 168/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0768 - teacher_forcing_accuracy: 0.9731 - val_loss: 0.0182 - val_teacher_forcing_accuracy: 0.9943\n",
      "Epoch 169/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0770 - teacher_forcing_accuracy: 0.9726 - val_loss: 0.0164 - val_teacher_forcing_accuracy: 0.9950\n",
      "Epoch 170/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0802 - teacher_forcing_accuracy: 0.9723 - val_loss: 0.0182 - val_teacher_forcing_accuracy: 0.9944\n",
      "Epoch 171/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0764 - teacher_forcing_accuracy: 0.9731 - val_loss: 0.0173 - val_teacher_forcing_accuracy: 0.9941\n",
      "Epoch 172/200\n",
      "20/20 [==============================] - 0s 21ms/step - loss: 0.0799 - teacher_forcing_accuracy: 0.9712 - val_loss: 0.0171 - val_teacher_forcing_accuracy: 0.9945\n",
      "Epoch 173/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0784 - teacher_forcing_accuracy: 0.9716 - val_loss: 0.0162 - val_teacher_forcing_accuracy: 0.9953\n",
      "Epoch 174/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0779 - teacher_forcing_accuracy: 0.9724 - val_loss: 0.0168 - val_teacher_forcing_accuracy: 0.9948\n",
      "Epoch 175/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0788 - teacher_forcing_accuracy: 0.9722 - val_loss: 0.0157 - val_teacher_forcing_accuracy: 0.9951\n",
      "Epoch 176/200\n",
      "20/20 [==============================] - 0s 21ms/step - loss: 0.0777 - teacher_forcing_accuracy: 0.9726 - val_loss: 0.0178 - val_teacher_forcing_accuracy: 0.9942\n",
      "Epoch 177/200\n",
      "20/20 [==============================] - 0s 21ms/step - loss: 0.0788 - teacher_forcing_accuracy: 0.9723 - val_loss: 0.0170 - val_teacher_forcing_accuracy: 0.9946\n",
      "Epoch 178/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0752 - teacher_forcing_accuracy: 0.9739 - val_loss: 0.0162 - val_teacher_forcing_accuracy: 0.9948\n",
      "Epoch 179/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0766 - teacher_forcing_accuracy: 0.9731 - val_loss: 0.0158 - val_teacher_forcing_accuracy: 0.9949\n",
      "Epoch 180/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0757 - teacher_forcing_accuracy: 0.9738 - val_loss: 0.0148 - val_teacher_forcing_accuracy: 0.9953\n",
      "Epoch 181/200\n",
      "20/20 [==============================] - 0s 21ms/step - loss: 0.0742 - teacher_forcing_accuracy: 0.9737 - val_loss: 0.0142 - val_teacher_forcing_accuracy: 0.9955\n",
      "Epoch 182/200\n",
      "20/20 [==============================] - 0s 21ms/step - loss: 0.0720 - teacher_forcing_accuracy: 0.9745 - val_loss: 0.0159 - val_teacher_forcing_accuracy: 0.9953\n",
      "Epoch 183/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0741 - teacher_forcing_accuracy: 0.9735 - val_loss: 0.0154 - val_teacher_forcing_accuracy: 0.9954\n",
      "Epoch 184/200\n",
      "20/20 [==============================] - 0s 21ms/step - loss: 0.0742 - teacher_forcing_accuracy: 0.9738 - val_loss: 0.0169 - val_teacher_forcing_accuracy: 0.9948\n",
      "Epoch 185/200\n",
      "20/20 [==============================] - 0s 21ms/step - loss: 0.0712 - teacher_forcing_accuracy: 0.9751 - val_loss: 0.0148 - val_teacher_forcing_accuracy: 0.9955\n",
      "Epoch 186/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0719 - teacher_forcing_accuracy: 0.9748 - val_loss: 0.0142 - val_teacher_forcing_accuracy: 0.9955\n",
      "Epoch 187/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0681 - teacher_forcing_accuracy: 0.9748 - val_loss: 0.0154 - val_teacher_forcing_accuracy: 0.9950\n",
      "Epoch 188/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0726 - teacher_forcing_accuracy: 0.9737 - val_loss: 0.0156 - val_teacher_forcing_accuracy: 0.9953\n",
      "Epoch 189/200\n",
      "20/20 [==============================] - 0s 21ms/step - loss: 0.0713 - teacher_forcing_accuracy: 0.9748 - val_loss: 0.0142 - val_teacher_forcing_accuracy: 0.9955\n",
      "Epoch 190/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0753 - teacher_forcing_accuracy: 0.9734 - val_loss: 0.0152 - val_teacher_forcing_accuracy: 0.9952\n",
      "Epoch 191/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0742 - teacher_forcing_accuracy: 0.9737 - val_loss: 0.0160 - val_teacher_forcing_accuracy: 0.9951\n",
      "Epoch 192/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0717 - teacher_forcing_accuracy: 0.9750 - val_loss: 0.0139 - val_teacher_forcing_accuracy: 0.9955\n",
      "Epoch 193/200\n",
      "20/20 [==============================] - 0s 21ms/step - loss: 0.0713 - teacher_forcing_accuracy: 0.9756 - val_loss: 0.0151 - val_teacher_forcing_accuracy: 0.9952\n",
      "Epoch 194/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0725 - teacher_forcing_accuracy: 0.9741 - val_loss: 0.0154 - val_teacher_forcing_accuracy: 0.9950\n",
      "Epoch 195/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0695 - teacher_forcing_accuracy: 0.9749 - val_loss: 0.0146 - val_teacher_forcing_accuracy: 0.9957\n",
      "Epoch 196/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0688 - teacher_forcing_accuracy: 0.9749 - val_loss: 0.0166 - val_teacher_forcing_accuracy: 0.9945\n",
      "Epoch 197/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0697 - teacher_forcing_accuracy: 0.9755 - val_loss: 0.0146 - val_teacher_forcing_accuracy: 0.9953\n",
      "Epoch 198/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0693 - teacher_forcing_accuracy: 0.9751 - val_loss: 0.0133 - val_teacher_forcing_accuracy: 0.9960\n",
      "Epoch 199/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0710 - teacher_forcing_accuracy: 0.9745 - val_loss: 0.0140 - val_teacher_forcing_accuracy: 0.9955\n",
      "Epoch 200/200\n",
      "20/20 [==============================] - 0s 20ms/step - loss: 0.0681 - teacher_forcing_accuracy: 0.9763 - val_loss: 0.0162 - val_teacher_forcing_accuracy: 0.9948\n"
     ]
    }
   ],
   "source": [
    "train_size = 5000\n",
    "sample_idx = np.random.choice(len(source_train), train_size, replace=False)\n",
    "\n",
    "X_train = source_train[sample_idx], target_train[sample_idx]\n",
    "y_train = labels_train[sample_idx]\n",
    "X_val = source_val, target_val\n",
    "y_val = labels_val\n",
    "\n",
    "history = argsort_model.fit(X_train, y_train, validation_data=(X_val, y_val), verbose=1, epochs=200, batch_size=256)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "element-wise accuracy: 98.92%\n",
      "full sequence accuracy: 95.30%\n",
      "teacher-forcing accuracy:  99.52%\n"
     ]
    }
   ],
   "source": [
    "# evaluate model on test data\n",
    "eval_dict = evaluate_seq2seq_model(argsort_model, source_test, target_test, labels_test, start_token, print_=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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204bOXJkh4qgHgscnoQSiYTLz893t912W5t24MABl5ub6374wx9GMMPkUFlZ6QC4devWOedSd53OOde3b1/33//93ym3xrq6OldUVOTWrl3rpk6d2paEUmWdt9xyizvjjDPM11Jljc45d9NNN7lzzz2Xvh7VWrvlr+N8u7OmAtu3b0dFRUW7NcfjcUydOvWYXnNNTQ0AoF+/fgBSc52tra1YvXo19u/fj0mTJqXcGq+55hpccMEF+OhHP9pOT6V1bt26FQUFBSgsLMTFF1+Mbdu2AUitNT722GMYP348Pv3pT2PgwIEYN24c7rvvvrbXo1prt0xCvt1ZU4FD60qlNTvncOONN+Lcc8/F6NGjAaTWOjdt2oTevXsjHo9j7ty5eOSRRzBq1KiUWuPq1avx4osvorS0NPRaqqxz4sSJePDBB/Hkk0/ivvvuQ0VFBSZPnoy9e/emzBoBYNu2bVi+fDmKiorw5JNPYu7cubj++uvx4IMPAojueHZZAdNk0NHurKlEKq352muvxcsvv4w//OEPoddSYZ2nnnoqNm7ciOrqajz00EO47LLLsG7durbXj/U1lpeX44YbbsCaNWvQo0cPGnesr3PmzJlt/z9mzBhMmjQJJ598MlauXIlzzjkHwLG/RgBIJBIYP348SkpKAADjxo3D5s2bsXz5cnzhC19oi/tnr7Vb3gn5dmdNBQ65cVJlzddddx0ee+wxPP30022tOYDUWmdmZiZOOeUUjB8/HqWlpTjjjDNwxx13pMwaN2zYgMrKSpx11llIT09Heno61q1bh//6r/9Cenp621qO9XUeTq9evTBmzBhs3bo1ZY4lAAwaNAijRo1qp5122mnYuXMngOg+m90yCWVmZuKss87C2rVr2+lr167F5MmTI5pV11JYWIj8/Px2a25qasK6deuOqTU753Dttdfi4Ycfxu9+9zsUFha2ez1V1mnhnENjY2PKrPH888/Hpk2bsHHjxraf8ePH45JLLsHGjRsxfPjwlFjn4TQ2NuLVV1/FoEGDUuZYAsCUKVNCfy7x+uuvt/V5i2ytXWZ56CSHLNr333+/27Jli5s/f77r1auXe+utt6Ke2lFTV1fnXnrpJffSSy85AG7p0qXupZdearOd33bbbS43N9c9/PDDbtOmTe5zn/vcMWcFvfrqq11ubq575pln2lle6+vr22JSYZ0LFy50zz77rNu+fbt7+eWX3c033+xisZhbs2aNcy411mjx9+4451JjnV/5ylfcM88847Zt2+aee+45d+GFF7rs7Oy275pUWKNzB2326enp7jvf+Y7bunWr+/GPf+yysrLcj370o7aYKNbabZOQc87dddddbujQoS4zM9OdeeaZbTbfY5Wnn37a7MN+2WWXOecOWiRvueUWl5+f7+LxuDvvvPPcpk2bop20J9b6ALgHHnigLSYV1nnllVe2nZsnnHCCO//889sSkHOpsUaLw5NQKqzz0N/CZGRkuIKCAjd79my3efPmttdTYY2H+OUvf+lGjx7t4vG4GzlypLv33nvbvR7FWtXKQQghRGR0y2dCQgghjg+UhIQQQkSGkpAQQojIUBISQggRGUpCQgghIkNJSAghRGQoCQkhhIgMJSEhhBCRoSQkhBAiMpSEhBBCRIaSkBBCiMhQEhJCCBEZ/x8s52v31Gy7bwAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# plot SimpleAbstractor's relations after training (to see how much it changed)\n",
    "relational_crossattention = argsort_model.abstractor.abstracter_layers[0].relational_crossattention.mha\n",
    "\n",
    "embedded_objects = argsort_model.source_embedder(np.expand_dims(objects, axis=0))\n",
    "\n",
    "_, attention_scores = relational_crossattention(\n",
    "    query=embedded_objects, key=embedded_objects, value=embedded_objects, return_attention_scores=True)\n",
    "\n",
    "rel_img = np.transpose(attention_scores[0])\n",
    "\n",
    "fig, ax = plt.subplots()\n",
    "im = ax.imshow(rel_img)\n",
    "ax.invert_yaxis()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {},
   "outputs": [],
   "source": [
    "argsort_model.save_weights('prelearning_models/task2_argsort_model.h5')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "313/313 [==============================] - 2s 5ms/step - loss: 0.0162 - teacher_forcing_accuracy: 0.9948\n"
     ]
    }
   ],
   "source": [
    "# reload model and check it was saved correctly\n",
    "argsort_model = reload_argsort_model('prelearning_models/task2_argsort_model.h5', object_dim, seqs_length, argsort_model_kwargs)\n",
    "argsort_model.compile(loss=loss, optimizer=create_opt(), metrics=metrics)\n",
    "argsort_model.evaluate(X_val, y_val);"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3.8.16 64-bit ('relml')",
   "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.16"
  },
  "orig_nbformat": 4,
  "vscode": {
   "interpreter": {
    "hash": "feb2622714ee4f3cfc5c273fa3fe6cf9410db521c7e03d7e619a7b4bef5cf3da"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
