{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import sys\n",
    "sys.path.append('../..'); sys.path.append('../'); \n",
    "\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from setGame import *\n",
    "import tensorflow as tf\n",
    "import utils"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "setgame = SetGame()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 216x72 with 3 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "num_cards=3\n",
    "obs, _, _ = setgame.init_state(num_cards=num_cards, shuffle=False)\n",
    "hand = setgame.state.dealt_cards\n",
    "\n",
    "obsfig, axarr = plt.subplots(1, len(hand), figsize=(len(hand)*1,1))\n",
    "for i in range(len(hand)):\n",
    "    card = hand[i]\n",
    "    axarr[i].imshow(setgame.image_of_card(card[0], card[1]))\n",
    "    axarr[i].axis('off')\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "import tensorflow as tf\n",
    "from tensorflow.keras import layers, Model\n",
    "from sklearn.model_selection import train_test_split\n",
    "\n",
    "class SetEmbedder():\n",
    "    def __init__(self, ff_dim1=128, ff_dim2=32):\n",
    "        img_input = layers.Input(shape=(70, 50, 4))\n",
    "        x = layers.Conv2D(32, (5, 5), activation='relu')(img_input)\n",
    "        x = layers.MaxPooling2D((4,4))(x)\n",
    "        x = layers.Conv2D(32, (5, 5), activation='relu')(x)\n",
    "        x = layers.MaxPooling2D((4,4))(x)\n",
    "        x = layers.Flatten()(x)\n",
    "        x = layers.Dense(ff_dim1, activation='relu')(x)\n",
    "        self.embedding_layer = layers.Dense(ff_dim2, activation='tanh')(x)\n",
    "        outputs = layers.Dense(12, activation='sigmoid')(self.embedding_layer)\n",
    "        self.model = Model(inputs=img_input, outputs=outputs)\n",
    "        self.embed = Model(inputs=img_input, outputs=self.embedding_layer)\n",
    "        self.model.summary()\n",
    "        opt = tf.keras.optimizers.Adam(learning_rate=0.001)\n",
    "        self.model.compile(loss='binary_crossentropy', optimizer=opt, metrics=['binary_accuracy'])\n",
    "        self.model_initial_weights = self.model.get_weights()\n",
    "\n",
    "    def train(self, X_train, y_train, epochs=2):\n",
    "        self.model.set_weights(self.model_initial_weights)\n",
    "        self.model.fit(X_train, y_train, epochs=epochs, batch_size=32, verbose=1)\n",
    "\n",
    "    def predict(self, X_test):\n",
    "        out = self.model.predict(X_test, verbose=0)\n",
    "        return out\n",
    "\n",
    "    def embed(self, X_test):\n",
    "        out = self.embed(X_test, verbose=0)\n",
    "        return out"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def convert_to_binary(attrs):\n",
    "    color = {'red':[1,0,0], 'green':[0,1,0], 'purple':[0,0,1]}\n",
    "    pattern = {'empty':[1,0,0], 'striped':[0,1,0], 'solid':[0,0,1]}\n",
    "    shape = {'diamond':[1,0,0], 'oval':[0,1,0], 'squiggle':[0,0,1]}\n",
    "    number = {'one':[1,0,0], 'two':[0,1,0], 'three':[0,0,1]}\n",
    "    binary_attrs = number[attrs[0]] + color[attrs[1]] + pattern[attrs[2]] + shape[attrs[3]]\n",
    "    return binary_attrs\n",
    "\n",
    "n = 1000\n",
    "X = np.empty((n, 70, 50, 4), dtype=np.float32)\n",
    "y = np.empty((n, 12), dtype=int)\n",
    "\n",
    "card_coord = [(i,j) for i in np.arange(9) for j in np.arange(9)]\n",
    "for i in np.arange(n):\n",
    "    c = np.random.choice(np.arange(81), size=1)[0]\n",
    "    (row, col) = card_coord[c]\n",
    "    attrs = setgame.attributes_of_card(row, col)\n",
    "    binary_attrs = convert_to_binary(attrs)\n",
    "    X[i] = setgame.image_of_card(row, col)\n",
    "    y[i] = binary_attrs\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model: \"model\"\n",
      "_________________________________________________________________\n",
      " Layer (type)                Output Shape              Param #   \n",
      "=================================================================\n",
      " input_1 (InputLayer)        [(None, 70, 50, 4)]       0         \n",
      "                                                                 \n",
      " conv2d (Conv2D)             (None, 66, 46, 32)        3232      \n",
      "                                                                 \n",
      " max_pooling2d (MaxPooling2D  (None, 16, 11, 32)       0         \n",
      " )                                                               \n",
      "                                                                 \n",
      " conv2d_1 (Conv2D)           (None, 12, 7, 32)         25632     \n",
      "                                                                 \n",
      " max_pooling2d_1 (MaxPooling  (None, 3, 1, 32)         0         \n",
      " 2D)                                                             \n",
      "                                                                 \n",
      " flatten (Flatten)           (None, 96)                0         \n",
      "                                                                 \n",
      " dense (Dense)               (None, 64)                6208      \n",
      "                                                                 \n",
      " dense_1 (Dense)             (None, 32)                2080      \n",
      "                                                                 \n",
      " dense_2 (Dense)             (None, 12)                396       \n",
      "                                                                 \n",
      "=================================================================\n",
      "Total params: 37,548\n",
      "Trainable params: 37,548\n",
      "Non-trainable params: 0\n",
      "_________________________________________________________________\n",
      "Epoch 1/20\n",
      "24/24 [==============================] - 2s 35ms/step - loss: 0.6493 - binary_accuracy: 0.6313\n",
      "Epoch 2/20\n",
      "24/24 [==============================] - 1s 36ms/step - loss: 0.5900 - binary_accuracy: 0.6888\n",
      "Epoch 3/20\n",
      "24/24 [==============================] - 1s 35ms/step - loss: 0.5242 - binary_accuracy: 0.7516\n",
      "Epoch 4/20\n",
      "24/24 [==============================] - 1s 35ms/step - loss: 0.4481 - binary_accuracy: 0.8218\n",
      "Epoch 5/20\n",
      "24/24 [==============================] - 1s 37ms/step - loss: 0.3783 - binary_accuracy: 0.8694\n",
      "Epoch 6/20\n",
      "24/24 [==============================] - 1s 38ms/step - loss: 0.3230 - binary_accuracy: 0.9107\n",
      "Epoch 7/20\n",
      "24/24 [==============================] - 1s 36ms/step - loss: 0.2799 - binary_accuracy: 0.9276\n",
      "Epoch 8/20\n",
      "24/24 [==============================] - 1s 36ms/step - loss: 0.2470 - binary_accuracy: 0.9404\n",
      "Epoch 9/20\n",
      "24/24 [==============================] - 1s 36ms/step - loss: 0.2221 - binary_accuracy: 0.9548\n",
      "Epoch 10/20\n",
      "24/24 [==============================] - 1s 36ms/step - loss: 0.1991 - binary_accuracy: 0.9657\n",
      "Epoch 11/20\n",
      "24/24 [==============================] - 1s 44ms/step - loss: 0.1798 - binary_accuracy: 0.9752\n",
      "Epoch 12/20\n",
      "24/24 [==============================] - 1s 38ms/step - loss: 0.1626 - binary_accuracy: 0.9804\n",
      "Epoch 13/20\n",
      "24/24 [==============================] - 1s 36ms/step - loss: 0.1482 - binary_accuracy: 0.9847\n",
      "Epoch 14/20\n",
      "24/24 [==============================] - 1s 36ms/step - loss: 0.1354 - binary_accuracy: 0.9929\n",
      "Epoch 15/20\n",
      "24/24 [==============================] - 1s 54ms/step - loss: 0.1229 - binary_accuracy: 0.9940\n",
      "Epoch 16/20\n",
      "24/24 [==============================] - 1s 38ms/step - loss: 0.1128 - binary_accuracy: 0.9976\n",
      "Epoch 17/20\n",
      "24/24 [==============================] - 1s 36ms/step - loss: 0.1050 - binary_accuracy: 0.9979\n",
      "Epoch 18/20\n",
      "24/24 [==============================] - 1s 36ms/step - loss: 0.0974 - binary_accuracy: 0.9980\n",
      "Epoch 19/20\n",
      "24/24 [==============================] - 1s 41ms/step - loss: 0.0905 - binary_accuracy: 0.9986\n",
      "Epoch 20/20\n",
      "24/24 [==============================] - 1s 44ms/step - loss: 0.0840 - binary_accuracy: 0.9990\n"
     ]
    }
   ],
   "source": [
    "X_train, X_test, y_train, y_test = train_test_split(X, y, test_size=0.25)\n",
    "cnn = SetEmbedder(ff_dim1=64, ff_dim2=32)\n",
    "cnn.train(X_train, y_train, epochs=20)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[0, 0, 1, 1, 0, 0, 1, 0, 0, 1, 0, 0]\n",
      "[0, 0, 1, 1, 0, 0, 1, 0, 0, 1, 0, 0] \n",
      "\n",
      "[1, 0, 0, 0, 1, 0, 0, 1, 0, 0, 0, 1]\n",
      "[1, 0, 0, 0, 1, 0, 0, 1, 0, 0, 0, 1] \n",
      "\n",
      "[0, 1, 0, 0, 0, 1, 1, 0, 0, 0, 1, 0]\n",
      "[0, 1, 0, 0, 0, 1, 1, 0, 0, 0, 1, 0] \n",
      "\n",
      "[0, 1, 0, 0, 0, 1, 0, 1, 0, 0, 1, 0]\n",
      "[0, 1, 0, 0, 0, 1, 0, 1, 0, 0, 1, 0] \n",
      "\n",
      "[0, 0, 1, 1, 0, 0, 0, 0, 1, 0, 1, 0]\n",
      "[0, 0, 1, 1, 0, 0, 0, 0, 1, 0, 1, 0] \n",
      "\n"
     ]
    }
   ],
   "source": [
    "for i in range(5):\n",
    "    j = np.random.choice(range(X_test.shape[0]))\n",
    "    pred = np.round(cnn.predict(X_test[j:(j+1)]))[0]\n",
    "    pred = [int(pred[b]) for b in range(len(pred))]\n",
    "    print(list(pred))\n",
    "    print(list(y_test[j:(j+1)][0]),\"\\n\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "99.77"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "out = cnn.predict(X_test)\n",
    "pred = np.array(np.round(out), dtype=int)\n",
    "np.round(100*(1-np.sum(pred != y_test) / (np.prod(pred.shape))), 2)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "def create_set_classification_dataset(num_seqs):\n",
    "\n",
    "    dim = len(cnn.embed(np.expand_dims(setgame.image_of_card(0, 0), axis=0)).numpy().squeeze())\n",
    "    img_shape = setgame.image_of_card(0, 0).shape\n",
    "    image_seqs = np.zeros((num_seqs, 3) + img_shape)\n",
    "    object_seqs = np.zeros((num_seqs, 3, dim))\n",
    "    card_seqs = np.zeros((num_seqs, 3, 2), dtype=int)\n",
    "    labels = np.zeros(num_seqs, dtype=int)\n",
    "    relations = np.zeros((num_seqs, 12), dtype=int)\n",
    "\n",
    "    for s in np.arange(0, num_seqs, 2):\n",
    "        _ = setgame.init_state(num_cards=6, shuffle=False)\n",
    "        hand = setgame.state.dealt_cards\n",
    "        for i in np.arange(3):\n",
    "            card = hand[i]\n",
    "            row, col = card[0], card[1]\n",
    "            image_seqs[s, i] = setgame.image_of_card(row, col)\n",
    "            object_seqs[s, i] = cnn.embed(np.expand_dims(setgame.image_of_card(row, col), axis=0)).numpy().squeeze()\n",
    "            card_seqs[s, i] = [card[0], card[1]]\n",
    "        labels[s] = 1\n",
    "        for i in np.arange(3):\n",
    "            card = hand[i+3]\n",
    "            row, col = card[0], card[1]\n",
    "            image_seqs[s+1, i] = setgame.image_of_card(row, col)\n",
    "            object_seqs[s+1, i] = cnn.embed(np.expand_dims(setgame.image_of_card(row, col), axis=0)).numpy().squeeze()\n",
    "            card_seqs[s+1, i] = [card[0], card[1]]\n",
    "        labels[s+1] = 0\n",
    "\n",
    "    for s in np.arange(num_seqs):\n",
    "        attrs = [setgame.attributes_of_card(card_seqs[s, k][0], card_seqs[s, k][1]) for k in range(3)]\n",
    "        for k in range(4):\n",
    "            relations[s, 3*k] = int(attrs[0][k]==attrs[1][k])\n",
    "            relations[s, 3*k+1] = int(attrs[0][k]==attrs[2][k])\n",
    "            relations[s, 3*k+2] = int(attrs[1][k]==attrs[2][k])\n",
    "\n",
    "    return image_seqs, card_seqs, object_seqs, labels, relations\n",
    "\n",
    "def create_set_same_different_dataset(num_seqs, attributes):\n",
    "\n",
    "    dim = len(cnn.embed(np.expand_dims(setgame.image_of_card(0, 0), axis=0)).numpy().squeeze())\n",
    "    attr_index = {'number':0, 'color':1, 'pattern':2, 'shape':3}\n",
    "    attr_ind = [attr_index[attr] for attr in attributes]\n",
    "\n",
    "    img_shape = setgame.image_of_card(0, 0).shape\n",
    "    image_seqs = np.zeros((num_seqs, 2) + img_shape)\n",
    "    object_seqs = np.zeros((num_seqs, 2, dim))\n",
    "    card_seqs = np.zeros((num_seqs, 2, 2), dtype=int)\n",
    "    labels = np.zeros((num_seqs, len(attr_ind)), dtype=int)\n",
    "    y_attrs = np.zeros((2, 3), dtype=int)\n",
    "    labels = np.empty((num_seqs, len(attr_ind)), dtype=int)\n",
    "\n",
    "    for s in range(num_seqs):\n",
    "        for j in np.arange(2):\n",
    "            c = np.random.choice(np.arange(81), size=1)[0]\n",
    "            (row, col) = card_coord[c]\n",
    "            image_seqs[s,j] = setgame.image_of_card(row, col)\n",
    "            object_seqs[s,j] = cnn.embed(np.expand_dims(setgame.image_of_card(row, col), axis=0)).numpy().squeeze()\n",
    "            card_seqs[s,j] = [row, col]\n",
    "\n",
    "        for k in range(len(attr_ind)):\n",
    "            attr = attr_ind[k]\n",
    "            for j in np.arange(2):\n",
    "                attrs = setgame.attributes_of_card(card_seqs[s,j][0], card_seqs[s,j][1])\n",
    "                binary_attrs = convert_to_binary(attrs)\n",
    "                y_attrs[j] = binary_attrs[(3*attr):(3*attr+3)]\n",
    "            labels[s,k] = np.dot(y_attrs[0], y_attrs[1])\n",
    "\n",
    "    labels = np.squeeze(labels)\n",
    "    return image_seqs, card_seqs, object_seqs, labels, \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "image_seqs, card_seqs, object_seqs, labels, relations = create_set_classification_dataset(100)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 216x108 with 3 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "SET? 0\n",
      "relations: [0 0 1 0 0 0 1 1 1 1 0 0]\n"
     ]
    }
   ],
   "source": [
    "s = np.random.choice(range(100), size=1)[0]\n",
    "fig, axarr = plt.subplots(1, 3, figsize=(3,1.5))\n",
    "for j in range(3):\n",
    "    axarr[j].imshow(image_seqs[s,j])\n",
    "    axarr[j].axis('off')        \n",
    "plt.show()\n",
    "print('SET?',labels[s])\n",
    "print('relations:', relations[s])\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "attrs = ['number', 'color', 'pattern', 'shape']\n",
    "image_seqs, card_seqs, object_seqs, labels = create_set_same_different_dataset(num_seqs=5000, attributes=attrs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 144x72 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "['number', 'color', 'pattern', 'shape'] : [1 0 0 0]\n"
     ]
    }
   ],
   "source": [
    "i = np.random.choice(range(1000))\n",
    "fig, axarr = plt.subplots(1, 2, figsize=(2,1))\n",
    "for j in range(2):\n",
    "    card = card_seqs[i,j]\n",
    "    axarr[j].imshow(image_seqs[i,j])\n",
    "    axarr[j].axis('off')\n",
    "plt.show()\n",
    "print(attrs,':',labels[i])\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "test_size = 0.2\n",
    "val_size = 0.1\n",
    "\n",
    "seqs_train, seqs_test, object_seqs_train, object_seqs_test, labels_train, labels_test = train_test_split(card_seqs, object_seqs, labels, test_size=0.2)\n",
    "\n",
    "seqs_train, seqs_val, object_seqs_train, object_seqs_val, labels_train, labels_val = \\\n",
    "train_test_split(seqs_train, object_seqs_train, labels_train, test_size=val_size/(1-test_size))\n",
    "\n",
    "source_train, source_val, source_test = object_seqs_train, object_seqs_val, object_seqs_test\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(3500, 4)"
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "source_train.shape\n",
    "labels_train.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "loss = tf.keras.losses.SparseCategoricalCrossentropy(from_logits=True, name='binary_crossentropy')\n",
    "create_opt = lambda : tf.keras.optimizers.Adam()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [],
   "source": [
    "from abstracters import RelationalAbstracter\n",
    "#from seq2seq_transformer import Encoder\n",
    "\n",
    "class AbstractorMultiClassifier(tf.keras.Model):\n",
    "    def __init__(self, num_labels, num_layers, num_heads, dff,\n",
    "            input_vocab, embedding_dim, dropout_rate=0.1, name='abstractor_classifier'):\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.abstractor = RelationalAbstracter(num_layers=num_layers, num_heads=num_heads, dff=dff, \n",
    "          use_pos_embedding=False, dropout_rate=dropout_rate, name='abstractor')\n",
    "        self.flattener = layers.Flatten()\n",
    "        self.hidden_layer = layers.Dense(128, activation='relu', name='hidden_layer')\n",
    "        self.final_layer = layers.Dense(num_labels, activation='sigmoid', name='final_layer')\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(x)\n",
    "        flattened_abstraction = self.flattener(abstracted_context)\n",
    "        hidden_layer = self.hidden_layer(flattened_abstraction)\n",
    "        predictions = self.final_layer(hidden_layer)\n",
    "\n",
    "        try:\n",
    "          # Drop the keras mask, so it doesn't scale the losses/metrics.\n",
    "          # b/250038731\n",
    "          del predictions._keras_mask\n",
    "        except AttributeError:\n",
    "          pass\n",
    "\n",
    "        return predictions\n",
    "        \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model: \"abstractor_classifier\"\n",
      "_________________________________________________________________\n",
      " Layer (type)                Output Shape              Param #   \n",
      "=================================================================\n",
      " source_embedder (TimeDistri  multiple                 2112      \n",
      " buted)                                                          \n",
      "                                                                 \n",
      " abstractor (RelationalAbstr  multiple                 141568    \n",
      " acter)                                                          \n",
      "                                                                 \n",
      " flatten_1 (Flatten)         multiple                  0         \n",
      "                                                                 \n",
      " hidden_layer (Dense)        multiple                  16512     \n",
      "                                                                 \n",
      " final_layer (Dense)         multiple                  516       \n",
      "                                                                 \n",
      "=================================================================\n",
      "Total params: 160,708\n",
      "Trainable params: 160,708\n",
      "Non-trainable params: 0\n",
      "_________________________________________________________________\n"
     ]
    }
   ],
   "source": [
    "abstractor_model = AbstractorMultiClassifier(\n",
    "    num_labels=4, num_layers=1, num_heads=4, dff=64, \n",
    "    input_vocab='vector', embedding_dim=64)\n",
    "\n",
    "abstractor_model.compile(loss='binary_crossentropy', optimizer=create_opt(), metrics=['binary_accuracy'])\n",
    "abstractor_model(object_seqs_train[:128])\n",
    "abstractor_model.summary()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model: \"abstractor_classifier\"\n",
      "_________________________________________________________________\n",
      " Layer (type)                Output Shape              Param #   \n",
      "=================================================================\n",
      " source_embedder (TimeDistri  multiple                 2112      \n",
      " buted)                                                          \n",
      "                                                                 \n",
      " abstractor (RelationalAbstr  multiple                 141568    \n",
      " acter)                                                          \n",
      "                                                                 \n",
      " flatten_1 (Flatten)         multiple                  0         \n",
      "                                                                 \n",
      " hidden_layer (Dense)        multiple                  16512     \n",
      "                                                                 \n",
      " final_layer (Dense)         multiple                  516       \n",
      "                                                                 \n",
      "=================================================================\n",
      "Total params: 160,708\n",
      "Trainable params: 160,708\n",
      "Non-trainable params: 0\n",
      "_________________________________________________________________\n"
     ]
    }
   ],
   "source": [
    "for i in range(abstractor_model.abstractor.num_layers):\n",
    "    abstractor_model.abstractor.abstracter_layers[i].trainable = True\n",
    "\n",
    "abstractor_model.summary()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 1/75\n",
      "32/32 [==============================] - 7s 37ms/step - loss: 0.6627 - binary_accuracy: 0.6488 - val_loss: 0.6645 - val_binary_accuracy: 0.6605\n",
      "Epoch 2/75\n",
      "32/32 [==============================] - 0s 11ms/step - loss: 0.6281 - binary_accuracy: 0.6630 - val_loss: 0.5299 - val_binary_accuracy: 0.7110\n",
      "Epoch 3/75\n",
      "32/32 [==============================] - 0s 11ms/step - loss: 0.3568 - binary_accuracy: 0.8202 - val_loss: 0.1640 - val_binary_accuracy: 0.9385\n",
      "Epoch 4/75\n",
      "32/32 [==============================] - 0s 10ms/step - loss: 0.1941 - binary_accuracy: 0.9125 - val_loss: 0.0984 - val_binary_accuracy: 0.9580\n",
      "Epoch 5/75\n",
      "32/32 [==============================] - 0s 11ms/step - loss: 0.1230 - binary_accuracy: 0.9540 - val_loss: 0.0316 - val_binary_accuracy: 0.9910\n",
      "Epoch 6/75\n",
      "32/32 [==============================] - 0s 11ms/step - loss: 0.0701 - binary_accuracy: 0.9795 - val_loss: 0.0113 - val_binary_accuracy: 0.9995\n",
      "Epoch 7/75\n",
      "32/32 [==============================] - 0s 11ms/step - loss: 0.0604 - binary_accuracy: 0.9808 - val_loss: 0.0088 - val_binary_accuracy: 0.9995\n",
      "Epoch 8/75\n",
      "32/32 [==============================] - 0s 11ms/step - loss: 0.0466 - binary_accuracy: 0.9837 - val_loss: 0.0084 - val_binary_accuracy: 0.9985\n",
      "Epoch 9/75\n",
      "32/32 [==============================] - 0s 10ms/step - loss: 0.0424 - binary_accuracy: 0.9855 - val_loss: 0.0092 - val_binary_accuracy: 0.9985\n",
      "Epoch 10/75\n",
      "32/32 [==============================] - 0s 11ms/step - loss: 0.0339 - binary_accuracy: 0.9875 - val_loss: 0.0065 - val_binary_accuracy: 0.9985\n",
      "Epoch 11/75\n",
      "32/32 [==============================] - 1s 17ms/step - loss: 0.0291 - binary_accuracy: 0.9900 - val_loss: 0.0029 - val_binary_accuracy: 0.9995\n",
      "Epoch 12/75\n",
      "32/32 [==============================] - 1s 16ms/step - loss: 0.0364 - binary_accuracy: 0.9875 - val_loss: 0.0038 - val_binary_accuracy: 0.9995\n",
      "Epoch 13/75\n",
      "32/32 [==============================] - 0s 13ms/step - loss: 0.0316 - binary_accuracy: 0.9887 - val_loss: 0.0024 - val_binary_accuracy: 1.0000\n",
      "Epoch 14/75\n",
      "32/32 [==============================] - 0s 10ms/step - loss: 0.0387 - binary_accuracy: 0.9852 - val_loss: 0.0049 - val_binary_accuracy: 0.9990\n",
      "Epoch 15/75\n",
      "32/32 [==============================] - 0s 10ms/step - loss: 0.0321 - binary_accuracy: 0.9872 - val_loss: 0.0043 - val_binary_accuracy: 0.9995\n",
      "Epoch 16/75\n",
      "32/32 [==============================] - 0s 10ms/step - loss: 0.0439 - binary_accuracy: 0.9847 - val_loss: 0.0093 - val_binary_accuracy: 0.9970\n",
      "Epoch 17/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0262 - binary_accuracy: 0.9918 - val_loss: 0.0039 - val_binary_accuracy: 0.9985\n",
      "Epoch 18/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0219 - binary_accuracy: 0.9912 - val_loss: 0.0071 - val_binary_accuracy: 0.9980\n",
      "Epoch 19/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0242 - binary_accuracy: 0.9915 - val_loss: 0.0029 - val_binary_accuracy: 0.9985\n",
      "Epoch 20/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0200 - binary_accuracy: 0.9923 - val_loss: 9.4614e-04 - val_binary_accuracy: 1.0000\n",
      "Epoch 21/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0221 - binary_accuracy: 0.9908 - val_loss: 0.0013 - val_binary_accuracy: 0.9995\n",
      "Epoch 22/75\n",
      "32/32 [==============================] - 0s 14ms/step - loss: 0.0236 - binary_accuracy: 0.9925 - val_loss: 0.0026 - val_binary_accuracy: 0.9995\n",
      "Epoch 23/75\n",
      "32/32 [==============================] - 0s 10ms/step - loss: 0.0170 - binary_accuracy: 0.9940 - val_loss: 0.0083 - val_binary_accuracy: 0.9985\n",
      "Epoch 24/75\n",
      "32/32 [==============================] - 0s 14ms/step - loss: 0.0225 - binary_accuracy: 0.9898 - val_loss: 7.5391e-04 - val_binary_accuracy: 1.0000\n",
      "Epoch 25/75\n",
      "32/32 [==============================] - 0s 10ms/step - loss: 0.0185 - binary_accuracy: 0.9925 - val_loss: 0.0027 - val_binary_accuracy: 0.9995\n",
      "Epoch 26/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0187 - binary_accuracy: 0.9937 - val_loss: 0.0069 - val_binary_accuracy: 0.9985\n",
      "Epoch 27/75\n",
      "32/32 [==============================] - 0s 10ms/step - loss: 0.0136 - binary_accuracy: 0.9952 - val_loss: 0.0042 - val_binary_accuracy: 0.9990\n",
      "Epoch 28/75\n",
      "32/32 [==============================] - 0s 11ms/step - loss: 0.0156 - binary_accuracy: 0.9942 - val_loss: 0.0089 - val_binary_accuracy: 0.9965\n",
      "Epoch 29/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0196 - binary_accuracy: 0.9923 - val_loss: 0.0030 - val_binary_accuracy: 0.9990\n",
      "Epoch 30/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0219 - binary_accuracy: 0.9930 - val_loss: 0.0020 - val_binary_accuracy: 0.9990\n",
      "Epoch 31/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0189 - binary_accuracy: 0.9933 - val_loss: 0.0038 - val_binary_accuracy: 0.9990\n",
      "Epoch 32/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0220 - binary_accuracy: 0.9923 - val_loss: 9.2510e-04 - val_binary_accuracy: 1.0000\n",
      "Epoch 33/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0150 - binary_accuracy: 0.9952 - val_loss: 0.0021 - val_binary_accuracy: 0.9995\n",
      "Epoch 34/75\n",
      "32/32 [==============================] - 0s 10ms/step - loss: 0.0195 - binary_accuracy: 0.9923 - val_loss: 7.5774e-04 - val_binary_accuracy: 1.0000\n",
      "Epoch 35/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0160 - binary_accuracy: 0.9935 - val_loss: 0.0013 - val_binary_accuracy: 0.9995\n",
      "Epoch 36/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0119 - binary_accuracy: 0.9952 - val_loss: 2.4307e-04 - val_binary_accuracy: 1.0000\n",
      "Epoch 37/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0125 - binary_accuracy: 0.9950 - val_loss: 0.0011 - val_binary_accuracy: 1.0000\n",
      "Epoch 38/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0147 - binary_accuracy: 0.9950 - val_loss: 0.0017 - val_binary_accuracy: 0.9995\n",
      "Epoch 39/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0160 - binary_accuracy: 0.9958 - val_loss: 4.4039e-04 - val_binary_accuracy: 1.0000\n",
      "Epoch 40/75\n",
      "32/32 [==============================] - 0s 11ms/step - loss: 0.0146 - binary_accuracy: 0.9952 - val_loss: 0.0073 - val_binary_accuracy: 0.9970\n",
      "Epoch 41/75\n",
      "32/32 [==============================] - 0s 13ms/step - loss: 0.0193 - binary_accuracy: 0.9930 - val_loss: 0.0058 - val_binary_accuracy: 0.9980\n",
      "Epoch 42/75\n",
      "32/32 [==============================] - 0s 8ms/step - loss: 0.0184 - binary_accuracy: 0.9937 - val_loss: 0.0189 - val_binary_accuracy: 0.9930\n",
      "Epoch 43/75\n",
      "32/32 [==============================] - 0s 8ms/step - loss: 0.0185 - binary_accuracy: 0.9927 - val_loss: 0.0081 - val_binary_accuracy: 0.9975\n",
      "Epoch 44/75\n",
      "32/32 [==============================] - 0s 8ms/step - loss: 0.0141 - binary_accuracy: 0.9945 - val_loss: 0.0054 - val_binary_accuracy: 0.9980\n",
      "Epoch 45/75\n",
      "32/32 [==============================] - 0s 8ms/step - loss: 0.0168 - binary_accuracy: 0.9927 - val_loss: 0.0022 - val_binary_accuracy: 0.9985\n",
      "Epoch 46/75\n",
      "32/32 [==============================] - 0s 8ms/step - loss: 0.0155 - binary_accuracy: 0.9942 - val_loss: 0.0014 - val_binary_accuracy: 0.9995\n",
      "Epoch 47/75\n",
      "32/32 [==============================] - 0s 8ms/step - loss: 0.0174 - binary_accuracy: 0.9937 - val_loss: 0.0046 - val_binary_accuracy: 0.9975\n",
      "Epoch 48/75\n",
      "32/32 [==============================] - 0s 8ms/step - loss: 0.0265 - binary_accuracy: 0.9902 - val_loss: 0.0042 - val_binary_accuracy: 0.9975\n",
      "Epoch 49/75\n",
      "32/32 [==============================] - 0s 14ms/step - loss: 0.0215 - binary_accuracy: 0.9927 - val_loss: 0.0126 - val_binary_accuracy: 0.9980\n",
      "Epoch 50/75\n",
      "32/32 [==============================] - 0s 11ms/step - loss: 0.0312 - binary_accuracy: 0.9875 - val_loss: 0.0026 - val_binary_accuracy: 0.9990\n",
      "Epoch 51/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0192 - binary_accuracy: 0.9925 - val_loss: 0.0036 - val_binary_accuracy: 0.9990\n",
      "Epoch 52/75\n",
      "32/32 [==============================] - 0s 10ms/step - loss: 0.0211 - binary_accuracy: 0.9918 - val_loss: 6.4412e-04 - val_binary_accuracy: 1.0000\n",
      "Epoch 53/75\n",
      "32/32 [==============================] - 0s 12ms/step - loss: 0.0220 - binary_accuracy: 0.9918 - val_loss: 9.0088e-04 - val_binary_accuracy: 1.0000\n",
      "Epoch 54/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0231 - binary_accuracy: 0.9920 - val_loss: 0.0019 - val_binary_accuracy: 1.0000\n",
      "Epoch 55/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0139 - binary_accuracy: 0.9945 - val_loss: 0.0015 - val_binary_accuracy: 1.0000\n",
      "Epoch 56/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0141 - binary_accuracy: 0.9940 - val_loss: 0.0031 - val_binary_accuracy: 0.9990\n",
      "Epoch 57/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0144 - binary_accuracy: 0.9935 - val_loss: 0.0036 - val_binary_accuracy: 0.9985\n",
      "Epoch 58/75\n",
      "32/32 [==============================] - 0s 11ms/step - loss: 0.0129 - binary_accuracy: 0.9948 - val_loss: 0.0019 - val_binary_accuracy: 0.9995\n",
      "Epoch 59/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0139 - binary_accuracy: 0.9945 - val_loss: 0.0022 - val_binary_accuracy: 0.9990\n",
      "Epoch 60/75\n",
      "32/32 [==============================] - 0s 10ms/step - loss: 0.0160 - binary_accuracy: 0.9927 - val_loss: 0.0045 - val_binary_accuracy: 0.9985\n",
      "Epoch 61/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0117 - binary_accuracy: 0.9950 - val_loss: 0.0087 - val_binary_accuracy: 0.9985\n",
      "Epoch 62/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0189 - binary_accuracy: 0.9927 - val_loss: 0.0086 - val_binary_accuracy: 0.9965\n",
      "Epoch 63/75\n",
      "32/32 [==============================] - 0s 10ms/step - loss: 0.0165 - binary_accuracy: 0.9935 - val_loss: 0.0027 - val_binary_accuracy: 0.9985\n",
      "Epoch 64/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0282 - binary_accuracy: 0.9895 - val_loss: 0.0037 - val_binary_accuracy: 0.9990\n",
      "Epoch 65/75\n",
      "32/32 [==============================] - 0s 10ms/step - loss: 0.0302 - binary_accuracy: 0.9880 - val_loss: 0.0035 - val_binary_accuracy: 0.9985\n",
      "Epoch 66/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0242 - binary_accuracy: 0.9935 - val_loss: 0.0025 - val_binary_accuracy: 0.9990\n",
      "Epoch 67/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0205 - binary_accuracy: 0.9923 - val_loss: 0.0014 - val_binary_accuracy: 1.0000\n",
      "Epoch 68/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0123 - binary_accuracy: 0.9940 - val_loss: 0.0015 - val_binary_accuracy: 0.9995\n",
      "Epoch 69/75\n",
      "32/32 [==============================] - 0s 8ms/step - loss: 0.0192 - binary_accuracy: 0.9937 - val_loss: 0.0016 - val_binary_accuracy: 0.9995\n",
      "Epoch 70/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0147 - binary_accuracy: 0.9942 - val_loss: 6.6226e-04 - val_binary_accuracy: 1.0000\n",
      "Epoch 71/75\n",
      "32/32 [==============================] - 0s 8ms/step - loss: 0.0128 - binary_accuracy: 0.9945 - val_loss: 4.6040e-04 - val_binary_accuracy: 1.0000\n",
      "Epoch 72/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0117 - binary_accuracy: 0.9955 - val_loss: 9.2312e-04 - val_binary_accuracy: 1.0000\n",
      "Epoch 73/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0117 - binary_accuracy: 0.9952 - val_loss: 0.0017 - val_binary_accuracy: 0.9995\n",
      "Epoch 74/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0151 - binary_accuracy: 0.9942 - val_loss: 0.0062 - val_binary_accuracy: 0.9980\n",
      "Epoch 75/75\n",
      "32/32 [==============================] - 0s 9ms/step - loss: 0.0163 - binary_accuracy: 0.9942 - val_loss: 0.0015 - val_binary_accuracy: 0.9995\n"
     ]
    }
   ],
   "source": [
    "train_size = 1000\n",
    "X_train = source_train[:train_size]\n",
    "y_train = labels_train[:train_size]\n",
    "X_val = source_val\n",
    "y_val = labels_val\n",
    "history = abstractor_model.fit(X_train, y_train, validation_data=(X_val, y_val), epochs=75, verbose=1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "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', 'binary_accuracy'));\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "99.9"
      ]
     },
     "execution_count": 22,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "out = abstractor_model.predict(source_test, verbose=0)\n",
    "pred = np.array(np.round(out), dtype=int)\n",
    "np.round(100*(1-np.sum(pred != labels_test) / (np.prod(pred.shape))), 2)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [],
   "source": [
    "def create_callbacks(patience=6):\n",
    "    callbacks = []\n",
    "    callbacks.append(tf.keras.callbacks.EarlyStopping(monitor='val_loss', \n",
    "        patience=patience, min_delta=0, start_from_epoch=20, mode='auto', restore_best_weights=True, verbose=True))\n",
    "    return callbacks\n",
    "    \n",
    "class VectorEmbedder(tf.keras.Model):\n",
    "    def __init__(self, embedding_dim=64, name='vector_embedder'):\n",
    "        super().__init__(name=name)\n",
    "        self.embedding_dim = embedding_dim\n",
    "        self.embedder = layers.Dense(self.embedding_dim, name='dense')\n",
    "\n",
    "    def call(self, inputs):\n",
    "        outputs = self.embedder(inputs)\n",
    "        return outputs\n",
    "\n",
    "class SimpleAbstractor(tf.keras.Model):\n",
    "    def __init__(self, sequence_len, embedding_dim, symbol_dim, embedder=None, name='simplest_abstractor_vectorial'):\n",
    "        super().__init__(name=name)\n",
    "        self.sequence_len = sequence_len\n",
    "        self.embedding_dim = embedding_dim\n",
    "        self.symbol_dim = symbol_dim\n",
    "\n",
    "        if embedder != None:\n",
    "            self.source_embedder = layers.TimeDistributed(embedder, name='source_embedder')\n",
    "        else:\n",
    "            self.source_embedder = layers.TimeDistributed(layers.Dense(embedding_dim), name='source_embedder')\n",
    "        self.source_embedder.trainable = False\n",
    "\n",
    "        normal_initializer = tf.keras.initializers.RandomNormal(mean=0., stddev=1., seed=11)\n",
    "        self.symbols = tf.Variable(\n",
    "            normal_initializer(shape=(self.sequence_len, self.symbol_dim)),\n",
    "            name='symbols', trainable=False)\n",
    "        self.scale = 50\n",
    "        self.flatten = layers.Flatten()\n",
    "        self.query_projection = layers.Dense(embedding_dim, activation=None, name='query_projection')\n",
    "        self.key_projection = layers.Dense(embedding_dim, activation=None, name='key_projection')\n",
    "        self.hidden_layer = layers.Dense(32, activation='relu', name='hidden_layer')\n",
    "        self.final_layer = layers.Dense(1, activation='sigmoid', name='final_layer')\n",
    "\n",
    "    def call(self, inputs):\n",
    "        source = inputs\n",
    "        E = self.source_embedder(source)\n",
    "        Q = self.query_projection(E)\n",
    "        K = self.key_projection(E)\n",
    "        self.Z = self.scale * tf.einsum('ijk,ilk->ijl', Q, K) / self.embedding_dim\n",
    "        self.R = tf.math.sigmoid(self.Z) # tf.nn.softmax(self.Z, axis=1)\n",
    "        self.A = tf.einsum('ijk,kl->ijl', self.R, self.symbols)\n",
    "        flattened_symbols = self.flatten(self.A)\n",
    "        hidden_units = self.hidden_layer(flattened_symbols)\n",
    "        output = self.final_layer(hidden_units)\n",
    "        return output"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "((1000, 2, 32), (500, 2, 32), (1000, 2, 32))"
      ]
     },
     "execution_count": 24,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "train_size = 1000\n",
    "X_train = source_train[:train_size]\n",
    "y_train = labels_train[:train_size]\n",
    "X_val = source_val\n",
    "y_val = labels_val\n",
    "X_train.shape, X_val.shape\n",
    "X_test = source_test\n",
    "y_test = labels_test\n",
    "X_train.shape, X_val.shape, X_test.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [],
   "source": [
    "embedder = VectorEmbedder(embedding_dim=64)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "test accuracy of attribute 0 (number): 99.30%\n",
      "test accuracy of attribute 1 (color): 100.00%\n",
      "test accuracy of attribute 2 (pattern): 99.60%\n",
      "test accuracy of attribute 3 (shape): 99.30%\n"
     ]
    }
   ],
   "source": [
    "abstractor = [None, None, None, None]\n",
    "\n",
    "for i in range(4):\n",
    "    abstractor[i] = SimpleAbstractor(sequence_len=2, symbol_dim=10, embedding_dim=64, embedder=embedder)\n",
    "    abstractor[i].compile(loss='binary_crossentropy', optimizer=create_opt(), metrics=['binary_accuracy'])\n",
    "    abstractor[i](X_train[:32])\n",
    "    #abstractor.summary()\n",
    "    history = abstractor[i].fit(X_train, y_train[:,i], validation_data=(X_val, y_val[:,i]), \\\n",
    "                    callbacks=create_callbacks(), epochs=20, verbose=0, batch_size=32)\n",
    "    out = abstractor[i](X_test)\n",
    "    yhat = np.array([int(o) for o in np.round(np.squeeze(out.numpy()))])\n",
    "    print('test accuracy of attribute %d (%s): %.2f%%' % (i, attrs[i], 100*np.mean(yhat==y_test[:,i])))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {},
   "outputs": [],
   "source": [
    "image_seqs, card_seqs, object_seqs, labels, relations = create_set_classification_dataset(5000)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "((5000, 3, 70, 50, 4), (5000, 3, 32), (5000, 12), (5000,))"
      ]
     },
     "execution_count": 28,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "\n",
    "Z_set = image_seqs\n",
    "X_set = object_seqs\n",
    "R_set = relations\n",
    "y_set = labels\n",
    "\n",
    "Z_set.shape, X_set.shape, R_set.shape, y_set.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 216x108 with 3 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "SET? 1\n",
      "SET? [1 1 1 0 0 0 0 0 0 0 0 0]\n"
     ]
    }
   ],
   "source": [
    "i = np.random.choice(range(len(Z_set)))\n",
    "fig, axarr = plt.subplots(1, 3, figsize=(3,1.5))\n",
    "for j in range(3):\n",
    "    axarr[j].imshow(Z_set[i,j])\n",
    "    axarr[j].axis('off')\n",
    "plt.show()\n",
    "print('SET?',y_set[i])\n",
    "print('SET?',R_set[i])\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "metadata": {},
   "outputs": [],
   "source": [
    "class SimpleAbstractorMH(tf.keras.Model):\n",
    "    def __init__(self, sequence_len, embedding_dim, symbol_dim, embedder=None, num_heads=4, name='simplest_abstractor_vectorial_mh', h1=128, h2=64):\n",
    "        super().__init__(name=name)\n",
    "        self.sequence_len = sequence_len\n",
    "        self.embedding_dim = embedding_dim\n",
    "        self.symbol_dim = symbol_dim\n",
    "        self.num_heads = num_heads\n",
    "        self.h1, self.h2 = (h1, h2)\n",
    "\n",
    "        if embedder != None:\n",
    "            self.source_embedder = layers.TimeDistributed(embedder, name='source_embedder')\n",
    "        else:\n",
    "            self.source_embedder = layers.TimeDistributed(layers.Dense(embedding_dim), name='source_embedder')\n",
    "        self.source_embedder.trainable = False\n",
    "\n",
    "        normal_initializer = tf.keras.initializers.RandomNormal(mean=0., stddev=1., seed=11)\n",
    "        #self.symbols = tf.matmul(tf.ones((self.sequence_len,1)),\n",
    "        #    tf.Variable(normal_initializer(shape=(1, self.symbol_dim)),\n",
    "        #    name='symbols', trainable=False))\n",
    "        self.symbols = tf.Variable(normal_initializer(shape=(self.sequence_len, self.symbol_dim)),\n",
    "            name='symbols', trainable=False)\n",
    "        self.scale = 50\n",
    "        self.flatten = layers.Flatten()\n",
    "        \n",
    "        \n",
    "        self.query_projection = [None]*self.num_heads\n",
    "        self.key_projection = [None]*self.num_heads\n",
    "\n",
    "        for h in range(self.num_heads):\n",
    "            self.query_projection[h] = layers.Dense(embedding_dim, activation=None, name='query_projection')\n",
    "            self.key_projection[h] = layers.Dense(embedding_dim, activation=None, name='key_projection')\n",
    "\n",
    "        self.hidden_layer1 = layers.Dense(self.h1, activation='relu', name='hidden_layer1')\n",
    "        self.hidden_layer2 = layers.Dense(self.h2, activation='relu', name='hidden_layer2')\n",
    "        self.final_layer = layers.Dense(1, activation='sigmoid', name='final_layer')\n",
    "\n",
    "    def call(self, inputs):\n",
    "        source = inputs\n",
    "        E = self.source_embedder(source)\n",
    "\n",
    "        self.Z = [None]*self.num_heads \n",
    "        self.R = [None]*self.num_heads \n",
    "        self.A = [None]*self.num_heads \n",
    "\n",
    "        for h in range(self.num_heads):\n",
    "            Q = self.query_projection[h](E)\n",
    "            K = self.key_projection[h](E)\n",
    "            self.Z[h] = self.scale * tf.einsum('ijk,ilk->ijl', Q, K) / self.embedding_dim\n",
    "            #self.Z[h] = tf.math.sigmoid(self.Z[h])\n",
    "            self.R[h] = tf.math.sigmoid(self.Z[h]) #tf.nn.softmax(5*self.Z[h], axis=1)\n",
    "            self.A[h] = tf.einsum('ijk,kl->ijl', self.R[h], self.symbols)\n",
    "            if h==0:\n",
    "                A = tf.concat([self.A[h]], axis=1)\n",
    "            else:\n",
    "                A = tf.concat([A, self.A[h]], axis=1)\n",
    "\n",
    "        flattened_symbols = self.flatten(A)\n",
    "        hidden_units = self.hidden_layer1(flattened_symbols)\n",
    "        hidden_units = self.hidden_layer2(hidden_units)\n",
    "        output = self.final_layer(hidden_units)\n",
    "        return output"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "((500, 3, 32), (125, 3, 32), (500, 3, 32))"
      ]
     },
     "execution_count": 31,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "train_size = 500\n",
    "val_frac = .2\n",
    "\n",
    "X_train, X_test, y_train, y_test, Z_train, Z_test = train_test_split(X_set, y_set, Z_set, train_size=int(train_size/(1-val_frac)), test_size=500)\n",
    "X_train, X_dev, y_train, y_dev, Z_train, Z_dev = train_test_split(X_train, y_train, Z_train, train_size=1-val_frac)\n",
    "X_train.shape, X_dev.shape, X_test.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model: \"simplest_abstractor_vectorial_mh\"\n",
      "_________________________________________________________________\n",
      " Layer (type)                Output Shape              Param #   \n",
      "=================================================================\n",
      " source_embedder (TimeDistri  multiple                 2112      \n",
      " buted)                                                          \n",
      "                                                                 \n",
      " flatten_6 (Flatten)         multiple                  0         \n",
      "                                                                 \n",
      " query_projection (Dense)    multiple                  4160      \n",
      "                                                                 \n",
      " query_projection (Dense)    multiple                  4160      \n",
      "                                                                 \n",
      " query_projection (Dense)    multiple                  4160      \n",
      "                                                                 \n",
      " query_projection (Dense)    multiple                  4160      \n",
      "                                                                 \n",
      " key_projection (Dense)      multiple                  4160      \n",
      "                                                                 \n",
      " key_projection (Dense)      multiple                  4160      \n",
      "                                                                 \n",
      " key_projection (Dense)      multiple                  4160      \n",
      "                                                                 \n",
      " key_projection (Dense)      multiple                  4160      \n",
      "                                                                 \n",
      " hidden_layer1 (Dense)       multiple                  23168     \n",
      "                                                                 \n",
      " hidden_layer2 (Dense)       multiple                  8256      \n",
      "                                                                 \n",
      " final_layer (Dense)         multiple                  65        \n",
      "                                                                 \n",
      "=================================================================\n",
      "Total params: 66,926\n",
      "Trainable params: 31,489\n",
      "Non-trainable params: 35,437\n",
      "_________________________________________________________________\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "50"
      ]
     },
     "execution_count": 32,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "abstractor_mh = SimpleAbstractorMH(sequence_len=3, symbol_dim=15, embedding_dim=64, embedder=embedder, num_heads=4)\n",
    "abstractor_mh(X_set[:32])\n",
    "abstractor_mh.compile(loss='binary_crossentropy', optimizer=create_opt(), metrics=['binary_accuracy'])\n",
    "\n",
    "use_pretrained = True\n",
    "if (use_pretrained):\n",
    "    for h in range(4):\n",
    "        abstractor_mh.query_projection[h].set_weights(abstractor[h].query_projection.get_weights())\n",
    "        abstractor_mh.query_projection[h].trainable=False\n",
    "        abstractor_mh.key_projection[h].set_weights(abstractor[h].key_projection.get_weights())\n",
    "        abstractor_mh.key_projection[h].trainable=False\n",
    "\n",
    "abstractor_mh.summary()\n",
    "abstractor_mh.scale = 50\n",
    "abstractor_mh.scale"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "metadata": {},
   "outputs": [],
   "source": [
    "def create_callbacks(patience=6):\n",
    "    callbacks = []\n",
    "    callbacks.append(tf.keras.callbacks.EarlyStopping(monitor='val_loss', \n",
    "        patience=patience, min_delta=0, start_from_epoch=20, mode='auto', restore_best_weights=True, verbose=False))\n",
    "    return callbacks\n",
    "\n",
    "def initialize_set_abstractor_model(use_pretrained_relations=True):\n",
    "    abstractor_mh = SimpleAbstractorMH(sequence_len=3, symbol_dim=15, embedding_dim=64, embedder=embedder, num_heads=4)\n",
    "    abstractor_mh(X_set[:32])\n",
    "    abstractor_mh.compile(loss='binary_crossentropy', optimizer=create_opt(), metrics=['binary_accuracy'])\n",
    "\n",
    "    if (use_pretrained_relations):\n",
    "        for h in range(4):\n",
    "            abstractor_mh.query_projection[h].set_weights(abstractor[h].query_projection.get_weights())\n",
    "            abstractor_mh.query_projection[h].trainable=False\n",
    "            abstractor_mh.key_projection[h].set_weights(abstractor[h].key_projection.get_weights())\n",
    "            abstractor_mh.key_projection[h].trainable=False\n",
    "\n",
    "    #abstractor_mh.summary()\n",
    "    abstractor_mh.scale = 50\n",
    "    return abstractor_mh\n",
    "\n",
    "def set_train_val_test_split(train_size, val_size, test_size):\n",
    "    X_train, X_test, y_train, y_test = \\\n",
    "        train_test_split(X_set, y_set, train_size=int(train_size/(1-val_frac)), test_size=test_size)\n",
    "    X_train, X_dev, y_train, y_dev = train_test_split(X_train, y_train, train_size=1-val_frac)\n",
    "    return X_train, y_train, X_dev, y_dev, X_test, y_test\n",
    "\n",
    "def train_set_abstractor_models(train_sizes, num_trials, num_epochs, pretrained=True):\n",
    "    from tqdm import tqdm\n",
    "    from sklearn.model_selection import train_test_split\n",
    "    accs = list()\n",
    "    n_val = 100\n",
    "    for n in tqdm(train_sizes):\n",
    "        trial_accs = []\n",
    "        for trial in np.arange(num_trials):\n",
    "            model = initialize_set_abstractor_model(use_pretrained_relations=pretrained)\n",
    "            X_train, y_train, X_val, y_val, X_test, y_test = \\\n",
    "                set_train_val_test_split(train_size=n, val_size=n_val, test_size=500)\n",
    "            model.fit(X_train, y_train, validation_data=(X_val, y_val), callbacks=create_callbacks(), epochs=num_epochs, verbose=0)\n",
    "            out = model(X_test)\n",
    "            yhat = np.array([int(o) for o in np.round(np.squeeze(out.numpy()))])\n",
    "            acc = 100*np.mean(yhat==y_test)\n",
    "            trial_accs.append(acc)\n",
    "        accs.append(trial_accs)\n",
    "        print('n=%d, accuracy=%.2f' % (n, np.mean(trial_accs, axis=0)))\n",
    "    return accs\n",
    "\n",
    "def plot_comparison(pretrained_n, pretrained_accs, scratch_n, scratch_accs):\n",
    "    import scipy.stats\n",
    "    scratch_accuracy = np.mean(scratch_accs, axis=1)\n",
    "    scratch_acc_sem = scipy.stats.sem(scratch_accs, axis=1)\n",
    "    pretrained_accuracy = np.mean(pretrained_accs, axis=1)\n",
    "    pretrained_acc_sem = scipy.stats.sem(pretrained_accs, axis=1)\n",
    "    fig, ax = plt.subplots(figsize=(6,4))\n",
    "    ax.plot(scratch_n, scratch_accuracy, label='Abstractor trained de novo')\n",
    "    ax.fill_between(scratch_n, scratch_accuracy - scratch_acc_sem,\n",
    "        scratch_accuracy + scratch_acc_sem, alpha=0.5)\n",
    "    ax.plot(pretrained_n, pretrained_accuracy, label='Using pretrained relations')\n",
    "    ax.fill_between(pretrained_n, pretrained_accuracy - pretrained_acc_sem,\n",
    "        pretrained_accuracy + pretrained_acc_sem, alpha=0.5)\n",
    "    ax.set_xlabel('Training Set Size')\n",
    "    ax.legend()\n",
    "    ax.set_ylabel('SET Classification Accuracy')\n",
    "    ax.grid(linestyle='dashed')\n",
    "    #fig.savefig('set_accuracies.pdf')\n",
    "    plt.show()\n",
    "\n",
    "def plot_accuracies(pretrained_n, pretrained_accs):\n",
    "    import scipy.stats\n",
    "    pretrained_accuracy = np.mean(pretrained_accs, axis=1)\n",
    "    pretrained_acc_sem = scipy.stats.sem(pretrained_accs, axis=1)\n",
    "    fig, ax = plt.subplots(figsize=(6,4))\n",
    "    ax.plot(pretrained_n, pretrained_accuracy, label='Using pretrained relations')\n",
    "    ax.fill_between(pretrained_n, pretrained_accuracy - pretrained_acc_sem,\n",
    "        pretrained_accuracy + pretrained_acc_sem, alpha=0.3)\n",
    "    ax.set_xlabel('Training Set Size')\n",
    "    ax.set_ylabel('SET Classification Accuracy')\n",
    "    ax.grid(linestyle='dashed')\n",
    "    plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  7%|▋         | 1/14 [00:37<08:12, 37.90s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=20, accuracy=56.68\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 14%|█▍        | 2/14 [01:20<07:51, 39.30s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=40, accuracy=58.02\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 21%|██▏       | 3/14 [01:57<07:04, 38.63s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=60, accuracy=60.90\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 29%|██▊       | 4/14 [02:34<06:20, 38.08s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=80, accuracy=63.62\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 36%|███▌      | 5/14 [03:12<05:42, 38.02s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=100, accuracy=71.16\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 43%|████▎     | 6/14 [03:54<05:15, 39.40s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=200, accuracy=86.76\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 50%|█████     | 7/14 [04:37<04:41, 40.26s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=300, accuracy=92.86\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 57%|█████▋    | 8/14 [05:24<04:14, 42.34s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=400, accuracy=96.04\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 64%|██████▍   | 9/14 [06:18<03:50, 46.01s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=500, accuracy=97.34\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 71%|███████▏  | 10/14 [07:18<03:20, 50.17s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=600, accuracy=97.46\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 79%|███████▊  | 11/14 [08:13<02:34, 51.53s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=700, accuracy=97.30\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 86%|████████▌ | 12/14 [09:05<01:43, 51.82s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=800, accuracy=97.74\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 93%|█████████▎| 13/14 [10:08<00:55, 55.15s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=900, accuracy=98.46\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 14/14 [11:14<00:00, 48.15s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=1000, accuracy=98.16\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "train_sizes = list(np.arange(20, 100, 20)) + list(np.arange(100, 1001, 100))\n",
    "accs = train_set_abstractor_models(train_sizes=train_sizes, num_trials=10, num_epochs=50)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plot_accuracies(train_sizes, accs)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "metadata": {},
   "outputs": [],
   "source": [
    "class SymbolicClassifier(tf.keras.Model):\n",
    "    def __init__(self, sequence_len, h1=16, h2=16, name='symbolic_classifier'):\n",
    "        super().__init__(name=name)\n",
    "        self.sequence_len = sequence_len\n",
    "        self.h1, self.h2 = (h1,h2)\n",
    "        self.hidden_layer1 = layers.Dense(self.h1, activation='relu', name='hidden_layer1')\n",
    "        self.hidden_layer2 = layers.Dense(self.h2, activation='relu', name='hidden_layer2')\n",
    "        self.final_layer = layers.Dense(1, activation='sigmoid', name='final_layer')\n",
    "\n",
    "    def call(self, inputs):\n",
    "        source = inputs\n",
    "        hidden_units = self.hidden_layer1(source)\n",
    "        hidden_units = self.hidden_layer2(hidden_units)\n",
    "        output = self.final_layer(hidden_units)\n",
    "        return output"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model: \"symbolic_classifier\"\n",
      "_________________________________________________________________\n",
      " Layer (type)                Output Shape              Param #   \n",
      "=================================================================\n",
      " hidden_layer1 (Dense)       multiple                  208       \n",
      "                                                                 \n",
      " hidden_layer2 (Dense)       multiple                  272       \n",
      "                                                                 \n",
      " final_layer (Dense)         multiple                  17        \n",
      "                                                                 \n",
      "=================================================================\n",
      "Total params: 497\n",
      "Trainable params: 497\n",
      "Non-trainable params: 0\n",
      "_________________________________________________________________\n"
     ]
    }
   ],
   "source": [
    "symbolic_model = SymbolicClassifier(3)\n",
    "symbolic_model(R_set[:32])\n",
    "symbolic_model.compile(loss='binary_crossentropy', optimizer=create_opt(), metrics=['binary_accuracy'])\n",
    "symbolic_model.summary()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "metadata": {},
   "outputs": [],
   "source": [
    "def create_callbacks(patience=6):\n",
    "    callbacks = []\n",
    "    callbacks.append(tf.keras.callbacks.EarlyStopping(monitor='val_loss', \n",
    "        patience=patience, min_delta=0, start_from_epoch=20, mode='auto', restore_best_weights=True, verbose=False))\n",
    "    return callbacks\n",
    "\n",
    "def initialize_set_symbolic_model():\n",
    "    model = SymbolicClassifier(sequence_len=3)\n",
    "    model(R_set[:32])\n",
    "    model.compile(loss='binary_crossentropy', optimizer=create_opt(), metrics=['binary_accuracy'])\n",
    "    return model\n",
    "\n",
    "def set_symbolic_train_val_test_split(train_size, val_size, test_size):\n",
    "    from sklearn.model_selection import train_test_split\n",
    "    R_train, R_test, y_train, y_test = \\\n",
    "        train_test_split(R_set, y_set, train_size=int(train_size/(1-val_frac)), test_size=test_size)\n",
    "    R_train, R_dev, y_train, y_dev = train_test_split(R_train, y_train, train_size=1-val_frac)\n",
    "    return R_train, y_train, R_dev, y_dev, R_test, y_test\n",
    "\n",
    "def train_set_symbolic_models(train_sizes, num_trials, num_epochs, pretrained=True):\n",
    "    from tqdm import tqdm\n",
    "    accs = list()\n",
    "    n_val = 100\n",
    "    for n in tqdm(train_sizes):\n",
    "        trial_accs = []\n",
    "        for trial in np.arange(num_trials):\n",
    "            model = initialize_set_symbolic_model()\n",
    "            R_train, y_train, R_val, y_val, R_test, y_test = \\\n",
    "                set_symbolic_train_val_test_split(train_size=n, val_size=n_val, test_size=500)\n",
    "            model.fit(R_train, y_train, validation_data=(R_val, y_val), callbacks=create_callbacks(), epochs=num_epochs, verbose=0)\n",
    "            out = model(R_test)\n",
    "            yhat = np.array([int(o) for o in np.round(np.squeeze(out.numpy()))])\n",
    "            acc = 100*np.mean(yhat==y_test)\n",
    "            trial_accs.append(acc)\n",
    "        accs.append(trial_accs)\n",
    "        print('n=%d, accuracy=%.2f' % (n, np.mean(trial_accs, axis=0)))\n",
    "    return accs\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  7%|▋         | 1/14 [00:20<04:25, 20.42s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=10, accuracy=52.14\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 14%|█▍        | 2/14 [00:38<03:57, 19.81s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=20, accuracy=55.56\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 21%|██▏       | 3/14 [00:59<03:41, 20.15s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=30, accuracy=57.00\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 29%|██▊       | 4/14 [01:20<03:24, 20.45s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=40, accuracy=61.14\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 36%|███▌      | 5/14 [01:45<03:14, 21.61s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=50, accuracy=66.72\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 43%|████▎     | 6/14 [02:12<03:06, 23.29s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=60, accuracy=64.12\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 50%|█████     | 7/14 [02:36<02:44, 23.52s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=70, accuracy=70.00\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 57%|█████▋    | 8/14 [03:00<02:22, 23.68s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=80, accuracy=70.84\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 64%|██████▍   | 9/14 [03:24<01:59, 23.82s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=90, accuracy=79.82\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 71%|███████▏  | 10/14 [03:50<01:37, 24.31s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=100, accuracy=75.04\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 79%|███████▊  | 11/14 [04:20<01:18, 26.01s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=300, accuracy=97.24\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 86%|████████▌ | 12/14 [04:56<00:58, 29.07s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=500, accuracy=98.56\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 93%|█████████▎| 13/14 [05:36<00:32, 32.38s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=700, accuracy=99.20\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 14/14 [06:20<00:00, 27.20s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "n=900, accuracy=99.34\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "symbolic_train_sizes = list(np.arange(10, 100, 10)) + list(np.arange(100, 1001, 200))\n",
    "symbolic_accs = train_set_symbolic_models(train_sizes=symbolic_train_sizes, num_trials=10, num_epochs=50)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 40,
   "metadata": {},
   "outputs": [],
   "source": [
    "def plot_symbolic_comparison(pretrained_n, pretrained_accs, scratch_n, scratch_accs):\n",
    "    import scipy.stats\n",
    "    scratch_accuracy = np.mean(scratch_accs, axis=1)\n",
    "    scratch_acc_sem = scipy.stats.sem(scratch_accs, axis=1)\n",
    "    pretrained_accuracy = np.mean(pretrained_accs, axis=1)\n",
    "    pretrained_acc_sem = scipy.stats.sem(pretrained_accs, axis=1)\n",
    "    fig, ax = plt.subplots(figsize=(6,4))\n",
    "    ax.plot(scratch_n, scratch_accuracy, label='MLP with purely symbolic input')\n",
    "    ax.fill_between(scratch_n, scratch_accuracy - 2*scratch_acc_sem,\n",
    "        scratch_accuracy + 2*scratch_acc_sem, alpha=0.5)\n",
    "    ax.plot(pretrained_n, pretrained_accuracy, label='Abstractor on images, pre-learned relations')\n",
    "    ax.fill_between(pretrained_n, pretrained_accuracy - 2*pretrained_acc_sem,\n",
    "        pretrained_accuracy + 2*pretrained_acc_sem, alpha=0.5)\n",
    "    ax.set_xlabel('Training Set Size')\n",
    "    ax.legend()\n",
    "    ax.set_ylabel('SET Classification Accuracy')\n",
    "    ax.grid(linestyle='dashed')\n",
    "    fig.savefig('set_symbolic_vs_abstractor.pdf')\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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0eoOJtm6WJuleWeszTqPypkVo5HMESdWt9/vp6amhq6urlFMkf5MmjSJvXv3jnYyzjpG1G1EzZCc7pSUFIqLi7Hbkx8taSijEAwG447PhEce7WyEo94M/qlwDw5LiIa0ubRNvgC7dWKsVhqru6amhoyMDEpKSsZd38hg8Pl8OByO0U7GWceIuo2oGQbWLSI0NjZSU1MzKOd5E6PUS5KIP3WAUEho7vARDAl/rVGc42jnoox6WlKKOZJ7MXnpE+dHFqu7q6uLvLy8CW0QgKhffKNhRN1G1AwD61ZKkZeXN+hWAUMZhViaOzWDUNfaRbNXcUPuB3Q58jg4aQ0oC/np48Mt9lCY6AbBxMREYyj/dUMZhaKiouh+ZNLaqVZtVuC56X725l9J0KLVEPImkFGI1W0UBtOGOpEwom4jagb9dBvKKBQXF0f327v8AJxs7WJSCtROWYPflhY9P5Gaj2J1jwWUUnzyk5+MHgcCAfLz87n22msBeOyxx/jCF77Q67qSkhIWLlxIeXk5a9eu7XdIXl9trV//+tf561//CsBPf/pTOjs7o+fS09OHpGekWL16NYNZSTARw2lb37hxYzQPkqWkpIQzZ84AcOGFFyZ93X/913/x29/+dlDP6otYzY899hgnT54ckfuOdfTqRzGUUYh1ICWidcScavVQkJOJx9E99NRqUeSkThyjMNYchqWlpbFr166o75bXXnst6drMhg0bqK6uprKyku985zt9xuvo6EgY/q1vfYs1a9YAvY3C2aDnYIeRpi/dZ4O33nor6bif+9zn+NSnPjUiz43VbCSjoFdeG2r0UU9aOv10+UNMyU6JC89JtWO1TPx292/+3272nGwb0XsumJrJ/f9QOmC8q666ipdeeomPfOQjrF+/nltuuYW//e1vST9n5cqV/PznP48Le+edd/je977HM888w0svvcTtt99Oa2sroVCIBQsWcOTIEdatW8e1117LyZMnOXnyJJdccgmTJk1iw4YNAHz1q1/lxRdfxOVy8fzzz1NYWBj3jG984xscPnyY2tpaTpw4wb//+7/zmc98ho0bN/KjH/2IF198EYAvfOELVFZWsm7dOkpKSrjjjjt49dVX+cIXvkBubi73338/Xq+XWbNm8eijj8bVUh555BF27drFT37yEwAefvhh9u7dy4MPdi9pEgwG+fSnP822bdtQSnHHHXdw3XXX8eEPf5gdO3YAcPDgQW6++Wbee+89SkpK+PjHP86GDRvw+/089NBDfOUrX+HQoUPcc889fO5znwOgra2NG2+8kf3797Ny5Up+9atfYbFYWL9+Pd/5zncQEa655hq+//3v98qT9PR03G43AD/4wQ/43e9+h8Vi4aqrruJ73/ter+8xPT2df/u3f2P16tUsW7aMDRs20NLSwiOPPMLFF1/MY489xrPPPovX6+Xo0aN8/OMf5/777+fYsWNce+217Nq1C4Cf//zn+P1+ysrK2LZtG7feeisul4stW7YYdq2F4WComkLP5oGT4f6EqVnxP5yJ1J8Ao98skoibb76ZP/zhD3R1dfH+++8P2jf8iy++yMKFC+PCKioq2L59O6AtVVhWVsa7777L1q1be93/i1/8IlOnTmXDhg1Rg9DR0cEFF1xAdXU1K1eu5OGHH0747Pfff5+XXnqJLVu28K1vfSupN9OUlBQ2b97MmjVreOCBB/jrX/9KVVUVlZWVcYU9aN/NCy+8gN+vNXE++uij3H777XFxduzYQW1tLbt27WLnzp3cfvvtzJo1i8zMzKhRePTRR1m3bl30mmnTprFlyxYuvvhi1q1bx9NPP83bb7/N17/+9Wicd955hx//+Mfs3LmTw4cP88wzz3Dy5Enuvfde3njjDXbs2MG7777Lc88916fWV155heeee46tW7dSXV3Nv//7vw/4/QQCAd555x1++tOf8s1vfjMuPU888QQ7duzgqaeeSti8FulM/chHPkJlZWU0/kQ3CBaLPsW3oWoKlZXxi72dau0ixWYhJzW+wyYvbeI0HUFv3RGSeaPXi0WLFnHs2DHWr1/P1VdfnfR1l1xyCVarlUWLFvHAAw/EnbPZbJx77rns3buX7du38+Uvf5k333yTYDDIxRdf3Mcdu3E4HNE29fPOO4/XXnstYbzrr78el8uFy+Xikksu4Z133iE7O7vfe3/sYx8D4O2332bPnj1cdNFFgDbWvKdjs7S0NC699FJefPFF5s+fj9/v72UAZ86cyZEjR/jnf/5nrrnmGtauXQvAZz/7WR599FEefPBB/vjHP/LOO+9Er7nuuusAWLhwIW63m4yMDDIyMkhJSaGlpQWApUuXMnOmthb5LbfcwubNm7Hb7axevZr8/HwAbr31Vt58801uuOGGhFr/+te/cvvtt5OamgpA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UhYvXsyiRYsGbHIsKSnh3nvvZenSpSxdujTqtmPdunV8+ctf5pJLLuHee+/l8OHDXHnllZx33nlcfPHFcTXDvvjf//3faFo++9nPRgvgnlojzJs3L5rvTz31FCUlJdx///1UVFSwcOHC6DP70u7xeLj55ptZtGgRH/vYx6LrfCTSHPucV199leXLl1NRUcFNN90UdRseS6I0//znP4+6bb/kkkui947UlB588EHKysooKyvjpz/9KQDHjh1j/vz5fOYzn6G0tJS1a9dGje7Pf/5zFixYwKJFi7j55psH/H4HRETO+gc4BkzqEfYD4L7w/n3A9we6z3nnnSeDYcOGDdH9d482yk9fOyCd3sCg7jEeidW9Z8+e7hMv3yvym6tH9vPyvQOm53e/+53ccccdIiKyfPlyee+996LpdDqdcvjwYQkEArJmzRp56qmnZNu2bbJmzZro9c3NzSIismrVKnn33Xej4eecc458//vfFxGRtrY2OXPmTPTcV7/6Vfn5z38uIiIf/ehH5Sc/+YmIiAQCAWlpaZGjR49KaWlpNP7TTz8ta9askUAgIHV1dTJt2jQ5efKkbNiwQVJTU+XIkSO9dNXW1sq0adOkoaFB/H6/XHLJJfLss8+KiAggL7zwgoiI3HPPPfLtb3+71/WPPvqofP7znxcRkdtuu00+9rGPSSgUkueee04yMjLk/fffl2AwKBUVFbJ9+3YREWlsbIzqWLVqlbz11lvi8XikuLg4msabb75ZrrnmGhER+cpXviK/+93vot/j7Nmzxe12yxe+8AX53//9XxER8Xq90tnZ2UfudX/XDzzwgIiIPP7449H733bbbXLNNddIIKD9ry699FI5cOCAiIi8/fbbcskll/R5v9OnT8uePXvk2muvFZ/PJyIid911lzz++OMJtVZXV4uIyPTp06P5HrlXJK9/+ctfyqc//el+tf/4xz+W22+/XUREqqurxWq1xv2uYu8bec7p06fl4osvFrfbLSIi3/ve9+Sb3/ymiMT/LvtKc0RvT/3btm2TsrIycbvd0t7eLgsWLJCqqio5evSoWK3WaL7fdNNN8tBDD4mIyJQpU6Srqyuqqydx//kwwDbpo1wdS+0m1wOrw/uPAxuBe/uKPBJMz3Phclj1fMTY5qrRab9ev349//Iv/wJoawesX7+eiooKILHr5ssuuyyhm+hERFxU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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plot_symbolic_comparison(train_sizes, accs, symbolic_train_sizes, symbolic_accs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "metadata": {},
   "outputs": [],
   "source": [
    "np.savez('symbolic_vs_abstractor', train_sizes=train_sizes, accs=accs, symbolic_train_sizes=symbolic_train_sizes, symbolic_accs=symbolic_accs)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 43,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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2c9+NrodDAbJysrihhtuAOD555/nS1/60nnnFRYWsnDhQkpLS7nqqqv6nZLXV1/rN7/5Tf72t78B8KMf/Yiurq7od8nJyUPSM1KsWrWKwUyc6I3h9K1v2LAhWgfxUlhYyNmzZwG4+OKL4z7vf/7nf/jVr341qGv1Razm559/njNnzoxIueMdrcZRdOUUegsg1djhZV8LnPQ5ufCCPMwmwZRk2xhYpx3jLWBYUlISe/fujcZueeedd+Juzaxfv56qqirKy8v57ne/2+dxnZ2dve7/zne+wxVXKHm1ezqF0aCvyQ4jRV+6R4P3338/7mPvu+8+Pve5z43IdWM168kpaFXXuuo+6o36Di9v11rJTDIzJ9cJQOYkcwp98e0/7WP/mY4RLXPB1BQe+VjxgMdde+21vPnmm3ziE59g7dq13Hnnnfz973+P+zorVqzg6aef7rbvww8/5PHHH+eVV17hzTff5O6776a9vZ1wOMyCBQs4fvw4a9as4YYbbuDMmTOcOXOG1atXM2XKFNavXw/A17/+dd544w0cDgevv/46OTk53a7xrW99i2PHjlFbW8vp06f593//d77whS+wYcMGnnzySd544w0AvvSlL1FeXs6aNWsoLCzknnvu4e233+ZLX/oSGRkZPPLII/h8PmbPns1zzz3XrZXy7LPPsnfvXn74wx8C8POf/5wDBw7w1FNPRY8JhUJ8/vOfZ9u2bQghuOeee7jxxhu59dZb2bVrFwBHjhzhjjvuYPv27RQWFvKpT32K9evXEwgEeOaZZ/ja177G0aNHeeihh7jvvvsA6Ojo4JZbbuHQoUOsWLGCn/70p5hMJtauXct3v/tdpJRcf/31fP/73dK2A0pLK5Is54knnuDXv/41JpOJa6+9lscf7x7f8lvf+hbJycn827/9G6tWraKiooL169fT1tbGs88+y6WXXsrzzz/Pq6++is/no7q6mk996lM88sgjnDhxghtuuCGavOfpp58mEAhQUlLCtm3b+PSnP43D4WDLli3RJDwG8aOrlkLP7oFQWLL/TAetnhDlhemY1EGZydZ9NNbdIr1xxx138Lvf/Q6v18vu3bsHHRv+jTfeYOHChd32lZWVsXPnTkBJVVhSUsJHH33UazKSBx54gKlTp7J+/fqoQ+js7OSiiy6iqqqKFStW8POf/7zXa+/evZs333yTLVu28J3vfCeuJ1O73c7mzZu54ooreOyxx/jb3/7Gjh07KC8v73azB+W3WbduHYFAAIDnnnuOu+++u9sxu3btora2lr1797Jnzx7uvvtuZs+eTUpKStQpPPfcc6xZsyZ6zrRp09iyZQuXXnopa9as4eWXX+aDDz7gm9/8ZvSYDz/8kB/84Afs2bOHY8eO8corr3DmzBm++tWv8t5777Fr1y4++ugjXnvttT61vvXWW7z22mts3bqVqqoq/v3f/33A3ycYDPLhhx/yox/9iG9/+9vd7HnxxRfZtWsXL730Uq/da5HB1E984hOUl5dHj5/sDsFk0ub2rauWQs80dM2dPmrblC6MaelKSGyb1YTTPrkGrvpKvxfPE71WLFq0iBMnTrB27Vquu+66uM9bvXo1ZrOZRYsW8dhjj3X7zmKxcMEFF3DgwAF27tzJV77yFTZt2kQoFOLSSy8dsOyEhIRon/rSpUt55513ej3upptuwuFw4HA4WL16NR9++CFpaWn9lv3JT34SgA8++ID9+/dzySVKXg6/339eYLOkpCQuu+wy3njjDebPn08gEDjPAc6aNYvjx4/zL//yL1x//fVcddVVAHzxi1/kueee46mnnuL3v/89H374YfScG2+8EYCFCxfidrtxOp04nU7sdjttbW0ALFu2jFmzZgFw5513snnzZqxWK6tWrSIrKwuAT3/602zatImbb765V61/+9vfuPvuu0lMVP6nMjIGTr/y8Y9/HFB+90hKUIArr7ySzMzM6DGbN28+77oJCQn4/f4BrzHZSEpK0qRcXTmFLVu2dPsHbHL5qG/34rRbSLIpP8WUpMnXddRT93jhxhtv5N/+7d/YsGFDNI/tQKxfv54pU6b0+f2ll17KW2+9hclk4oorrmDNmjWEQiGefPLJAcu2Wq3Rp06z2RzNtduTntP8hBBYLBbC4XB0X8+54ZF/YCklV155JWvXru3Xln/6p3/iu9/9LvPmzTuvlQBKPt+qqir++te/8pOf/IQ//OEP/PKXv+Tqq6/m29/+NpdddhlLly6N3lDhXO5qk8kU/RzZjmjtTdtgox5E8gsPhog9PX/3eH7rjo4OzZ6axzNut1uTXgBd/ZI9F3s0uXzUtXvJS7VH9y0sSB1tszRnvC7uueeee/jmN7953lPwcFixYgU/+tGPuPDCC8nKyqK5uZmDBw9SXHx+q8jpdOJyuQZ9jddffx2v10tzczMbNmzgwgsvZMaMGezfvx+fz0d7ezvvvvtur+dedNFF/OMf/+Do0aMAdHV1cfjw+eHEKioqOH36NL/97W+58847z/v+7NmzhMNhbr31Vh599NFo8nebzcbVV1/N/fff36szGYgPP/yQ6upqwuEwv//971m+fDkVFRVs3LiRs2fPEgqFWLt2LStXruyzjKuuuopf/vKX0UH8lpaWPo8diHfeeYeWlhY8Hg+vvfYal1xyCTk5OTQ2NtLc3IzP5+Ott96KHj/UOp2IaBWiSFcthZ4cbnDh9gXJTVGcQm6qnXnqYLOB9hQUFPDlL3+51++ef/75bv3WH3xwXmzGXqmoqKChoSHaPbNo0SKys7N7fXK99957ufbaa8nLy4uOK8TDsmXLuP766zl16hTf+MY3mDp1KgC33347ixYtoqioiCVLlvR6blZWFs8//zx33nln1Fk/9thjzJkz57xjb7/9dnbt2kV6evp539XW1nL33XdHn5i/973vRb/79Kc/zSuvvBLtUhoMlZWVPPzww+zZs4cVK1Zwyy23YDKZ+N73vsfq1auRUnLddddx00039VnGNddcw65duygvLychIYHrrruu35li/bF8+XI++9nPcvToUT71qU9Fu0K/+c1vUlFRwcyZM7v9dmvWrOG+++4zBpqHga4C4gWDQSwWxQ9KKfny73axruoMnyyfRm6qndsvnEZ+2uT7I4rVfeDAAebPn7i5IeJlKF0Y8RA7a0ZrbrjhBh588EEuvzz+hIRSSn7wgx/Q3t7Oo48+qqF12vP888+zbds2/vu//7vf47Sq6/FOvLp7+5/vLyCerrqPYpvpHZ4gp1u7MJsEWU4bRTnJk9IhAL12T0x2JnIU2La2NubMmYPD4RiUQwBlEPxXv/pVny2wychEruvhoJVuXXUfNTY2smDBAuWzy0t9u5dspw2bxcSlF2SNsXXaEatbL/Q1SDxcvvWtb2lSbixpaWlDduQvvvgiTufk6AJds2ZNtym1faFVXY93tNKtq5ZCLM2dPho7fOSl2pmfl0Jq4uSahmpgYGAwFHTlFEpKSqKfjzd1EpKS3FQ7hVO0me87XojVrRfsdvvAB01C9Khbj5pBO926cgqxUzMPNyjL8fPTEilIn5xjCRHG65RULZnIEyiGgx5161EzaKdbV07hyJEj0c+HG1w47RZmZyVht07uVH6xuvWCHh0h6FO3HjWDdrp15RRiOdzgIi/FzvSMxLE2RZe8+uqrCCE4ePBgdN9gQzePZETMtrY2fvrTn45IWUNl3bp15wWO0wuxIbjHmuGGMI/n/J4Req+77rpoqJGxRldOYdq0aQDUt3s56/aTm2pnmg6cQkT3eGLt2rUsX76c3/3ud0Muoz+nMNiwB0NxCiMdBvvGG2/k4YcfHlYZ4ynhzGjNCorVrHVo8pG6Tk+n8Oc//3nA+Fk9MZLsjACRMMhn3T6mZyQyLcPB1Em6NiGWnuGfo7z1MDx3/ci+3hr4puZ2u/nHP/7Bs88+e55TiIRuXrBgAffddx/hcJhQKMSaNWsoKSlh4cKF/PCHP+Tll1+OhklevHgxHo+HwsJCvvOd77B8+XJef/11fv7zn3PhhRdSWlrKrbfeGv0nbGho4JZbbqG0tJTS0lLef/99Hn74YY4dO8bixYt56KGHkFLy0EMPRa/5+9//HlBaM6tXr+ZTn/pUr+E51q5dy8KFCykpKeGrX/1qdH9ycjJf//rXKS0t5aKLLqKhoeG8c2OTC61Zs4b777+f1atXM2vWLDZu3Mg999zD/Pnzu03TvP/++ykvL6e4uJhHHnkkeqP485//zLx581i+fDkPPPBAtAXW2dnJPffcw4UXXsiSJUt4/fXXAdi3bx/Lli1j8eLFLFq0aMAux8LCQr761a+ybNkyli1bFg3bsWbNGr7yla+wevVqvvrVr3Ls2DGuueYali5dyqWXXtqtZdgXv/nNb6K2fPGLX4zegHtqjTBv3rxovb/00ksUFhbyyCOPUFZWxsKFC6PX7Eu7x+PhjjvuYNGiRXzyk5+M5vnoTXPsdd5++20qKyspKyvjtttui4YNj6U3m59++ulo2PbVq1dHy460lJ566ilKSkooKSnhRz/6EQAnTpxg/vz5fOELX6C4uJirrroq6nSffvppFixYwKJFi7jjjjsG/H0HREo5YV9Lly6Vg2H9+vXRzx9VN8tXdpwe1PkTlVjd+/fvP/fFn78q5S+vG9nXn786oD2//vWv5T333COllLKyslJu3749aqfNZpPHjh2TwWBQXnHFFfKll16S27Ztk1dccUX0/NbWVimllCtXrpQfffRRdP+MGTPk97//fSmllB0dHfLs2bPR777+9a/Lp59+Wkop5e233y5/+MMfSimlDAaDsq2tTVZXV8vi4uLo8S+//LK84oorZDAYlPX19XLatGnyzJkzcv369TIxMVEeP378PF21tbVy2rRpsrGxUQYCAbl69Wr56quvSimlBOS6deuklFI+9NBD8tFHHz3v/Oeee07+8z//s5RSyrvuukt+8pOflOFwWL722mvS6XTK3bt3y1AoJMvKyuTOnTullFI2NzdHdaxcuVK+//770uPxyIKCgqiNd9xxh7z++uullFJ+7Wtfk7/+9a+jv2NRUZF0u93yS1/6kvzNb34jpZTS5/PJrq6uPmrv3G/92GOPSSmlfOGFF6Ll33XXXfL666+XwWBQSinlZZddJg8fPiyllPKDDz6Qq1ev7rO8pqYmuX//fnnDDTdIv98vpZTy/vvvly+88EKvWquqqqSUUk6fPj1a75GyInX9k5/8RH7+85/vV/sPfvADeffdd0sppayqqpJms7nb31VsuZHrNDU1yUsvvVS63W4ppZSPP/64/Pa3vy2l7P532ZfNEb099W/btk2WlJRIt9stXS6XXLBggdyxY4esrq6WZrM5Wu+33XabfOaZZ6SUUubl5Umv1xvV1ZNu//MqwDbZx31VV4vXeqL78YRrx6b/eu3atfzrv/4roOQOWLt2LWVlZUDvoZsvv/zyXsNE90YkRDXA3r17+Y//+A/a2tpwu91cffXVALz33nvRVJBms5nU1FRaW1u7lbN582buvPNOzGYzOTk5rFy5ko8++oiUlBSWLVvGzJkzz7v2Rx991GeI6XjDcsfysY99DCEECxcuJCcnJ9oyKS4u5sSJEyxevJg//OEPPPPMMwSDQerq6jh48CAOh4NZs2ZFbbzzzjt55plnAHj77bdZt25dNGqs1+vl1KlTVFZW8p//+Z/U1NTw8Y9/nKKiogHtiwTqu/POO3nwwQej+2+77TbMZjNut5v333+f2267LfrdQIOj7777Ltu3b+fCCy8ElKf47OxsgPO07t+/n0WLFgHd6x26h+J+5ZVX+tW+adMmHnjgAUCJlRUpszcGEwJ9IJt7Y/Pmzdxyyy3RqLof//jH+fvf/86NN97IzJkzWbx4cVTXqVOnojZ/+tOf5uabb+4znPlg0JVTSElJ6bath/EEOF/3WNLc3Mx7773H3r17EUIQCoUQQvDEE08AvYdK7itMdG9E/plMJhNr1qzhtddeo7S0lOeff54NGzbEbafsZ7pfX3Hs+zsn3rDcsQwU6rq6uponn3ySjz76iPT0dNasWYPf7+/XDiklf/zjH5k7d263/fPnz6eiooI333yTq6++ml/84hdcdtll/doXW1exnyO/TzgcJi0tLZr0J0IoFGLp0qWAMo7yne98p5t9d911V7cAf0CvWmPDPPSsk95Ccfelvaf9/TGYEOgD2dwb/dVd7N+A2WyOdqu9+eabbNq0iXXr1vHoo4+yb9++aKyzoaCrMYXI0yhAYoKFLJ2k3YzVPda8/PLLfO5zn+PkyZOcOHGC06dPM3PmTDZv3gz0Hrq5rzDR/YVJTkpKwuVykZeXRyAQ4MUXX4x+d/nll/Ozn/0MUG5QHR0d55W1YsUKfv/73xMKhWhqamLTpk0sW7asX22DDTE9XDo6OkhKSiI1NZWGhgbeeustbDYb8+bN4/jx49FkNZHxEICrr76a//qv/4refCKZ6o4fP86sWbN44IEHuPHGG9m9ezeg/Fa1tbW9Xj9S7u9///ten5JTUlKYOXMmL730EqDc8KqqqjCbzezatYtdu3Z1cwiR67388ss0NjYCStjtkydP9qo1Qrw39L60r1ixIvr3sXfv3qj2/ognBHp/Nvf1t7tixQpee+01urq66Ozs5NVXX+0zQVRCQgLhcJjTp0+zevVqnnjiiWireDjoyilEbjwA0zIcuomsGKt7rFm7di233HJLt3233norv/3tb4FzoZtLSkqYOXMmt9xyC7W1taxatYrFixezZs2a6FNkJExyZKA5FpfLxaOPPkpFRQVXXnkl8+bNi3734x//mPXr17Nw4UKWLl3Kvn37yMzM5JJLLqGkpISHHnqIW265hUWLFlFaWspll13GE088QW5ubr/a8vLyoiGmS0tLKSsr6zfE9HApLS1lyZIlFBcXc88993DJJZfg9XpxOBz89Kc/5ZprrmH58uXk5OSQmqrkCfnGN75BIBBg0aJFlJSU8I1vfANQbuwlJSUsXryYgwcP8rnPfY5wOMzRo0f7zJzm8/moqKjgxz/+cTSfdE9efPFFnn32WUpLSykuLo4O7vbFggULeOyxx7jqqqtYtGgRV155JXV1db1qjdDf03UsfWm///77cbvdLFq0iCeeeGJA5w/dQ6AvWrSIiy666LxB9P5sjoRtjww0RygrK2PNmjUsW7aMiooK/umf/qnPMOw+n49QKMRnPvMZFi5cyJIlS3jwwQcHPYvpPPoabNDyBZwA9gC7UAc8gAzgHeCI+p4+UDnDGWjWE30ONE9iOjo6xtqEMSGi2+VySSmlDIfD8v7775dPPfXUoMvas2ePfPDBB3v9rudA6Vii97oeiMEONI9lS2G1lHKxPBfT+2HgXSllEfCuum1gYDAEfv7zn7N48WKKi4tpb2/ni1/84qDLKCkp4amnntLAOoPxzJgk2RFCnADKpZRnY/YdAlZJKeuEEHnABinl+SNCMQw2yU44HNZlLtdY3UaSncmNHnXrUTNol2RnrGYfSeBtIYQE/ldK+QyQI6WsA1AdQ3ZvJwoh7gXuBZg6dWp0RsmsWbNwOp1UVVUBkJmZSXFxMZs2bQLAYrGQmppKIBCgo6MDgPLychoaGjh9+jQARUVF2Gw29u7dC0B2djZz5syJ9snbbDYqKyvZtm1bdDCnoqKCmpqa6GDc3LlzMZvN7N+/H4Dc3FxmzpzJli1bAHA4HFRUVLB169ZoP3hlZSXV1dXU19cDSr9qKBTi0KFDAOTn51NQUMDWrVsBZSFUeXk5W7ZsiU7xW758OYcPH44O0JWUlODz+Thy5Agej4c5c+aQk5OD1+ulo6MDs9kcHYyNkJycjMfjic5qcDgchEIh/H4/oAxsWSyW6CIws9lMYmJitzKcTiddXV3RMhITEwkGg93KMJvNUe1msxmHw9FtcMzpdNLZ2RlNNZmYmEggECAQCETrQQgRnclhsViw2+3RMoQQmEwmpJTRMpKSkvD7/f2WYbPZ6OzsjJaRnJyM2+2O9lknJSXh8/mis1nsdjtSymgdWK1WEhISomWYTCaSkpK6lZGcnIzX6+23DKvVGv2NI2XEW0/hcBi73T5h6ik5OblbGUOpJ5PJRDgcnlD1FPmNh1NPkbrur54ienbs2NHtvtcfY9VSmCqlPKPe+N8B/gVYJ6VMizmmVUp5fnLaGAbbUtiwYQOrVq0amtETmFjd1dXVOJ1OMjMzJ/XTlcvlmjTJZgaDHnXrUTMMrFtKSXNzMy6X67x1NeOupSClPKO+NwohXgWWAQ1CiLyY7qPGsbBtslNQUEBNTQ1NTU1jbYqmeL1eXcbZ16NuPWqG+HTb7XYKCgoGVe6oOwUhRBJgklK61M9XAd8B1gF3AY+r7/3PXRsCpaWlI13khCBWt9Vq7XU17mSjtbWV9PR+G5qTEj3q1qNm0E73WIy65gCbhRBVwIfAm1LKv6A4gyuFEEeAK9XtEaWvhU6THT3q1qNm0KduPWoG7XSPulOQUh6XUpaqr2Ip5X+q+5ullJdLKYvU95aRvvbx48dHusgJgR5161Ez6FO3HjWDdrr1Nz/TwMDAwKBPxmT20UghhGgCTg7ilCnA+EjvNLroUbceNYM+detRMwxP9wwpZVZvX0xopzBYhBDb+pqGNZnRo249agZ96tajZtBOt9F9ZGBgYGAQxXAKBgYGBgZR9OYUnhlrA8YIPerWo2bQp249agaNdOtqTMHAwMDAoH/01lIwMDAwMOgHwykYGBgYGETRjVMQQlwjhDgkhDgqhJg0CXyEENOEEOuFEAeEEPuEEF9W92cIId4RQhxR39Njzvma+jscEkJcPXbWDw8hhFkIsVMI8Ya6rQfNaUKIl4UQB9U6r5zsuoUQD6p/23uFEGuFEPbJqFkI8UshRKMQYm/MvkHrFEIsFULsUb97Wgw2HHJfKdkm0wswA8eAWUACUAUsGGu7RkhbHlCmfnYCh4EFwBPAw+r+h4Hvq58XqPptwEz1dzGPtY4hav8K8FvgDXVbD5pfAP5J/ZwApE1m3UA+UA041O0/AGsmo2ZgBVAG7I3ZN2idKDHlKgEBvAVcOxg79NJSWAYclUrcJT/wO0C7jOqjiJSyTkq5Q/3sAg6g/CPdhHIDQX2/Wf18E/A7KaVPSlkNHEX5fSYUQogC4HrgFzG7J7vmFJQbx7MAUkq/lLKNSa4bJZqzQwhhARKBM0xCzVLKTUDPmG+D0qmmHUiRUm6Riof4Vcw5caEXp5APnI7ZrlH3TSqEEIXAEmArPTLZAZFMdpPlt/gR8O9AOGbfZNc8C2gCnlO7zX6hhp+ftLqllLXAk8ApoA5ol1K+zSTW3IPB6sxXP/fcHzd6cQq99alNqrm4Qohk4I/Av0opO/o7tJd9E+q3EELcADRKKbfHe0ov+yaUZhULSvfCz6SUS4BOlC6FvpjwutU+9JtQukimAklCiM/0d0ov+yaU5jjpS+ew9evFKdQA02K2C1CaoJMCIYQVxSG8KKV8Rd3doDYl6ZHJbjL8FpcANwohTqB0BV4mhPgNk1szKDpqpJRb1e2XUZzEZNZ9BVAtpWySUgaAV4CLmdyaYxmszhr1c8/9caMXp/ARUCSEmCmESADuQMn0NuFRZxY8CxyQUj4V81Ukkx10z2S3DrhDCGETQswEilAGpiYMUsqvSSkLpJSFKHX5npTyM0xizQBSynrgtBBirrrrcmA/k1v3KeAiIUSi+rd+Ocq42WTWHMugdKpdTC4hxEXq7/U5BpvFcqxH3EdxZP86lJk5x4Cvj7U9I6hrOUrzcDewS31dB2QC7wJH1PeMmHO+rv4OhxjkzITx9gJWcW720aTXDCwGtqn1/RqQPtl1A98GDgJ7gV+jzLiZdJqBtSjjJgGUJ/7PD0UnUK7+VseA/0aNXBHvywhzYWBgYGAQRS/dRwYGBgYGcWA4BQMDAwODKIZTMDAwMDCIYjgFAwMDA4MohlMwMDAwMIhiOAWDCYcQIlMIsUt91QshamO2EwY4t1wI8XQc13h/hGxNFEK8qEat3CuE2KyuPu/vnP+vn+/uUcvarZZ3k7r/O0KIK0bCZgN9Y0xJNZjQCCG+BbillE/G7LNIKYNjZ9U5hBBfA7KklF9Rt+cCJ6SUvn7OcUspz3McahDAjShRcdtV55IllYBoBgYjgtFSMJgUCCGeF0I8JYRYD3xfCLFMCPG+Gjju/cgqYCHEKnEu/8K31Bj2G4QQx4UQD8SU5445foM4l8PgxUh8eiHEdeq+zWrc+jd6MS0PqI1sSCkPRRyCEOIzQogP1RbO/wolP8TjKBFBdwkhXuxRVjbgAtxqWe6IQ1D1f0JtCUVaTXuEEFL9frYQ4i9CiO1CiL8LIeaNwM9uMAmxjLUBBgYjyBzgCillKBJmWkoZVLtVvgvc2ss584DVKLkoDgkhfiaVGDuxLAGKUWLI/AO4RAixDfhf9RrVQoi1fdj0S+BtIcQnUFakviClPCKEmA98ErhEShkQQvwU+LSU8mEhxJeklIt7KasKaACqhRDvAq9IKf8Ue4CUchvKqmeEEP8P+Iv61TPAfeq1K4CfApf1YbOBjjGcgsFk4iUpZUj9nAq8IIQoQgkDYu3jnDfVJ3efEKIRyKF76GFQYsrUAAghdgGFKE/rx2O6btYC9/YsXEq5SwgxC7gKJbjbR0KISpQYPkvVbQAH54Kd9Yrq7K4BLlTP/6EQYqmU8ls9jxVC3I4SLO8qtZvpYuAlcS4Jl62/axnoF8MpGEwmOmM+Pwqsl1LeIpQ8Exv6OCe2bz9E7/8TvR0Td4pDKaUbJbrnK0KIMEpsKj9Kq+Fr8ZajliVRArx9KIR4B3gO+FbsMUKIYpR4QStUR2IC2vpofRgYdMMYUzCYrKRyri9/jQblHwRmqQ4HlK6g8xBCXCLUvLrqzKgFwEmUrqRPCCGy1e8yhBAz1NMCQgmH3rOsqUKIsphdi9WyYo9JRQkn/jkpZROAVPJrVAshblOPEUKI0sFLNtADRkvBYLLyBEr30VeA90a6cCmlRwjxf4C/CCHO0nd45tnAz9TBaRPwJvBHKaUUQvwHyniDCSUy5j+j3OSfAXYLIXZIKT8dU5YVeFIIMRXwomRhu6/H9W4GZgA/j3QVqS2ET6t2/Idazu9QxigMDLphTEk1MBgiQohkKaVbveH/BDgipfzhWNtlYDAcjO4jA4Oh8wV14HkfSnfV/46tOQYGw8doKRgYGBgYRDFaCgYGBgYGUQynYGBgYGAQxXAKBgYGBgZRDKdgYGBgYBDFcAoGBgYGBlH+fxSJZMQqhkPbAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "def plot_symbolic_comparison(pretrained_n, pretrained_accs, scratch_n, scratch_accs):\n",
    "    import scipy.stats\n",
    "    scratch_accuracy = np.mean(scratch_accs, axis=1)\n",
    "    scratch_acc_sem = scipy.stats.sem(scratch_accs, axis=1)\n",
    "    pretrained_accuracy = np.mean(pretrained_accs, axis=1)\n",
    "    pretrained_acc_sem = scipy.stats.sem(pretrained_accs, axis=1)\n",
    "    fig, ax = plt.subplots(figsize=(6,4))\n",
    "    ax.plot(scratch_n, scratch_accuracy, label='MLP with purely symbolic input')\n",
    "    ax.fill_between(scratch_n, scratch_accuracy - scratch_acc_sem,\n",
    "        scratch_accuracy + scratch_acc_sem, alpha=0.5)\n",
    "    ax.plot(pretrained_n, pretrained_accuracy, label='Abstractor on images, pre-learned relations')\n",
    "    ax.fill_between(pretrained_n, pretrained_accuracy - pretrained_acc_sem,\n",
    "        pretrained_accuracy + pretrained_acc_sem, alpha=0.5)\n",
    "    ax.set_xlabel('Training Set Size')\n",
    "    ax.legend()\n",
    "    ax.set_ylabel('SET Classification Accuracy')\n",
    "    ax.grid(linestyle='dashed')\n",
    "    fig.savefig('set_symbolic_vs_abstractor.pdf')\n",
    "    plt.show()\n",
    "\n",
    "dat = np.load('symbolic_vs_abstractor.npz')\n",
    "train_sizes = dat['train_sizes']\n",
    "accs = dat['accs']\n",
    "symbolic_train_sizes = dat['symbolic_train_sizes']\n",
    "symbolic_accs = dat['symbolic_accs']\n",
    "plot_symbolic_comparison(train_sizes, accs, symbolic_train_sizes, symbolic_accs)"
   ]
  }
 ],
 "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
}
