{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "from typing import List, Tuple, Dict, Any, Optional\n",
    "from td.environments import (\n",
    "    Blockworld,\n",
    "    Environment,\n",
    "    Rainbow,\n",
    "    TinySVG,\n",
    "    NanoSVG,\n",
    "    CSG2D,\n",
    "    TinySVGOffset,\n",
    "    environments,\n",
    ")\n",
    "from td.grammar import Grammar\n",
    "from td.samplers import ConstrainedRandomSampler\n",
    "from td.samplers.mutator import (\n",
    "    Mutation,\n",
    "    random_mutation,\n",
    "    AddParents,\n",
    "    forward_process,\n",
    "    forward_process_with_guards,\n",
    ")\n",
    "from td.learning.tokenizer import Tokenizer\n",
    "from td.learning.constrained_decoding import DecoderState\n",
    "from lark import Token\n",
    "from lark.parsers.lalr_interactive_parser import InteractiveParser\n",
    "from collections import defaultdict\n",
    "import random\n",
    "import tqdm\n",
    "import jax\n",
    "import jax.numpy as jnp\n",
    "import flax.linen as nn\n",
    "import optax\n",
    "from flax.training import train_state\n",
    "import enum\n",
    "import pickle\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from scipy import stats\n",
    "import iceberg as ice\n",
    "from lark import Tree, Transformer\n",
    "from td.grammar import Grammar, Compiler\n",
    "from td.environments.environment import Environment\n",
    "from td.environments.goal_checker import GaussianImageGoalChecker\n",
    "\n",
    "\n",
    "jnp.set_printoptions(suppress=True)\n",
    "np.set_printoptions(suppress=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "# env = Rainbow()\n",
    "# env = Blockworld()\n",
    "# env = TinySVG()\n",
    "# env = NanoSVG()\n",
    "# env = CSG2D()\n",
    "# env = TinySVGOffset()\n",
    "env_name = \"csg2da\"\n",
    "env = environments[env_name]()\n",
    "sampler = ConstrainedRandomSampler(env.grammar)\n",
    "tokenizer = Tokenizer(env.grammar)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(+ (- (Quad 1 9 5 0 N) (Circle B 5 5)) (- (Quad E 5 F B J) (Quad E E 8 7 L)))\n",
      "s\n",
      "  binop\n",
      "    add\n",
      "    s\n",
      "      binop\n",
      "        subtract\n",
      "        s\n",
      "          quad\n",
      "            one\n",
      "            nine\n",
      "            five\n",
      "            zero\n",
      "            sevendeg\n",
      "        s\n",
      "          circle\n",
      "            eleven\n",
      "            five\n",
      "            five\n",
      "    s\n",
      "      binop\n",
      "        subtract\n",
      "        s\n",
      "          quad\n",
      "            fourteen\n",
      "            five\n",
      "            fifteen\n",
      "            eleven\n",
      "            threedeg\n",
      "        s\n",
      "          quad\n",
      "            fourteen\n",
      "            fourteen\n",
      "            eight\n",
      "            seven\n",
      "            fivedeg\n",
      "\n"
     ]
    }
   ],
   "source": [
    "random.seed(2)\n",
    "expr = sampler.sample(env.grammar.start_symbol, min_primitives=4, max_primitives=7)\n",
    "print(expr)\n",
    "print(env.grammar.parse(expr).pretty())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'(+ (Circle 6 1 B) (Circle D 7 3))'"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "sampler.sample(env.grammar.start_symbol, min_primitives=2, max_primitives=2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(+ (Circle 5 7 5) (Quad B 9 9 F G))\n",
      "                  ^^^^^^^^^^^^^^^^ --> (+ (Circle 5 3 7) (Quad A 8 A C G))\n"
     ]
    }
   ],
   "source": [
    "expr = sampler.sample(env.grammar.start_symbol, min_primitives=2, max_primitives=2)\n",
    "mut = random_mutation(expr, env.grammar, sampler)\n",
    "print(mut.pretty(expr))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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ICIiJUVBRERKioKIiJQUBRERKSkKIiJSUhRERKSkKIiISElREBGRkqIgIiIlRUFEREqKgoiIlBQFEREpKQoiIlJSFEREpKQoiIhISVEQEZGSoiAiIiVFQURESoqCiIiUFAURESkpCiIiUlIURESkpCiIiEhJURARkZKiICIiJUVBRERKioKIiJQUBRERKSkKIiJSUhRERKSkKIiISElREBGRkqIgIiIlRUFEREqKgoiIlBQFEREpKQoiIlJSFEREpKQoiIhISVEQEZGSoiAiIiVFQURESoqCiIiUFAURESkpCiIiUlIURESkpCiIiEhJURARkZKiICIiJUVBRERKioKIiJQUBRERKSkKIiJSUhRERKSkKIiISElREBGRkqIgIiIlRUFEREqKgoiIlBQFEREpKQoiIlJSFEREpKQoiIhISVEQEZGSoiAiIiVFQURESoqCiIiUFAURESkpCiIiUlIURESkpCiIiEhJURARkZKiICIiJUVBRERKioKIiJQUBRERKSkKIiJSUhRERKSkKIiISElREBGRkqIgIiIlRUFEREqKgoiIlBQFEREpKQoiIlJSFEREpKQoiIhISVEQEZGSoiAiIiVFQURESoqCiIiUFAURESkpCiIiUlIURESkpCiIiEhJURARkZKiICIiJUVBRERKxnvvJz2EiIgcDVopiIhISVEQEZGSoiAiIiVFQURESoqCiIiUFAURESkpCiIiUlIURESkpCiIiEjp/wdHHp4nh7zUbQAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "img = env.compile(\n",
    "    # \"(Arrange h (Arrange h (Arrange h (Box purple) (Ball purple)) (Box red)) (Arrange v (Box yellow) (Box orange)))\"\n",
    "    # \"(Arrange h (Rectangle 4 4 blue none 4) (Move (Ellipse 9 9 red black 2) 1 3 true true) 1)\"\n",
    "    expr\n",
    ")\n",
    "plt.imshow(img)\n",
    "plt.axis(\"off\")\n",
    "plt.show()\n",
    "# img.scale(10)\n",
    "# img"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10000/10000 [00:04<00:00, 2098.11it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Max length 175\n",
      "Mean length 90.8094\n",
      "Median length 91.0\n",
      "Min length 23\n"
     ]
    },
    {
     "data": {
      "image/png": "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pLKyMg0cOFBut9vpk56ersbGRh04cKCttQAAgCAW0ExLamqqVq9erdtvv13Hjh3Tc889p3/+53/W/v375fP5FBERodjYWL/XuN1u+Xw+SZLP5/MLLOfbz7ddTlNTk5qampz9xsbGQIYNAACCQEChZezYsc6vBw0apNTUVPXp00d//OMf1bVr13Yf3HkLFizQc889d83ODwAAbnxtuuU5NjZWt912m/785z/L4/HozJkzqq+v9+tTW1vrrIHxeDwX3U10fv9S62TOKygoUENDg7PV1NS0ZdgAAMBCbQotJ0+e1KeffqqePXtq6NChCg8PV2lpqdNeVVWl6upqeb1eSZLX69W+fftUV1fn9CkpKZHL5VJKSspl3ycyMlIul8tvAwAAnUtAl4d+9rOfady4cerTp4+OHj2qZ555Rl26dNHEiRMVExOjKVOmKD8/X3FxcXK5XHriiSfk9Xo1YsQISdLo0aOVkpKiSZMmadGiRfL5fJo1a5ZycnIUGRl5TQoEAADBIaDQ8sUXX2jixIn661//qltuuUWjRo3Sjh07dMstt0iSFi9erNDQUGVmZqqpqUnp6elatmyZ8/ouXbqouLhYU6dOldfrVXR0tLKzszV37tz2rQoAAASdEGOM6ehBBKqxsVExMTFqaGjosEtFt85c77f/l4UZHTIOAABs0daf3zx7CAAAWIHQAgAArEBoAQAAViC0AAAAKxBaAACAFQgtAADACoQWAABgBUILAACwAqEFAABYgdACAACsQGgBAABWILQAAAArEFoAAIAVCC0AAMAKhBYAAGAFQgsAALACoQUAAFiB0AIAAKxAaAEAAFYgtAAAACsQWgAAgBUILQAAwAqEFgAAYAVCCwAAsAKhBQAAWIHQAgAArEBoAQAAViC0AAAAKxBaAACAFQgtAADACoQWAABgBUILAACwAqEFAABYgdACAACsQGgBAABWILQAAAArEFoAAIAVCC0AAMAKhBYAAGAFQgsAALACoQUAAFiB0AIAAKwQ1tEDAAJx68z1Fx37y8KMDhgJAOB6Y6YFAABYgdACAACsQGgBAABWILQAAAArsBA3SLFgFQAQbJhpAQAAViC0AAAAKxBaAACAFdoUWhYuXKiQkBDl5eU5x06fPq2cnBzFx8ere/fuyszMVG1trd/rqqurlZGRoW7duikhIUHTp0/X2bNn2zIUAAAQ5FodWnbt2qXf/va3GjRokN/xadOm6a233tLatWu1detWHT16VOPHj3faz507p4yMDJ05c0bbt2/Xyy+/rNWrV2vOnDmtrwIAAAS9VoWWkydPKisrS7/73e900003OccbGhq0cuVKvfjii7rvvvs0dOhQrVq1Stu3b9eOHTskSRs3blRlZaVeeeUVDR48WGPHjtW8efNUWFioM2fOtE9VAAAg6LQqtOTk5CgjI0NpaWl+xysqKtTc3Ox3vF+/fkpKSlJZWZkkqaysTAMHDpTb7Xb6pKenq7GxUQcOHLjk+zU1NamxsdFvw7Vx68z1fhsAADeKgL+n5bXXXtPu3bu1a9eui9p8Pp8iIiIUGxvrd9ztdsvn8zl9vh5Yzrefb7uUBQsW6Lnnngt0qAAAIIgENNNSU1Ojp556Sq+++qqioqKu1ZguUlBQoIaGBmerqam5bu8NAABuDAGFloqKCtXV1emuu+5SWFiYwsLCtHXrVi1ZskRhYWFyu906c+aM6uvr/V5XW1srj8cjSfJ4PBfdTXR+/3yfC0VGRsrlcvltAACgcwkotNx///3at2+f9u7d62zDhg1TVlaW8+vw8HCVlpY6r6mqqlJ1dbW8Xq8kyev1at++faqrq3P6lJSUyOVyKSUlpZ3KAgAAwSagNS09evTQHXfc4XcsOjpa8fHxzvEpU6YoPz9fcXFxcrlceuKJJ+T1ejVixAhJ0ujRo5WSkqJJkyZp0aJF8vl8mjVrlnJychQZGdlOZQEAgGDT7g9MXLx4sUJDQ5WZmammpialp6dr2bJlTnuXLl1UXFysqVOnyuv1Kjo6WtnZ2Zo7d257DwUAAASRNoeW9957z28/KipKhYWFKiwsvOxr+vTpo7fffrutbw0AADoRnj0EAACsQGgBAABWILQAAAArEFoAAIAVCC0AAMAKhBYAAGAFQgsAALACoQUAAFiB0AIAAKxAaAEAAFYgtAAAACsQWgAAgBUILQAAwAqEFgAAYAVCCwAAsAKhBQAAWIHQAgAArEBoAQAAViC0AAAAKxBaAACAFQgtAADACoQWAABgBUILAACwAqEFAABYgdACAACsQGgBAABWILQAAAArEFoAAIAVCC0AAMAKhBYAAGCFsI4eABBsbp253m//LwszOmgkABBcmGkBAABWYKYFQKd24cyYxOwYcKNipgUAAFiB0AIAAKxAaAEAAFZgTQtwA2BdBQBcGaEFQNDgdnMguHF5CAAAWIHQAgAArEBoAQAAViC0AAAAKxBaAACAFQgtAADACoQWAABgBUILAACwAqEFAABYgdACAACsQGgBAABWILQAAAArBBRali9frkGDBsnlcsnlcsnr9eqdd95x2k+fPq2cnBzFx8ere/fuyszMVG1trd85qqurlZGRoW7duikhIUHTp0/X2bNn26caAAAQtAJ6ynOvXr20cOFC9e3bV8YYvfzyy3rwwQe1Z88eDRgwQNOmTdP69eu1du1axcTEKDc3V+PHj9eHH34oSTp37pwyMjLk8Xi0fft2HTt2TJMnT1Z4eLjmz59/TQoE0D4ufIKyxFOUAVxfAYWWcePG+e3/6le/0vLly7Vjxw716tVLK1eu1Jo1a3TfffdJklatWqX+/ftrx44dGjFihDZu3KjKykpt2rRJbrdbgwcP1rx58zRjxgw9++yzioiIaL/KgK+58AcuP2wBwD6tXtNy7tw5vfbaazp16pS8Xq8qKirU3NystLQ0p0+/fv2UlJSksrIySVJZWZkGDhwot9vt9ElPT1djY6MOHDhw2fdqampSY2Oj3wYAADqXgEPLvn371L17d0VGRurxxx/XunXrlJKSIp/Pp4iICMXGxvr1d7vd8vl8kiSfz+cXWM63n2+7nAULFigmJsbZevfuHeiwAQCA5QIOLbfffrv27t2r8vJyTZ06VdnZ2aqsrLwWY3MUFBSooaHB2Wpqaq7p+wEAgBtPQGtaJCkiIkLf+c53JElDhw7Vrl279Jvf/EYPP/ywzpw5o/r6er/ZltraWnk8HkmSx+PRzp07/c53/u6i830uJTIyUpGRkYEOFQAABJE2f09LS0uLmpqaNHToUIWHh6u0tNRpq6qqUnV1tbxeryTJ6/Vq3759qqurc/qUlJTI5XIpJSWlrUMBAABBLKCZloKCAo0dO1ZJSUk6ceKE1qxZo/fee0/vvvuuYmJiNGXKFOXn5ysuLk4ul0tPPPGEvF6vRowYIUkaPXq0UlJSNGnSJC1atEg+n0+zZs1STk4OMykAAOAbBRRa6urqNHnyZB07dkwxMTEaNGiQ3n33Xf3Lv/yLJGnx4sUKDQ1VZmammpqalJ6ermXLljmv79Kli4qLizV16lR5vV5FR0crOztbc+fObd+qAABA0AkotKxcufIb26OiolRYWKjCwsLL9unTp4/efvvtQN4WAAAg8IW4wLXCN64CAL4JD0wEAABWILQAAAArcHnoEq7Vc2q4/AEAQOsx0wIAAKxAaAEAAFbg8hAAAJfAJf0bDzMtAADACsy0AADa3bW6oQGdGzMtAADACoQWAABgBUILAACwAmtaAFxTrG0A0F4ILQCADkGgRaC4PAQAAKxAaAEAAFYgtAAAACsQWgAAgBVYiAsANzgWrAJ/x0wLAACwAqEFAABYgdACAACsQGgBAABWYCEuAHQgFtkCV4/QAgBB6MIwJBGIYD8uDwEAACsw0wKA/5UDsAIzLQAAwAqEFgAAYAVCCwAAsAKhBQAAWIGFuACATofF53ZipgUAAFiB0AIAAKxAaAEAAFYgtAAAACsQWgAAgBUILQAAwAqEFgAAYAVCCwAAsAKhBQAAWIHQAgAArEBoAQAAViC0AAAAKxBaAACAFQgtAADACoQWAABgBUILAACwAqEFAABYgdACAACsEFBoWbBggYYPH64ePXooISFBDz30kKqqqvz6nD59Wjk5OYqPj1f37t2VmZmp2tpavz7V1dXKyMhQt27dlJCQoOnTp+vs2bNtrwYAAAStsEA6b926VTk5ORo+fLjOnj2rp59+WqNHj1ZlZaWio6MlSdOmTdP69eu1du1axcTEKDc3V+PHj9eHH34oSTp37pwyMjLk8Xi0fft2HTt2TJMnT1Z4eLjmz5/f/hUCAHAd3Tpzvd/+XxZmdNBIgk9AoWXDhg1++6tXr1ZCQoIqKip0zz33qKGhQStXrtSaNWt03333SZJWrVql/v37a8eOHRoxYoQ2btyoyspKbdq0SW63W4MHD9a8efM0Y8YMPfvss4qIiGi/6gAAQNBo05qWhoYGSVJcXJwkqaKiQs3NzUpLS3P69OvXT0lJSSorK5MklZWVaeDAgXK73U6f9PR0NTY26sCBA20ZDgAACGIBzbR8XUtLi/Ly8jRy5EjdcccdkiSfz6eIiAjFxsb69XW73fL5fE6frweW8+3n2y6lqalJTU1Nzn5jY2Nrhw0AACzV6pmWnJwc7d+/X6+99lp7jueSFixYoJiYGGfr3bv3NX9PAABwY2lVaMnNzVVxcbG2bNmiXr16Occ9Ho/OnDmj+vp6v/61tbXyeDxOnwvvJjq/f77PhQoKCtTQ0OBsNTU1rRk2AACwWEChxRij3NxcrVu3Tps3b1ZycrJf+9ChQxUeHq7S0lLnWFVVlaqrq+X1eiVJXq9X+/btU11dndOnpKRELpdLKSkpl3zfyMhIuVwuvw0AAHQuAa1pycnJ0Zo1a/SnP/1JPXr0cNagxMTEqGvXroqJidGUKVOUn5+vuLg4uVwuPfHEE/J6vRoxYoQkafTo0UpJSdGkSZO0aNEi+Xw+zZo1Szk5OYqMjGz/CoEgxq2VADqTgELL8uXLJUnf+973/I6vWrVKP/7xjyVJixcvVmhoqDIzM9XU1KT09HQtW7bM6dulSxcVFxdr6tSp8nq9io6OVnZ2tubOndu2SgAAQFALKLQYY67YJyoqSoWFhSosLLxsnz59+ujtt98O5K0BdHLMKgFo9S3PANCZEaKA648HJgIAACsQWgAAgBUILQAAwAqsaQEAWOPCtUQS64k6E2ZaAACAFQgtAADACoQWAABgBda0AAAuizUkuJEw0wIAAKxAaAEAAFYgtAAAACsQWgAAgBUILQAAwAqEFgAAYAVCCwAAsAKhBQAAWIHQAgAArEBoAQAAViC0AAAAKxBaAACAFQgtAADACoQWAABgBUILAACwAqEFAABYgdACAACsQGgBAABWCOvoAQA2uXXmer/9vyzM6KCRAEDnw0wLAACwAjMtFrrwf/sS/+MHAAQ/ZloAAIAVCC0AAMAKhBYAAGAF1rSgzVhjAwC4HphpAQAAViC0AAAAKxBaAACAFQgtAADACizEBQBYjcdrdB6EFgCA41J3AwI3Ci4PAQAAKzDTAgDtgO8rAq49ZloAAIAVCC0AAMAKhBYAAGAFQgsAALACoQUAAFiB0AIAAKzALc8AAHQwbpm/Osy0AAAAKxBaAACAFQIOLdu2bdO4ceOUmJiokJAQvfHGG37txhjNmTNHPXv2VNeuXZWWlqZDhw759Tl+/LiysrLkcrkUGxurKVOm6OTJk20qBAAABLeAQ8upU6d05513qrCw8JLtixYt0pIlS1RUVKTy8nJFR0crPT1dp0+fdvpkZWXpwIEDKikpUXFxsbZt26bHHnus9VUAAICgF/BC3LFjx2rs2LGXbDPG6KWXXtKsWbP04IMPSpL+8Ic/yO1264033tCECRN08OBBbdiwQbt27dKwYcMkSUuXLtUPfvAD/frXv1ZiYmIbygEQDHjSMIBLadc1LYcPH5bP51NaWppzLCYmRqmpqSorK5MklZWVKTY21gkskpSWlqbQ0FCVl5df8rxNTU1qbGz02wAAQOfSrqHF5/NJktxut99xt9vttPl8PiUkJPi1h4WFKS4uzulzoQULFigmJsbZevfu3Z7DBgAAFrDi7qGCggI1NDQ4W01NTUcPCQAAXGft+uVyHo9HklRbW6uePXs6x2trazV48GCnT11dnd/rzp49q+PHjzuvv1BkZKQiIyPbc6hAUOILqgAEs3adaUlOTpbH41FpaalzrLGxUeXl5fJ6vZIkr9er+vp6VVRUOH02b96slpYWpaamtudwAABAEAl4puXkyZP685//7OwfPnxYe/fuVVxcnJKSkpSXl6df/vKX6tu3r5KTkzV79mwlJibqoYcekiT1799fY8aM0aOPPqqioiI1NzcrNzdXEyZM4M4hAABwWQGHlo8++kjf//73nf38/HxJUnZ2tlavXq2f//znOnXqlB577DHV19dr1KhR2rBhg6KiopzXvPrqq8rNzdX999+v0NBQZWZmasmSJe1QDgAACFYBh5bvfe97MsZctj0kJERz587V3LlzL9snLi5Oa9asCfStAQBAJ2bF3UMAAACEFgAAYAVCCwAAsAKhBQAAWKFdv1wOAICOxpcsBi9mWgAAgBUILQAAwAqEFgAAYAVCCwAAsAKhBQAAWIHQAgAArMAtzwDQSVx4KzC3AcM2hBbg/+MDHQBubIQWALhGCMJA+yK04JrgwxoA2hff9MtCXAAAYAlCCwAAsAKhBQAAWIHQAgAArEBoAQAAVuDuIQDXFXdAAGgtZloAAIAVCC0AAMAKhBYAAGAFQgsAALACoQUAAFiB0AIAAKxAaAEAAFbge1oAADcEvsMHV8JMCwAAsAKhBQAAWIHQAgAArMCaFiDIsU4AQLAgtKBTutQPcgBX58J/P4RgXC9cHgIAAFYgtAAAACtweQgALsDlD+DGxEwLAACwAqEFAABYgdACAACsQGgBAABWYCEuAAQBvnuo7fg9vPERWoAbFHewfDN+wACdD5eHAACAFZhpAQAgiAXT88eYaQEAAFZgpgUAcM2xBgntgZkWAABgBWZaAOAGwowEcHmEFgDthh+4AK6lDr08VFhYqFtvvVVRUVFKTU3Vzp07O3I4AADgBtZhMy3/8z//o/z8fBUVFSk1NVUvvfSS0tPTVVVVpYSEhI4aFoAgx2wQYK8OCy0vvviiHn30Uf3kJz+RJBUVFWn9+vX6/e9/r5kzZ3bUsAAAuGqE4OurQ0LLmTNnVFFRoYKCAudYaGio0tLSVFZWdlH/pqYmNTU1OfsNDQ2SpMbGxmsyvpamL/32L/U+V9PnSq+52te15jytfa/2qqs93vtSr2uvuq5Ga//cr+fvYWvqutS52uv3/lqOpz3O29pzc57rc56rOXdH/x1rzXluxM+/C193xzPvXtRn/3PpVzx3oM6/rzGmdScwHeDIkSNGktm+fbvf8enTp5vvfve7F/V/5plnjCQ2NjY2Nja2INhqampalR+suHuooKBA+fn5zn5LS4uOHz+u+Ph4hYSEXPV5Ghsb1bt3b9XU1Mjlcl2Lod4wqDV4daZ6qTV4daZ6qfUfjDE6ceKEEhMTW3X+DgktN998s7p06aLa2lq/47W1tfJ4PBf1j4yMVGRkpN+x2NjYVr+/y+UK+r8451Fr8OpM9VJr8OpM9VLr38XExLT6vB1yy3NERISGDh2q0tJS51hLS4tKS0vl9Xo7YkgAAOAG12GXh/Lz85Wdna1hw4bpu9/9rl566SWdOnXKuZsIAADg6zostDz88MP6v//7P82ZM0c+n0+DBw/Whg0b5Ha7r9l7RkZG6plnnrnoUlMwotbg1Znqpdbg1Znqpdb2E2JMa+87AgAAuH54yjMAALACoQUAAFiB0AIAAKxAaAEAAFYIutCyYMECDR8+XD169FBCQoIeeughVVVV+fU5ffq0cnJyFB8fr+7duyszM/OiL7qz0cKFCxUSEqK8vDznWLDVeuTIET3yyCOKj49X165dNXDgQH300UdOuzFGc+bMUc+ePdW1a1elpaXp0KFDHTji1jl37pxmz56t5ORkde3aVf/0T/+kefPm+T2vw9Zat23bpnHjxikxMVEhISF64403/Nqvpq7jx48rKytLLpdLsbGxmjJlik6ePHkdq7h631Rvc3OzZsyYoYEDByo6OlqJiYmaPHmyjh496ncOW+q90p/t1z3++OMKCQnRSy+95Hc8mGo9ePCgHnjgAcXExCg6OlrDhw9XdXW1027T5/OV6j158qRyc3PVq1cvde3aVSkpKSoqKvLr0x71Bl1o2bp1q3JycrRjxw6VlJSoublZo0eP1qlTp5w+06ZN01tvvaW1a9dq69atOnr0qMaPH9+Bo267Xbt26be//a0GDRrkdzyYav3b3/6mkSNHKjw8XO+8844qKyv1wgsv6KabbnL6LFq0SEuWLFFRUZHKy8sVHR2t9PR0nT59ugNHHrjnn39ey5cv13/913/p4MGDev7557Vo0SItXbrU6WNrradOndKdd96pwsLCS7ZfTV1ZWVk6cOCASkpKVFxcrG3btumxxx67XiUE5Jvq/fLLL7V7927Nnj1bu3fv1uuvv66qqio98MADfv1sqfdKf7bnrVu3Tjt27LjkV7kHS62ffvqpRo0apX79+um9997Txx9/rNmzZysqKsrpY9Pn85Xqzc/P14YNG/TKK6/o4MGDysvLU25urt58802nT7vU26onFlmkrq7OSDJbt241xhhTX19vwsPDzdq1a50+Bw8eNJJMWVlZRw2zTU6cOGH69u1rSkpKzL333mueeuopY0zw1TpjxgwzatSoy7a3tLQYj8dj/vM//9M5Vl9fbyIjI81///d/X48htpuMjAzz7//+737Hxo8fb7KysowxwVOrJLNu3Tpn/2rqqqysNJLMrl27nD7vvPOOCQkJMUeOHLluY2+NC+u9lJ07dxpJ5vPPPzfG2Fvv5Wr94osvzLe+9S2zf/9+06dPH7N48WKnLZhqffjhh80jjzxy2dfY/Pl8qXoHDBhg5s6d63fsrrvuMr/4xS+MMe1Xb9DNtFyooaFBkhQXFydJqqioUHNzs9LS0pw+/fr1U1JSksrKyjpkjG2Vk5OjjIwMv5qk4Kv1zTff1LBhw/Rv//ZvSkhI0JAhQ/S73/3OaT98+LB8Pp9fvTExMUpNTbWu3rvvvlulpaX65JNPJEn/+7//qw8++EBjx46VFFy1ft3V1FVWVqbY2FgNGzbM6ZOWlqbQ0FCVl5df9zG3t4aGBoWEhDjPVwumeltaWjRp0iRNnz5dAwYMuKg9WGptaWnR+vXrddtttyk9PV0JCQlKTU31u6QSbJ/Pd999t958800dOXJExhht2bJFn3zyiUaPHi2p/eoN6tDS0tKivLw8jRw5UnfccYckyefzKSIi4qIHLrrdbvl8vg4YZdu89tpr2r17txYsWHBRW7DV+tlnn2n58uXq27ev3n33XU2dOlVPPvmkXn75ZUlyarrwW5VtrHfmzJmaMGGC+vXrp/DwcA0ZMkR5eXnKysqSFFy1ft3V1OXz+ZSQkODXHhYWpri4OKtrl/5+zX/GjBmaOHGi87C5YKr3+eefV1hYmJ588slLtgdLrXV1dTp58qQWLlyoMWPGaOPGjfrhD3+o8ePHa+vWrZKC7/N56dKlSklJUa9evRQREaExY8aosLBQ99xzj6T2q7fDvsb/esjJydH+/fv1wQcfdPRQromamho99dRTKikp8btOGqxaWlo0bNgwzZ8/X5I0ZMgQ7d+/X0VFRcrOzu7g0bWvP/7xj3r11Ve1Zs0aDRgwQHv37lVeXp4SExODrlb8XXNzs370ox/JGKPly5d39HDaXUVFhX7zm99o9+7dCgkJ6ejhXFMtLS2SpAcffFDTpk2TJA0ePFjbt29XUVGR7r333o4c3jWxdOlS7dixQ2+++ab69Omjbdu2KScnR4mJiRddBWiLoJ1pyc3NVXFxsbZs2aJevXo5xz0ej86cOaP6+nq//rW1tfJ4PNd5lG1TUVGhuro63XXXXQoLC1NYWJi2bt2qJUuWKCwsTG63O2hqlaSePXsqJSXF71j//v2d1fjna7pwNbqN9U6fPt2ZbRk4cKAmTZqkadOmOTNqwVTr111NXR6PR3V1dX7tZ8+e1fHjx62t/Xxg+fzzz1VSUuLMskjBU+/777+vuro6JSUlOZ9Xn3/+uX7605/q1ltvlRQ8td58880KCwu74udVsHw+f/XVV3r66af14osvaty4cRo0aJByc3P18MMP69e//rWk9qs36EKLMUa5ublat26dNm/erOTkZL/2oUOHKjw8XKWlpc6xqqoqVVdXy+v1Xu/htsn999+vffv2ae/evc42bNgwZWVlOb8OllolaeTIkRfdvv7JJ5+oT58+kqTk5GR5PB6/ehsbG1VeXm5dvV9++aVCQ/3/eXbp0sX5H1ww1fp1V1OX1+tVfX29KioqnD6bN29WS0uLUlNTr/uY2+p8YDl06JA2bdqk+Ph4v/ZgqXfSpEn6+OOP/T6vEhMTNX36dL377ruSgqfWiIgIDR8+/Bs/r4LpZ1Fzc7Oam5u/8TOr3ept5eLhG9bUqVNNTEyMee+998yxY8ec7csvv3T6PP744yYpKcls3rzZfPTRR8br9Rqv19uBo24/X797yJjgqnXnzp0mLCzM/OpXvzKHDh0yr776qunWrZt55ZVXnD4LFy40sbGx5k9/+pP5+OOPzYMPPmiSk5PNV1991YEjD1x2drb51re+ZYqLi83hw4fN66+/bm6++Wbz85//3Olja60nTpwwe/bsMXv27DGSzIsvvmj27Nnj3C1zNXWNGTPGDBkyxJSXl5sPPvjA9O3b10ycOLGjSvpG31TvmTNnzAMPPGB69epl9u7d6/eZ1dTU5JzDlnqv9Gd7oQvvHjImeGp9/fXXTXh4uFmxYoU5dOiQWbp0qenSpYt5//33nXPY9Pl8pXrvvfdeM2DAALNlyxbz2WefmVWrVpmoqCizbNky5xztUW/QhRZJl9xWrVrl9Pnqq6/Mf/zHf5ibbrrJdOvWzfzwhz80x44d67hBt6MLQ0uw1frWW2+ZO+64w0RGRpp+/fqZFStW+LW3tLSY2bNnG7fbbSIjI839999vqqqqOmi0rdfY2Gieeuopk5SUZKKiosy3v/1t84tf/MLvB5mttW7ZsuWS/0azs7ONMVdX11//+lczceJE0717d+NyucxPfvITc+LEiQ6o5sq+qd7Dhw9f9jNry5YtzjlsqfdKf7YXulRoCaZaV65cab7zne+YqKgoc+edd5o33njD7xw2fT5fqd5jx46ZH//4xyYxMdFERUWZ22+/3bzwwgumpaXFOUd71BtizNe+YhMAAOAGFXRrWgAAQHAitAAAACsQWgAAgBUILQAAwAqEFgAAYAVCCwAAsAKhBQAAWIHQAgAArEBoAQAAViC0AAAAKxBaAACAFQgtAADACv8PLyBOO5hyf84AAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "def sampler_lengths(\n",
    "    env: Environment,\n",
    "    sampler: ConstrainedRandomSampler,\n",
    "    tokenizer: Tokenizer,\n",
    "    N: int = 10000,\n",
    "):\n",
    "    return [\n",
    "        len(\n",
    "            tokenizer._tokenize_one(\n",
    "                sampler.sample(\n",
    "                    env.grammar.start_symbol, min_primitives=2, max_primitives=10\n",
    "                )\n",
    "            )\n",
    "        )\n",
    "        for _ in tqdm.trange(N)\n",
    "    ]\n",
    "\n",
    "\n",
    "lengths = sampler_lengths(env, sampler, tokenizer, N=10000)\n",
    "print(\"Max length\", max(lengths))\n",
    "print(\"Mean length\", np.mean(lengths))\n",
    "print(\"Median length\", np.median(lengths))\n",
    "print(\"Min length\", min(lengths))\n",
    "plt.hist(lengths, bins=100)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10000/10000 [00:15<00:00, 633.01it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Shorter: 30.79%, Equal: 49.24%, Longer: 19.97%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "def mutation_length_effects(\n",
    "    env: Environment,\n",
    "    sampler: ConstrainedRandomSampler,\n",
    "    tokenizer: Tokenizer,\n",
    "    N: int = 10000,\n",
    "):\n",
    "    shorter = 0\n",
    "    equal = 0\n",
    "    longer = 0\n",
    "\n",
    "    for _ in tqdm.trange(N):\n",
    "        expr = sampler.sample(env.grammar.start_symbol)\n",
    "        mut = random_mutation(expr, env.grammar, sampler)\n",
    "        new_expr = mut.apply(expr)\n",
    "\n",
    "        expr = tokenizer._tokenize_one(expr)\n",
    "        new_expr = tokenizer._tokenize_one(new_expr)\n",
    "\n",
    "        if len(new_expr) < len(expr):\n",
    "            shorter += 1\n",
    "        elif len(new_expr) == len(expr):\n",
    "            equal += 1\n",
    "        else:\n",
    "            longer += 1\n",
    "\n",
    "    return shorter / N, equal / N, longer / N\n",
    "\n",
    "\n",
    "shorter, equal, longer = mutation_length_effects(env, sampler, tokenizer, N=10000)\n",
    "print(\n",
    "    f\"Shorter: {shorter*100:.2f}%, Equal: {equal*100:.2f}%, Longer: {longer*100:.2f}%\"\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10000/10000 [00:06<00:00, 1528.44it/s]\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "{' ': 399640,\n",
       " '(': 130000,\n",
       " ')': 130000,\n",
       " 'Circle': 35180,\n",
       " 'Quad': 34820,\n",
       " '-': 30177,\n",
       " '+': 29823,\n",
       " 'B': 15610,\n",
       " '0': 15438,\n",
       " '4': 15425,\n",
       " 'C': 15396,\n",
       " '6': 15366,\n",
       " '2': 15325,\n",
       " 'F': 15315,\n",
       " '5': 15285,\n",
       " '8': 15285,\n",
       " 'D': 15281,\n",
       " '1': 15242,\n",
       " '9': 15218,\n",
       " '7': 15208,\n",
       " '3': 15206,\n",
       " 'E': 15113,\n",
       " 'A': 15107,\n",
       " 'L': 4416,\n",
       " 'G': 4382,\n",
       " 'M': 4357,\n",
       " 'I': 4349,\n",
       " 'H': 4340,\n",
       " 'N': 4337,\n",
       " 'K': 4330,\n",
       " 'J': 4309}"
      ]
     },
     "execution_count": 34,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "def token_counts(\n",
    "    env: Environment,\n",
    "    tokenizer: Tokenizer,\n",
    "    sampler: ConstrainedRandomSampler,\n",
    "    N: int = 10000,\n",
    "):\n",
    "    counts = defaultdict(int)\n",
    "    for _ in tqdm.trange(N):\n",
    "        expr = sampler.sample(\n",
    "            env.grammar.start_symbol, min_primitives=7, max_primitives=7\n",
    "        )\n",
    "        tokens = tokenizer._tokenize_one(expr)\n",
    "        for t in tokens:\n",
    "            counts[t] += 1\n",
    "\n",
    "    counts = {\n",
    "        tokenizer._untokenize_one([k]): v\n",
    "        for k, v in sorted(counts.items(), key=lambda x: -x[1])\n",
    "    }\n",
    "    return counts\n",
    "\n",
    "\n",
    "counts = token_counts(env, tokenizer, sampler, N=10000)\n",
    "counts"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAUkAAAFICAYAAADd1gwNAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8g+/7EAAAACXBIWXMAAA9hAAAPYQGoP6dpAABDBElEQVR4nO3deXwUZZoH8F/1kU5C7gAhQEgkmRBJGBgOQUUUuQdQURRUNAmo6EzkkIgjrgeLDHLPcMqNsIjKgAgr9ykSQLnCfUsSAkkgkINcne5+9g+2ESHpdKqruqq7n+/nU5+dla6qp9/q+uWtqreqBCIiMMYYq5JG6QIYY0zNOCQZY8wGDknGGLOBQ5IxxmzgkGSMMRs4JBljzAYOScYYs4FDkjHGbOCQZIwxGzwyJK9fv47XXnsNXl5eEAThD9Obb75pc16TySR6vfevS+qpfv362LBhAxy9iWrnzp1o0qSJrLWK5Uj7A8CePXug0+lk3xbOmBy1c+dOhISESFKLj48PvvnmG4d/e6pEKlBZWUmXLl0ii8XitHUWFhbShAkTKCgoiADcnUaMGGFzvqKiItHrvHc9ck0RERG0ceNG0TUSEVksFtqwYQNFRETIVqdYjrQ/EdG+ffvIx8fHKdtC7skRWVlZ1KpVK0nriY6OpuPHjztUlxop3pM0Go2YOnUq+vTpgy1btjjtL1FAQABSU1OxdOlSNG3aFMCdnl5AQIDN+crKypxRnmhZWVkYOnQoNm/eLHoZgiCgZ8+emDdvHiIiIiSsznGOtr+Xlxe8vLwkqsY1WSwWLF68GMePH5d0uZcuXcKMGTNgNBolXa7ilEzoiooK+uc//0l16tS52wvatGmTU2uwWCy0d+9eio+PJ61WSwsWLLD5+UuXLoleF5zYy4iMjKTNmzeLrpXoTtts3LhRlh6lWJcvX3boO509e5YaNWqkeC9QyTbMysqipk2bylJTaGgoHTt2zKFtpDaKhWR5eTmNHz/+bkBapyZNmjg9KImIjh49Sh06dKAdO3bY/Nz+/ftFr8PZO1FUVJQkQblp0yZq0qSJKnbwo0ePOvR9bty4QS1btlQ84JRswxUrVpDBYJClJo1GQ5MmTXJoG6mNIiFZXl5O48aNI19f3yobWqmgvHjxYo3nvFavXi16+UrsSFL0KImINm/eTJGRkYrv4D/++KND38NsNlPPnj0VDzgl2zAxMVHWunr37k0lJSUObSc1cfo5yfLycnzxxRcYP348SktLq/xMZmYm3nrrLWzevNmpV8uaNm0Kf39/m5+5evWqk6qRRkZGhiRt2b17d8yfPx9RUVHSFSfClStXHJpfo9GgXbt2ElXjmn799VdZl3/+/HkUFRXJug5ncnpIbt26FVOmTEF5ebnNz1mDcuvWraoZVkBEyMrKUrqMWpMyKOfNm4fIyEgJq6ud7Oxsh5fRqVMnaLVaCapxTfn5+bIu/9atW6ioqJB1Hc7k9JDs1KkTBg4cCJ1OV+NnMzMz8cYbbzj1qrctFRUVOH/+vNJliCJV79zao2zSpImE1dkvOzsblZWVDi0jNjZW8R6xkuQOyaKiIve6wq3EMX5BQQG98cYbpNfr7TrHYb3q7cxxlFW5efMmtWzZUvT89nxXuaeIiAjasGGDw225adMmh656i9WpUyeHx0qaTCbZz8s5YxIrJiZG1roaNmxImZmZDm0jNVHs6vatW7dcLigzMzMpKChI9PxK71RSBqWjw4PEatCgAeXk5Iie3+qHH34grVar+LZQIiTlvnCVkJBAeXl5Dm8jtVBsMHlQUBAmT56M119/HXq9vsbPZ2Vl4c0331T00PvEiRPVXmxyJfcOOBfbloIgoEePHk4fcF5YWIiMjAyHl9O+fXvEx8dLUJHr+dOf/iTr8lu0aIHAwEBZ1+FMit5xExQUhClTpiAxMdElgnL//v0Onw9Ti6ysLIfPUSpxZ05lZSVOnjzp8HLq16+PV199FRqN4jedOd2zzz5b4ygOsXQ6HZ5++mn3uqtJ2Y7sHdZzlDqdzu7DRWcfepeUlFCfPn0cOsyx57s5e5JiTKqYe70dacOUlBQym80O1Ux05/RJbGys4ttA7CRWcXExdezYUZaaoqKi6LfffnN426iJQKSCy8a4cxhlvZfanie9REZGYsGCBejWrZsTqrvz5KCnn34aJ06ccKjnpUaRkZGYP38+unfvLnoZRITNmzfjrbfesmuYlCNt2KFDB2zYsAHBwcGilnFvDbNnz8bIkSMdfrqQEsS2IRHhhx9+qPGpPenp6Th79qzdy9XpdBg3bhzef/999xpipVg8V6GgoICGDBli9wn1yMhI2rJli1Nqq6yspNTUVNJoNKKXYc93Umpy9i2MjrShn58fnTx50qFarW7evEmdO3dWvP3FTHJ7//33a1VPhw4dKDc3V/a6nE1VIUl056p3cnKy3UEZFRXltKC8ePEiRUVFiZ5f6Z3KGUFJdGd4UE23MDrShlqtlr788kuH67TavXs31a9fX/H2d+WQjImJocOHD8tekxJUF5JE6g1Ks9lMEyZMED2/0juVPVNkZKQk53snT55sc/s52ob9+/en8vJyh2q0MplMNG3aNNke+uDuIdm8eXPaunWr4uOY5aLKkCT6PSjtvZgTGRlJ27Ztk31DFRQUiJ5X6Z3K3qlJkya0ceNG0W25e/fuGgcsO9qGYWFhdP78edHLuV9xcTG99tprpNFoFG9/VwlJb29v6t69u2SnPtRKtSFJJC4ot2zZotq/aErvVLWZrE84r01bWiwWuwJSipAUBIHmzZsnejlVuXbtGvXt29dlglJu1YWkn58ftW7dmhYtWuTw3U+uQNUhSeReQVldzXq9nurWrUvh4eEUFBSkmp20Nnfm1CYgpQhJ4M4juUpLS0UvqypZWVnUq1cv1WyDqiZBECghIUHS712V2bNnU1xcHEVERFDbtm2pV69e9NFHH9HmzZvp5s2bsq9fLVQfkkTuE5RV1RoeHk7Tp0+nU6dO0Y0bN2j//v30zjvvPPAwYjUHpTUgo6Oj7V6uFG0YHBwsy8WCq1ev0sCBA1V526JGo6EePXpIeqqhOiaTicrKyqi0tJTKy8upoqJCkvGprsYlQpLozlANVw/K+2sMDQ2lDRs2PPDDKysrowkTJqjmQoKtQ28xASlVSAKg999/X5ZtnJ+fTykpKap6aZiXlxcNGjSIrl69Kvn3ZdVzmZAkcv2gvL++4cOHV/uX+ebNm7LdFSFlUIoJSClDMjY2VrYnzpSWltKXX35J4eHhird/aGgoTZo0iYqLi2X5rqx6LhWSRK596H1/bevWrav2s2azmT744APFd857p/tvYazNOUi5QlKj0dT48jZHmM1m+vXXX+mZZ54hLy8vp7e5TqejLl260K5du6iyslK278mq53IhSSQ+KJV2f13btm2r9rMWi4XGjh2reDBW1ZabN292KCClDEngzp0ecl9IKCwspCVLllDLli2dclFHEASKjY2lGTNmeNRFEjVyyZAkqv2AczUE5f01jRs3rtoebmlpKfXu3VvxUKxqatCggcOvZZWqDYE7owOWLFkiepn2slgsdPXqVZo5c6ZsYWkNx3/+8590+fJlVRwBeTqXDUki1wvK++uJiYmh48ePP/A5i8VCq1atIn9/f8UDUa5Jqja0Tm3btqUbN244snlq5dq1a7R69Wp67bXXKCYmxu6HR1c1abVaioqKohdffJGWL19OGRkZHI4qopqnAIlVUFCA9957D8uWLYPZbK7x81FRUZg/f77Tnh50r6qeAvTII4/g888/R7t27eDn54fr169j/fr1GDdunMNvBlQzsT+76p6kpNPpMH36dKSkpDhSVq0REa5cuYKTJ09i+/bt2LdvH27cuIHCwkKUlJTAaDSisrISRAQvLy/o9Xr4+voiICAAdevWRbt27dC1a1ckJCQgMjLSvZ6e4yZcPiSB34Ny+fLltXrMWteuXZ36+LLq1hUYGIhmzZohKCgI2dnZOHfunNs83Lc6UockcOcFXxs2bEB0dLTYshxWWVmJmzdv4vr16ygoKIDRaERFRQWICAaDAQaDAYGBgahbty5CQ0Pd6+G0bsotQhJwjaBU6/MklSBHSAqCgLfffhv//ve/7XrSPWP2cJtn1wcFBWHatGl47bXX7HpdbUZGBt58801s27ZNFa+rZY4jIqxcuRK7d+9WuhTmRtwmJAEOSnbniGLcuHG4ceOG0qUwN+FWIQncCcqpU6fWOih37NjBQekm0tLSMGXKFBiNRqVLYW7A7UISAIKDg2vVo7xy5QrWr1/v9hdLPIXJZMKXX36JTZs28R8+5jC3uXBTFXsu5mi1Wrz44ouYOXMm6tatK2s9fOHmd3JcuLlfXFwc1q1bJ/t7ppl7c8uepFVN5yi1Wi1eeuklzJo1S/aAZM539uxZfPDBB7h165bSpTAX5tYhCfwxKO8dqKvVavHyyy9j1qxZCA0NVbBCJhciwrp16/Dpp5+irKxM6XKYi3L7kAR+D8rXX38dWq0WWq0WgwYNwowZMxASEqJ0eUxGZrMZ8+fPx+zZs13y3dpMeW59TvJ+BQUFGD16NIgIkydPRlBQkFPXz+ckf+eMc5L3CgwMxIwZM/Dqq6/yrX+sVjwqJAGgtLQUAODr6+v0dXNI/s7ZIQncGfUwY8YMvPzyyxyUdrh27Rrq16/v8W3lEYfb9/L19VUkIJnybt26hREjRmDNmjWwWCxKl6N6y5Ytw86dO5UuQ3EeF5LMs+Xn5yMlJQWrVq3ic5Q1uHnzJqZOnYrbt28rXYqiah5pzZibycvLwzvvvIO8vDy89dZbMBgMSpfkVLm5uXZ9rqSkBHv27MG5c+fQunVrmatSLw5J5pFu3bqFf/zjH7h69SrGjBkDf39/pUtyGnsH11dUVMBiseDAgQMckox5otLSUkybNg3Z2dkYP348IiIilC7JKYqLi+3+rCAIyMzMlLEa9eNzksyjGY1GrFixAs8//zx2797NF3TYAzgkmcezWCw4ePAgBgwYgDlz5vDdOffQaDRo2LCh0mUoikOSsf+Xm5uL0aNHY8iQITh9+rTS5aiCl5cXHnnkEaXLUBSHJGP3KCsrw8qVK9GnTx/MnTu3Vufv3FGbNm3QvHlzpctQlMfdcaMkvuPmd0rccVNbXl5e6NKlC1JTU9GxY0e3eWmXvW0YEBCA5cuX45lnnpG5InXjniRj1TAajdi4cSP69euHt956C0eOHLHrtcXuwMvLCyNHjkTPnj2VLkVx3JN0Iu5J/s4VepL3q1+/PgYOHIjBgwfj4YcfdtmeZU1tGBoaihEjRuC9997jW3jBIelUHJK/c8WQtKpbty769u2LN998E23atHG5sBw9ejS+//575Obmory8HCaTCQaDAf7+/njkkUcwcuRIPPnkk3a9+sSVWSwWaDQ1H0xzSDqRGnZwtXDlkLQKDAzEtGnTkJycrKq6arJ48WI8/fTTuHTpEq5evYqSkhLUq1cPcXFxiImJcbnQF6OsrAwrVqzAG2+8UeNnOSSdyJV2JLm5Q0gCQEJCAjZu3IjGjRsrXYrdfHx88Pbbb+Pjjz/2yIdO37p1C+PGjcPcuXPtGhPLF25qie/IYPc6deoUli5d6lJvZSwvL8fMmTMxaNAgnDlzRulynIaIcOHCBQwZMgQzZ85EeXm5XfNxSNbSjz/+qHQJTEUsFgsWLVqE8+fPK11KrZjN5rtX7jds2OD2j40zm81Yv349nn32Waxdu7Z235eY3QoKCuipp54SPT8Anv5/crc2TEpKopKSEtHfy5nurz0oKIhGjhxJmZmZSpcmi6tXr9Lo0aMpMDBQ1O+QQ7IWVq5cSQaDQfT8Su/IaprcrQ0NBgP9+9//JovFIvq7OUtV9Ws0GoqPj6cVK1ZQWVmZ0iVKoqysjL755htq1aoVabVa0b9DDkk75ebmUocOHdxyB+eQlGaqX78+bd26VfR3cxZb38HX15f69+9Pv/zyC5lMJqVLFcVkMtGBAwfohRdeIB8fH4d/hxySdjCZTDR+/Pi7f43EUnonVtPkrm2YkJBAhw4dEv39nMGe7xEWFkbJycm0Z88eKi8vV7pku5SXl9P+/fvprbfeorCwMMl+hxySdjhy5AiFh4e7/Q7OISldUP7666+iv6PcavNdAgIC6IUXXqANGzZQcXGx0qVXqbi4mHbv3k0vvfQSBQUFSf475HGSNSguLkZycjJWr15997+JbTK1jfFTkru3YfPmzbFgwQI8+uijqqtZTD0+Pj7o1KkTBgwYgCeeeAIRERGKvhvIaDQiMzMTmzZtwrp167Bv3z5RLyyz53fIIWmDxWLBrFmzMGrUqD8MGXD3HdwZPKENmzRpgs8//xwvvvgivL29lS7nLkfaUBAEhIWF4fHHH0fv3r3x5JNPokGDBrLf401EuH37Nm7evIn9+/dj7dq12LVrF3Jzcx0ao8oh6aA9e/bgpZdeQk5Ozh/+uyfs4HLzlDb09fXFO++8g3/84x+oW7eu0uUAkK4NtVotQkND0aFDB7Rr1w6tWrVCXFwcQkJC4OvrK/oPg8ViQUVFBSoqKlBUVITDhw9j3759OHToEI4dO4abN29K9jQmDkkHZGVl4bnnnsPhw4cf+DdP2cHl5EltqNVq0bVrV8ycOdPuNxXKSY421Gg00Ol00Ov1iIqKQlxcHB5++GE0adIEjRo1gre3Nxo0aAAfHx+EhoYiICAA5eXlyM3NhcViQX5+Pm7cuIGMjAycPXsWly9fRmZmJjIyMlBUVITKykpZ7mrikGSMMQfxbYmMMWYDhyRjjNnAIckYYzZwSDLGmA0ckowxZgOHJGOM2cAhyRhjNnBIMsaYDRySjDFmA4ckY4zZwCHJGGM2cEgyxpgNHhGSFosFLVq0gCAIkkxiSbV+KSe9Xo+RI0eivLwca9euxdSpU2E0GiVsfWkp3V73Tm3btsWVK1eUbpJaU7rd1DTZwyNCklVNq9Vi6NCh+Oyzz+Dt7Y2+ffsiOjoay5Ytc/v3MEvh8OHDGDNmjKgnYjPXwSHpobRaLd5++22MHz8egYGBd/9b37590aBBA3z33XeSPdjUXWm1WjRs2BB6vV7pUpiMOCQ9kDUgP//887sBee+/9erVCyEhIdi5cycsFotCVaqbXq/H8OHD8dFHHyn6rhcmP53SBTDn0ul0GDp06B96kPfTarXo3r079u/fj0uXLiEmJsauZRcWFuLixYs4cuQIjh8/jvz8fOTn5+PWrVvQ6XQIDg5GeHg4IiMjMWbMGCm/llNZA/LTTz+Fn5+f0uUwuUnzUkd1M5vNlJCQ4PGvQ9Xr9ZSSkkKFhYV21WuxWOjChQs2XyVqMpnoyJEjNGLECGrVqhWFhISQIAhu24ZeXl40evRo1b5e1R5Kt6GaJrvaS+btoQockncCctiwYVRUVFTruqt6Ob3ZbKZjx47Rm2++SfXr1/eINgRAjz32GOXk5IiuXw2UbkM1TXa1l8zbQxU8PSS9vLxo+PDhkvV+iouLac6cOdSwYUOPaUPrpNfrKTk5ma5evSpJWypB6TZU02RXe8m8PVTBk0PSy8uLRowYIVlAXr58mV5++WXy8vLymDa8f9JoNNSlSxc6ffq0JG3qbEq3n5ome/DVbTem1+vxzjvvYNy4cZJdYEhPT8eWLVtUPeBcbhaLBdu3b8drr72GM2fOKF0OkxmHpJuyBuTnn38u6RXY3r17Y/LkyQgODpZsma7q4MGDGDJkCC5evKh0KUxOMvfsVcHTDrd1Oh29++67dl/Fri2TyUSLFi2ioKAgt21DeydBEKhz586UmZkpYQvLS+k2U9NkV3vJvD1UwZNCUqvV0t///ncqKCiQsAUfVFlZSQsXLqTAwEC3a8PaToIg0MCBA11mWJDS7aWmyR58uO1GqrrVUC46nQ6JiYmYMmWK7OtSOyLCmjVrsHDhQtzJIOZWZP2TpRKe0JO09iBv3bolXcPZobKykhYsWFDrQ2+x5GxDR6cGDRrQrl27JGxdeSjdTmqa7ME9STeg0+nu9iCDgoKcvu6kpCRMmTLF4y/m5OTk4JNPPkFBQYHSpTAJ8b3bLk6v1999WEVAQIAiNViDUqPRIDU1FTdv3lSkDrEMBgOMRqMkh8r79+/Ht99+i6FDh0pQmfro9XoEBQUhJCQEderUgZeXFwwGA4gIRqMR5eXluH37Nm7duoXCwkK3eOQeh6QLu3eYj7+/v6K1aLVaJCYmQhAEpKamIj8/X9F67BUVFYVx48Zh6dKl2LFjh8NBaTQaMWPGDPTu3RuNGzeWqErl6HQ6hIeHIy4uDl26dEG7du1Qr149BAcHIyAg4A8hWVFRAaPRiKKiIty8eRO5ublIS0vD9u3bceHCBeTl5bnmU6VkPPWhGu54TlLqWw2lYjabaenSpRQaGqr6NoyOjqadO3eSxWKhjIwM6t69u10P56hp0mq1NG7cODKbzRK2rHRqql+n01FERAT169ePli9fTufPn6fKykrR6ysvL6fjx4/TnDlzqFu3bhQWFkYajUbx85H2/g45JF0wJKW+1VBqFouFvvrqK6pbt65q2zAmJob27Nnzh2VnZGRQt27dJNmB4+LiKDc319GmlIWtcG/atCmNHTuWzp8/TyaTSfJ1G41GOnr0KKWkpFB4eLgkf5Q4JCXgTiGp1+tV2YO8n8VioWXLllXboxRLijaMjY2lvXv3Vrl8a1A6uvN6eXnRggULyGKxiP6ucqmq3rCwMPr444/p8uXLTqnZZDLR8ePHKSUlhYKDgzkkleYuIWl93JnaA9LKeugdEhKimjasqgd5P6kOvTt27KjKbXVvjVqtljp37kz79u1T5PRAZWUlbd68mdq2bavIIbg9OCRdJCSttxqKeR6kkkwmEy1evJgCAgIUb8OYmBjau3evXT0lKYIyICCA0tLSRH9XuVjr8/f3p9TUVMrLy1O6JMrMzKQhQ4aQt7c3h6QSXD0knXWroVwOHz5MUVFRirZhbGwspaWl1epQ8vLly9S1a1fRQanRaGj8+PGqO+QGQBEREbRq1aoqH6islJKSEpo9e3aNF/04JGXgyiHp6gF5/Phxat26taJt2KxZM9q3b5+osPrtt9+oS5cuon8rXbt2Vd0hd3R0NG3btk2VV99NJhOtXLmS6tWrxyHpTK4akkrdaiiV48ePU9u2bRVtw9DQUNqzZ49DvblLly5R165dRf1WQkJC6Pz586LXLYdffvlFdb3be5lMJlq1apVTepT24NsSVUrJWw2lcOrUKSQnJ+PgwYOK1lFSUoLt27ejoqJC9DIeeughLFiwAN26dYMgCLWat7S0FCdPnhS9bjm0a9eu1t/DmbRaLfr164dJkyYpfpMEAAf+pLsQV+tJ6vV6evfdd136ELu6HqSz2vDeycfHhz799FMqKytz6HtdvnyZevToUatzlBqNhiZMmODQej2V0WikyZMnk8FgULQnySGpspC0DvOR64G5cjt+/Di1adNG8T8090/e3t6SBGVGRgb17NmzVkGZnJxMRqPRofV6qpKSEkpKSpJteJA9PCIkLRYLDRkyRPUhqdZbDe117NixKi/SqCEkpQzKrKws6tWrl907bseOHV32j54aZGZmUqtWrTgk5bZnzx6HXjcg9w6u9lsNa5Kenl6rH7JYjm47KYOyT58+dgVlXFwc3bhxw6H1ebrVq1eTn58fh6ScKioq6JlnnlFlSOr1eo8KSCVDErhzjvKzzz5zeIxgdnY29e3bt8agjIyMVMWAbVdWVlZGL7/8suT3etvDY0KSiGjt2rWSnAQWq7qAdOVDbDEBqXRIWoNy7NixkgRlTT3KevXq0bVr1xxaD7tzU0JYWBiHpJwKCgqoY8eOqglJV73V0EpsQKohJAGQr6+vJEGZlZVFvXv3rraXExAQQNnZ2Q6tg90ZPzl06FAOSbktWrTI4StlYt27DK1WSykpKS57Qv/YsWMOnUwXS8odROqg7NOnT5VByT1J6Rw4cMDmI/g4JCXw22+/UdOmTRUNSXe91dAVQ1LKoMzOzqbnnnvugaCMioric5ISqayspJ49e3JIyslkMtFLL72k2A7uDrca2jMO0pVCUuqgfPbZZ/8QlAkJCZSfn+/QctnvFi9eTFqtlkNSTsuWLVNkB9fpdC7dgzx58mSNd9K4akgCv1/1lmJ40L09yqeeesplzzur0YULFygiIoJDUk7nzp2j8PBwp+/gw4cPd9mAtOdWQ1cPSWtQSjGOMjs7m/r160eCIPAdNxIzmUwOPZmptr9Dj3zARUREBNq0aeP09U6YMAGBgYFOX6+jTpw4gaSkJMUfVuEMZWVlmDhxIr744guUl5eLXk7Dhg0xe/ZsvPTSS4iNjYVer5ewSs+m1WrRsWNHp63PI0PS29sbPXr0cPqTUHx8fJy6PikcP34ciYmJOHTokNKlOE15eTkmTpyIiRMnOhSU4eHhmDNnDvr37y9hdQwA2rZtC43GOfHlkSEJAE8++aRLhpYzHT16FK+//joOHz6sdClOV15eji+++MLhHmVISAhiYmIkrIwBd3rqzjoq89iQrFu3LiIiIpQuQ7WOHj2KxMREHD16VOlSFFNeXo5JkyZh4sSJDj2PkkkvNDQUwcHBTlmXx4ZkYGAgHnroIaXLUCVrQB47dkzpUhR37zlKDkr18Pf3h5+fn1PW5bEh6evri+joaKXLUB0OyAeVlZVxj1JlvLy8nHYxTOeUtahUQkICtFotzGaz0qWoQnp6OpKSkhwOyLi4OPj5+eHSpUt3z+cJggC9Xo+CggIJKnW+0tJSTJw4EQDwwQcfwGAwKFyRZ/P29oa3t7dT1uWxPUngTkg66wqZ2h07dgxJSUlIT093aDkajQaDBw/G1q1bceLECZw9exZnz57FoUOHRL0jRk2sQck9SuU583fk0T3JJk2awN/fHzdv3lS6FEVZh/lIcZFGEAQ0btwYQUFBd19gVlFRgfnz5+OHH34AETm8DiVxj1IdjEYjTCaTU9bl0d0og8HgtCtkAJCRkeG0ddnrxIkTSE5OluwqtiAIaNCgwd3/32QyYdasWZgyZQqMRqMk61BaaWmpJMODmHhGo9FpvXmPDkmdTue0K2QAMHDgQJw6dcpp66uJHAPF9Xo9GjVqBOBOQC5ZsgRjx45FWVmZZOtQA6nuzGHiFBcXo7S01Cnr8viQdOZ7fffv34/ExEScOHHCaeuszrFjx2QZKN6wYUMEBASgrKwM06ZNQ2pqKoqLiyVdh1pwUConPz8f+fn5TlmXR4ekVqt1ak8SAA4ePIjk5GRFgzI9PV22geJRUVGoqKjA6NGj8cknn6CoqEjydaiJ9RbGCRMmcFA6UU5OjtNGSnh0SOr1eqeHJKBsUMp9J42XlxdGjBiBuXPneswVYOudOTNnzlS6FI9x+PBhpw3d8+iQdPbh9r2sQXn69GmnrdMakI4O87Fl69at+OGHHzxu7KlOp0NYWJjSZXgEk8mE3bt3O219Hh2SGo3GaQNSq3Lw4EEMHjwYZ86ckX1dzrqTxmQyufwwn9rSarUYPnw4Xn75ZaVL8QiZmZlO2WesPDokLRaL4ueR9u/fj+TkZJw7d062dfCthvIRBAHPPfccUlNT+ZmRTpKWlobs7Gynrc+jQ9JsNqtiaMqBAweQnJyM8+fPS75sqW41ZA8SBAHdunXD9OnT7w6cZ/Iym81Ys2aN0waSAxySThtrZQsRYd++fZIHpVS3GrIHaTQa9OzZE4sWLeJH7jnR6dOnsX//fqeu0+NDUg09SeBOUKalpUkWlFLeasj+SKvVol+/fpg/fz4aN26sdDkew2KxYMWKFcjJyXHqej06JE0mk6oGOluDcvDgwQ4FpfWdNByQ0vP398dHH32EBQsWcEA62eXLl7Fy5UrnXxiU7ZVmLuDatWsUFRWlujf9CYJAjz/+OJ09e7bW6zh27Bi1bt1a9rcKOjrJ3YZyTNHR0fTtt9/ymw8VUFlZScOGDSONRuP036FHh+S5c+fIx8dHlTu4IAj02GOP0ZkzZ+xefnp6uksEpKuFpLe3N/Xv35+OHTsmum6pWSwWpUtwqp9++olCQ0MV+R16dEhu3ryZ9Hq9qnfwDh060OnTp2tc9tGjR6lVq1aKh58a21DsJAgCJSQk0LJly6ikpER0zXL46aeflC7BaW7cuEFdu3ZV7Hfo0SE5e/Zs0mq1qt/BawrKI0eOUMuWLRUPPjW3YW0mnU5HsbGx9Mknn1BGRoboWuVSWlpKL7zwgtJlOEVFRQWNHj1a1H7KISmBkSNHkiAILrGDd+jQocpzlEeOHKE///nPioees9qwf//+1KhRI1FHALYmQRAoKCiI2rdvT3PnzqXs7GzVHtLu3LmT/P39lS5DdmazmVatWkX+/v6K/g49NiTLysqob9++Tt3BHd2gHTt2pHPnzt1dnqsGpCNtWFlZSZcuXaLFixdTYmIixcbGUkBAQK17GoIgkK+vL9WvX586depE48ePp71791JxcbFqw5HoTi+yf//+DrWhK7BYLLRlyxaKiIhQ/HcoEHnYjbb/LycnB507dxZ1D6jYJnP0vRyCIKBTp074n//5H9y+fRsDBgxw2TtppPjZmUwmlJSU4NSpU0hPT8eZM2dw6tQpXLx48e5DWUtKSmAwGODj4wMfHx8EBASgRYsWaNOmDRISEhAfH4/w8HAYDAaXeP9OcXExEhMTsXbtWlgsFlHLyM3NVf3DOA4cOICBAwfi8uXLsq7Hnt+hx4YkY4zZw6MHkzPGWE04JBljzAYOScYYs4FDkjHGbOCQZIwxGzgkGWPMBg5JxhizgUOSMcZs4JBkjDEbOCQZY8wGDknGGLOBQ5IxxmzgkGSMMRvsDklBEFQ9Pfzww0hPT8f169fx17/+FVqtVrZ1KeHKlStISEiwqz5fX1/MnTsXZrO52uXNnz9f0e3Faq+oqAhjxoxBcHCwQ23ozO1sMBjwt7/9DUVFRRK2RNXOnTuHp556qlb7vl3sfQgmVPCg1uqmZs2a0b59++7WmpmZST169JD8zWrWSQlGo5EmT55MderUsavGoKAgmj9/PplMpiqXN2/ePEW3GRPHaDTS1q1bqXPnzqKX4axt7OXlRR988AEVFxdL2AK2ZWZmUq9eveze9+3h8iEZGxtLaWlpD9SblZVFPXv2FPV6BrXu4BUVFTRx4kTy9fW1q87g4GBasGABmc3mB5bFIenaCgsLRc/rjO0rCAK9/fbbTg1Iq4yMDHrsscck+x26dEjGxMTQ3r17q61ZrqBUUm2DMiQkpMqg5JD0XM7Yvt27d6ecnBzFvuOhQ4coOjpakt+hy4ZkTQFplZWVRb169ZI0KJVWXl5e6x7lwoUL/xCUHJKeS+5t27x5czp16pTSX5N++OGHGt/VbQ+XDMmYmBjas2eP3Y2VnZ1NvXv3liwo1aCiooImTZpk9znKkJAQWrhw4d1zlBySnkvO7err60vr169X+isS0Z23LX766ac2z0/aw+VCMjY21q4e5P2kDEq1qKiooMmTJ5Ofn59ddVvPUVZWVnJIejA5t2v//v2ppKRE6a94V0ZGBjVr1sxzQrJZs2ZVXqSxl1RBqSZGo5GmTJli97uJrVe9Z86cySHpoeTapoGBgfTTTz8p/fUeMG3atGpfOWwPlwnJ+4f5iJWdne3wOUq1qayspGnTptkdlIGBgfTkk09ySHooubbpoEGDqLy8XOmv94CrV69SfHy8e4dkXFwc7d+/X7KXxjt61VuNKisradKkSXYHpdITU44c29Pb25tWrVql9Fer1siRI903JK09SKkC0soalGIGnKuV0WikSZMm2X2OkkPSM8mxPRs1akTZ2dlKf7Vqbdq0iXQ6najfoarv3Y6NjcXSpUvRvn17yW9la9y4MRYsWIBu3bpBo1F1M9hNr9djxIgR+Pjjj1GnTh2ly2EepGPHjqhbt67SZVQrPj4eTZs2FTWvatMhJiYGS5cuRYcOHWS719fdg9LX11fpcpgH0Gq16NixI7y8vJQupVoNGzZEixYtRM2rymSIiYnBkiVL8Oijj8q+roiICCxcuBDdu3d3mwcveHl5YeTIkfj444/h4+OjdDnMzRkMBkRGRipdhk0ajQYPPfSQqHl1EtfisOjoaAwaNAjZ2dn47rvvnLbebt264dChQ7h+/brT1ikna1ASEcaNG4eysjKlS2JuSq/Xo169ekqXUaNGjRqJmk9VIRkdHY2lS5ciNTUVn332mdLluDyDwYD33nsPADgomWy8vLxcIiQbNmwoaj7VhKT1EPvxxx93m8NeNbAGpSAIGDduHEpLS5UuialUy5YtERwcbPfnb9++jYMHD0Kv16v6oo1VgwYNRM2nipCMjY3FokWL0LFjR6VLcUv3B2VJSYnSJTEVmjx5Mrp162b3548fP45WrVoBAIhIpqqUp/iFm2bNmmHp0qUckDKznqP85JNP4Ofnp3Q5zI2YzWbcvn1b6TJqJPbp6IqGpDUgnXEVm/0elB999BEHJZOM0WhEXl6e0mXU6MaNG6LmU+xwOy4uDkuWLEH79u2VKsEj6fV6jBo1Cnl5eZg+fbrS5dTalStXFFt3UFAQ/3GpgtFoRE5OjtJl1Ojs2bOi5lMkJJs1a3Y3IPkijfPp9XrExcUpXYYosbGxiq17zpw5SEpKUmz9aqTRaNCzZ0+0a9dO6VJsMpvNOHXqlKh5nR6S1kNsDkhluWrbKzmMyWQyKbZuNRIEAc8//zxmzpyp+iFA165dw5kzZ0TN69RzkrGxsViyZImstxoy+8THx4seXMsYcGdc86xZs1C/fn2lS6nRqVOncPnyZVHzOi0kY2JisGjRIr5IoxIdOnTAl19+yUHJRPPx8VF9DxK4Mzzpu+++g9FoFDW/U0KyadOmWLx4MQ/zURGNRoO//vWv+PLLL0XficCYK7hw4QK2bt0qen7ZQ5IDUr00Gg169+7NQcncFhFh8eLFyMrKEr0MWUMyOjoaS5YsQadOnfgcpEoJgoA+ffpwUDK3lJ6ejuXLlzt0R5BsIWm9F/uJJ57ggFQ5DkrmjoqKivDpp58iOzvboeXIMgSIH1ZRtb59+1b7bwaDAW+++SZ69OhR5b+npaXh3//+9x8eUPHQQw/h448/luTkuTUo582bh6FDh+Lq1asOL5MxpZhMJsyZMwcbN250fGH2viMCdr7ron79+rR7926H3knToUMHxd/BYmsSq7rleXl50ahRo2y+r7i8vJzGjRtHPj4+BICioqJox44dkr/7x2w20/r166lRo0Yu1YbOmBYsWCBhSyunuu+3ZcsWpUuThMlkoqVLl1JgYKAkv0PJD7fLyspw7tw5HnhrJ71ej3fffRefffaZzdctGAwGvP/++xgzZgyaN2+OJUuW4KmnnpK8p37vVW8eHsRcTWVlJVauXIlRo0ahsLBQmoXam86oxV/cwMBAWrBgARmNRlF/CTylJ6nX62nkyJFUXFxs9zLKysro8uXLkvcg72c2m+nHH3+UrUcplpLbnXuS6mY0Gulf//oXBQQESPo7lOXCTWFhIVJTU/HVV19xj7IaOp0Ow4YNw3//93/X6qEJ3t7eiIyMlP1cr/WeXO5Reg5X31dNJhMOHTok/fNS7U1piPjLGxgYSAsXLqTKyspa/UVw956kTqejkSNHUlFRkehlOYtc5yjFUnK7u3tPslevXnTo0CGly3PI9evXacCAAaTVatXdk7S6t0dpNpvlXJXL0Ol0ePfddzF27Fj4+/srXU6N+BylZxAEAQaDweW3cd26dTFr1iw8//zz0Gq10izU3oSGA3+Bg4KCaNGiRWQymexal7v2JPV6PY0aNYoKCwtFL0Mp1h5lw4YNuSfp4qr6bs8++yxdvXpV6dIkk5eXRy+++CJpNBqHf4dOeVRaQUEBUlNTIQgCEhMTodEo/tYIRbz33nv4r//6L5d8cOu9tzC+/fbbio2jfOaZZxRZLwDVv1tarH79+mH27NkIDw9XuhTJ1KtXD7NnzwYArF69GhaLRfzC7E1mSPCXODg4mBYvXlxjj9Jde5JlZWWi51ULi8VC69atc7hHyZRj3QaCIFC/fv3cqgd5v7y8POrfv3+1PUp7ODUkrUFZ06G3u4aku5AiKJlyPCUgrXJzc6sNSns4PSSBO+cobV315pBUP4vF4tA5SqYcjUZDL7zwAl27dk3pUpymuqC0hyIhWVNQcki6BmtQihkexJTz0ksvUU5OjtJlOF1ubi7169fvD0FpD8VCEqj+zhwOSdchdhwlU86tW7eULkExOTk59Nxzz90NSnsoepn53nGUlZWVSpbCROJxlK4nKChI6RIUExYWhrlz5+KZZ56xe5SN4mNx+BZG18dByVxJgwYNMHfuXJuPLryX4iEJcFC6Aw5K5koaNGiAxYsX2/VZVYQkcCcoR40axUHpwjgomSsJCQmx63MCkX0vf3DWE8aDgoJgNBr/8ARutbGzyTyWxWLBhg0bbD7hnNuQuQrVhaQr4B28ZkSEH3/8sdqg5DZkrkI1h9vMvQiCwK+rZW5BtT1JHx8fREZGolWrVmjdujUaN26MkJAQ3Lx5E1euXMGhQ4eQnp6Oy5cvo7y83Km1uVMvKDc3F2fPnkVaWhouXbqEK1euwGAwoHHjxoiNjUWHDh3QtGlThIaGilo+EeF///d/H3gohju1IXNz9g7ChJMGant7e1OXLl1o7dq1lJubW+PA0O+//546d+5M3t7ePJhchK+//pqCg4Or/a516tShtm3b0syZM0XfpVHVLYyMuQpVhWR4eDgtXLiwVu98ISIqLi6m+fPnU3h4OIdkLd3/FsbqJo1GQ4888gjt2LGDzGZzrddz/y2MjLkK1YRks2bNaPv27aJfcGWxWGjLli0UGxvLIVlL1qC0pzceHh5OK1eutPsByve69xZGxlyFKkKycePGtHv3bkm+0K5duyR7eranhCTRnWddjh071q6gDA0NpbVr14r6g2Y2m2nbtm0yfANmL7nftOluFA9Jf39/WrlypahDuKqYzWZavnw5+fn5cUjWUmlpqd1BGR0dTUeOHFG6ZCbCmjVrRB0JeCpFQ1IQBBoyZAiVl5dL+qXKysooMTGRQ1IEa1AaDIYa2+G5556TfNsx+dWrV4+++eYbDko7KTpOMiQkBO+88w4MBoOky/X29sbf/vY3BAcHS7pcT+Dj44PU1FR8+OGHNW6XHTt2YPfu3U6qjEnl+vXrePfdd/Gf//yH32JqB0VD8rHHHkN8fLwsy27RogUee+wxWZbt7nx9ffH+++9jzJgx8Pb2rvZzRUVFWLVqFe9oLsgalA6/JMtFmc1mrF271q7PKhaSWq0W3bp1s7kTOsLb2xtdunTx2DczOsoalB9++KHNbbRr1y4UFRU5sTImlevXryMlJQWrVq3yqKA0m834z3/+g7feesuuzyuWIHXq1MFf/vIX2ZYvCAL+8pe/oE6dOrKtw935+Pjc7VH6+PhU+ZkbN27gzJkzTq6MScXao/SUoDSbzVi1ahVSUlJw/fp1u+ZRLCQNBgPCwsJkXUeDBg3g5eUl6zpqY9++fS53O15NQVlRUYHc3FwFKmNS8ZSgNJvN+O6775CSkoIbN27YPZ9iIanX61G3bl1Z11G3bl3o9XpZ11Ebr7zyCnbv3u1yQent7V1tUFZUVNTqB8fU6d6LOe4YlGazGStXrkRKSgry8/NrNa9iIUlEsj9cV20XFC5fvozk5GTs2rXL5YLSYDBUGZSCIECr1SpYGZOK9RzlmjVr3CoorQE5bNgw3Lx5s9bzKxaSlZWVuHXrlqzryM/PV90Lxi5fvozBgwe7fFBaL+Z4eXnxUCs3cv36dfz973/HmjVrVNfJEMNsNuPbb7/F8OHDReeNYiFZWlqKc+fOybqOc+fOqfIJ564elKmpqXeDsk6dOoiJiVG6LCahvLw8/P3vf8f333/v0kFpPQcptgdppWhIHjx4ULZuvcViwcGDB1FWVibL8h1lDcpffvlF6VJqzdvbG6mpqUhNTUVkZGStQzIvL0+myphUrEG5YcMGl/tDDtw5nbdu3ToMGzas1ucg72f3Q3cZY8wT8UhrxhizgUOSMcZs4JBkjDEbOCQZY8wGDknGGLOBQ5IxxmzgkGSMMRs4JBljzAYOScYYs4FDkjHGbOCQZIwxGzgkGWPMBp3SBdzr6NGjePnll1X/zhR3fCZIWloaHn/8caetzx3b0FWMGTMGOTk5yMnJwbVr13Dr1i2Ul5ejrKwM5eXlqKyshJ+fH/z8/FCnTp27/zssLAxxcXGIiYlB06ZN8dBDDyE0NBR6vV5VbwCQmuqeArRv3z4kJSXJ/qxJR6isySTBIek5BEFwaH69Xg9vb294e3sjLCwMbdu2Rfv27dG6dWvExcWhTp06bvW0etWFJHBnh01MTMSFCxeULqVKKmwyh3FIeg5HQ7IqWq0WAQEBqF+/Ph555BE88cQT6NSpE6Kjo6HTqeqAtdZUGZIA8PPPPyM5OVmVQanSJnMIh6TnkCMk76fValG/fn20b98ezz//PJ544glERkY6Zd1SU21IAvIHpV6vR3x8PNq3b4/HH38cDRs2hL+/P/Lz85GdnY309HTs378f586dQ1FR0d35VNxkonFIeg5nB5VOp0Pjxo3xxBNPIDExEY899li173FXJVK5PXv2UExMDAGQbNLr9dSjRw/64YcfKC8vz+b6CwoKaN++fTRkyBAKDQ0lF2gyUfbu3StpG9c0MeU4czvfP/n7+1OvXr1o7dq1VFBQoHRT2MUlfq179+6l2NhYSTZSvXr1aOrUqVRYWFirGkwmE23bto1atWol07dUFoek51AyJK2TwWCgTp060fLly+nWrVtKN4lNLvNr3bdvHzVr1syhDRMZGUnr1q0ji8Uiuo6srCwJv5V6cEh6DqUD8t5Jp9NR9+7daffu3WQ0GpVumiqp+pzk/Q4cOICkpCRR4ygbNmyIr776Cl27dpWhMtfH5yQ9h/WcpF6vh6+vL3x9fREQEICmTZsiIiIC/v7+d8dG+vv7w9fXF2VlZSguLsbt27dRXFyMwsJCXLlyBb/99hsKCwtRWlqK0tJSmEwmUTUFBwdj8ODBGDZsGCIiIlR1gcelQhIQF5QGgwGzZs3CkCFDVNX4asIh6Tn69u2Lli1bIj4+Ho0bN0ZERATCwsKg1+uh0dy5Cc+6n9y7v1i3mfX/WiwWGI1GXLt2DVlZWcjKysLx48eRlpaGCxcuoKioqFavdBYEAfHx8fjggw/wwgsvqObijsuFJBHhl19+qVVQ9u3bF19//TX8/Pxkrs51cUh6DqPRCJ1OdzcQpWSxWFBZWYlr167hyJEjSEtLw5YtW3Dx4kWUlpbatd19fHzwyiuv4LPPPkPjxo0lr7G2XC4krQ4cOIDExEScPXvW5ucMBgPWr1+Pbt26Oaky18QhyeRgNptx+/ZtHD16FJs3b8bWrVtx+vTpGgNTo9Ggbdu2mDx5Mh5//HFF7+CRPSTPnz+PDRs2SL5ca49y1apVNs+DtG/fHhs3bkRwcLDkNbgTDkkmN4vFguLiYpw+fRrffPMNVq9ejatXr8JisVQ7T/369TFmzBgMGTJEuSNBua8Mffvtt4pePevYsSNt2rSJ8vLyyGw2y/11XRZf3WbOVFlZSefPn6exY8dSXFwcaTSaan8r3t7eNGjQIMrOzlakVtl7kt999x0GDBgg5ypq5OfnhwYNGqB9+/bo1q0b2rRpg9jYWHh5eSlal5pwT5IpgYiQnZ2NtWvXYt68eTh58mSVvw1BEPD0009j9uzZaNasmdOLlJXSPcn7J41GQw0aNKCuXbvS119/TTdu3JC7CVwC9ySZ0jIzM2n06NF372yramrTpg398ssvTq3L4x66a7FYkJOTg23btiEpKQk9evTAjBkzkJGRwb0bxhQUERGB8ePHY926dejduzcMBsMDnzl06BBeeeUVbN++3Wn7q8eF5L2MRiMOHTqEESNGoHv37pg6dSquX7+udFnMhl9//ZX/mLkxnU6Hxx57DN988w2mTZtW5RCgCxcuIDk5GTt27HBKTR4dklZEhHPnzuGDDz7A888/jy1btsBoNCpdFqvCK6+8gr1793JQujk/Pz+88847WLNmDbp06fLAMymzsrLw9ttv45dffpG9FllD0jo+ylVYLBb8/PPP6N+/P9577z1cunSJd0aVuXDhApKSkvDzzz/ztnFzgiCgXbt2+Pbbb/H+++8/MIzvwoULGDp0KE6fPi1vIXKd7MzKyqKkpCQyGAyKX6wRMwmCQM2aNaM1a9ZQZWWlXM2kGq5y4cY6f3R0NP30008OPayEuY7Kykpau3YtNW3a9IHf0lNPPUWZmZmyrVvykDSZTLRr1y565JFHbI59cpUpODiYJk+eTGVlZVI3laq4WkgCoJiYGNqzZw8HpYewWCx04MABatWqFQmC8IcOTb9+/WR7PqWkIVlRUUHz58+nsLAwxcNNysnHx4dSUlIoPz9fyuZSFVcMSWtQ7t27V8KWYGp35swZ6ty58x+CUq/X02effSbLUZ9kIVleXk7Tpk0jPz8/xUNNjkmn09Gzzz5LFy5ckKrJVMVVQxIANWvWjPbv3y9hazC1y8rKov79+/8hKIODg2n9+vWSH1lIEpJlZWU0fvx48vX1VTzM5JwEQaCEhAQpmkx1XDkkAVBcXJzTBxkzZV29epX69Onzh99B8+bN6fz585Kux+GQLC0tpU8++YR8fHwUDzG17+Bq5uohad1BOCg9y2+//UadO3e++xsQBIEGDBhAxcXFkq3DoSFAJpMJ06dPx8SJE2v1cE3G5HDq1CkkJSU5ZewcU4eoqCjMmzcPHTp0AAAQEdauXYsVK1ZINkTMoZBcvXo1Jk6ciIqKCkmKYcxRp06dQmJiIg4cOKB0KcxJ/vSnP2H+/PmIjY0FAFRUVGDKlCm4dOmSJMsXHZL79+/H6NGj//A+asbU4MyZM0hKSsK+fft4wLmHSEhIwJQpU+4OOL948SKmTp0qyZ1zokIyKysLI0eORGZmpsMFMCYHa1AeOHCAg9IDCIKAXr16ITU1FV5eXiAirFixAj///LPDy651SFZUVGDMmDF8OMNU79y5c3cPvTko3Z9Op8OwYcPw4osvQhAEFBUV4Z///CcKCgocWm6tQ3Lt2rX4/vvv+UfHXAIHpWfx8/PD+PHj8ec//xkAsHv3bmzevNmhZdYqJK9cuYIJEyagpKTEoZUy5kznzp3jQ28P0qRJE3z88cfw9/eHyWTCggULUFxcLHp5dofkjRs3MGPGDBw/flz0yhhTytmzZ5GcnIxff/1V6VKYzARBQN++ffH8889DEASkpaU5dG7S7pDs168f5s2bZ/PNZoyp2ZkzZ5CYmMhB6QG8vLzw4YcfomnTpigrK8OCBQtEX+m2OyR//vlnHu7DXN6ZM2eQnJyMw4cPK10Kk1lsbCxGjRoFg8GAnTt3ir7JgJ9MzjzOyZMnkZiYiCNHjihdCpORIAh49dVX0a5dOxQUFGDZsmWilsMhyTzSiRMn8Prrr3NQurmAgAAMHz78bm/yypUrtV4GhyTzWNagPHTokNKlMBn16NEDjz76KDIyMrBnz55az88hyTzaiRMnkJiYiIMHDypdCpOJv78/UlJSoNFosHr1apjN5lrNzyHJPJ71HCW/rtZ9devWDe3bt8e+fftw8eLFWs3LIclcyoABA6DVaiVf7qlTp+4eenNQup+AgAC8+uqryMvLw/bt22s1L4ckcylz5szBq6++KktQWsdRclC6p27duqFRo0bYunUrTCaT3fNxSDKXEhISgunTp8sWlNbnUfLFHPfTpEkTdO/eHceOHavVmG8OSeZyrEH5yiuvyBaUSUlJPODczWi1WgwcOBDFxcU4ceKE3fNxSDKXFBISgn/9618YNGgQNBrpf8Y84Nw9tWrVCk2bNq3VduWQZC4rJCQE06ZNky0oT5w4gaSkJKSnp0u+bKaM4OBgPProozhw4AAqKyvtmodDkrk0a1C++uqrsgTlsWPHkJiYiGPHjkm+bOZ8giCgW7duOHbsmN3v5uKQZC4vNDQU06dPx2uvvSZLUKanpyMxMZEfE+gmWrRoAR8fH1y/ft2uz3NIMrcQGhqKqVOnyhaUR48exeuvv849SjdQr149tGjRAhkZGXZ9XidzPeweTz31lNIlVKuwsFDpEhwWGhqKadOmQRAELF++vNa3n9Xk6NGjSExMxNdff42HH35Y0mUz5zEYDGjdurXdD7vgkHSi3bt3K12C27Oeo9TpdFi6dGmtBg3bo7y8XJLXlDJlxcfH2/18ST7cZm4nODgYU6ZMQXJyMnQ66foB8fHxWLZsGVq2bCnZMpkyYmJiuCfJPFtgYCAmT54MAFiyZInDPUprQLZu3VqK8pjC/P394evra9dnuSfJ3JY1KAcPHgy9Xi96OfHx8fjqq684IN2Ij48PGjZsaNdnOSSZW7MG5ZAhQ0QdelsDsk2bNjJUx5RiMBgQHh5u12c5JJnbCwgIwMSJE2sdlM2bN8fSpUs5IN0UH24zdg9rUA4ePNiuh2LExcVh6dKlaNu2rROqY0rw8fGx63Mckszp4uPjFVlvYGAgJk2ahDfeeMNmUDZv3hzLly9Hu3btnFgdc7Y6derY9TkOSeZU1nGMSgkMDMQXX3yBIUOGVBmUzZs3x1dffcU9SA9Qr149uz7HIcmcRqfTYfjw4Xj66acVrSMoKAgTJ058oEeZkJCAZcuWcUB6CHuvbgtk53PqBUFwqCB3IvbR/p7ehj169MDKlSsRHBysdCkA7tyK+eGHH2LhwoV4+OGH8dVXX6FVq1ZKl8VUhkNSBA7J2mvWrBn+85//ICYmBgcPHkTHjh2VLgkAUFRUhNmzZ6N3797485//rHQ5TIU4JEXgkKydunXrYtasWWjYsCFmz56NDRs21OodI4wpiUNSBA7J2mnTpg1CQ0ORlpaG27dvAxDfhow5G4ekCByStSMIwgNtxiHJlFZaWmrXgHK+us1kx4HI1CgvL8+uz9ndk2SMMU/EPUnGGLOBQ5IxxmzgkGSMMRs4JBljzAYOScYYs4FDkjHGbOCQZIwxGzgkGWPMBg5Jxhiz4f8AYeZcGiot1vMAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 400x400 with 16 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "expressions = [\n",
    "    env.sample_non_empty(\n",
    "        lambda: sampler.sample(env.grammar.start_symbol, max_primitives=7)\n",
    "    )\n",
    "    for _ in range(16)\n",
    "]\n",
    "images = np.array([env.compile(e) for e in expressions])\n",
    "images = images.reshape((4, 4) + env.compiled_shape)\n",
    "fig, axes = plt.subplots(4, 4, figsize=(4, 4))\n",
    "for i in range(4):\n",
    "    for j in range(4):\n",
    "        axes[i, j].imshow(images[i, j], cmap=\"gray\")\n",
    "        axes[i, j].axis(\"off\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 400x400 with 16 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "expressions = [\n",
    "    env.sample_non_empty(\n",
    "        lambda: sampler.sample(env.grammar.start_symbol, max_primitives=7)\n",
    "    )\n",
    "    for _ in range(16)\n",
    "]\n",
    "images = np.array([env.compile_observation(e) for e in expressions])\n",
    "images = images.reshape((4, 4) + env.compiled_shape)\n",
    "fig, axes = plt.subplots(4, 4, figsize=(4, 4))\n",
    "for i in range(4):\n",
    "    for j in range(4):\n",
    "        axes[i, j].imshow(images[i, j], cmap=\"gray\")\n",
    "        axes[i, j].axis(\"off\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 114,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "expr = sampler.sample(env.grammar.start_symbol, min_primitives=2, max_primitives=10)\n",
    "plt.imshow(env.compile_observation(expr), cmap=\"gray\")\n",
    "plt.axis(\"off\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 87,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(- (+ (Circle 2 B 0) (+ (Circle F 7 E) (Circle 3 4 F))) (+ (+ (Circle 9 B 0) (Circle 1 F 9)) (- (Circle 3 4 5) (Circle 6 D B))))\n"
     ]
    }
   ],
   "source": [
    "expr = sampler.sample(env.grammar.start_symbol, min_primitives=7, max_primitives=7)\n",
    "print(expr)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(+ (- (+ (Circle 3 4 8) (Circle 2 3 B)) (Quad 9 B 5 6 H)) (- (- (Quad F A E 5 J) (Circle 2 4 7)) (+ (Circle 7 9 D) (Circle E F 7))))\n",
      "                                                                                                            ^ --> 0\n"
     ]
    }
   ],
   "source": [
    "mut = random_mutation(expr, env.grammar, sampler)\n",
    "print(mut.pretty(expr))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 60,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  5%|▌         | 529/10000 [00:02<00:43, 217.33it/s]"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[1;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "Cell \u001b[1;32mIn[60], line 26\u001b[0m\n\u001b[0;32m     22\u001b[0m     \u001b[38;5;28mprint\u001b[39m(\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mMedian length\u001b[39m\u001b[38;5;124m\"\u001b[39m, np\u001b[38;5;241m.\u001b[39mmedian(lengths))\n\u001b[0;32m     23\u001b[0m     \u001b[38;5;28mprint\u001b[39m(\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mMin length\u001b[39m\u001b[38;5;124m\"\u001b[39m, \u001b[38;5;28mmin\u001b[39m(lengths))\n\u001b[1;32m---> 26\u001b[0m \u001b[43msample_forward_process\u001b[49m\u001b[43m(\u001b[49m\u001b[43mN\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m10000\u001b[39;49m\u001b[43m)\u001b[49m\n",
      "Cell \u001b[1;32mIn[60], line 11\u001b[0m, in \u001b[0;36msample_forward_process\u001b[1;34m(N)\u001b[0m\n\u001b[0;32m      2\u001b[0m target_expressions \u001b[38;5;241m=\u001b[39m [\n\u001b[0;32m      3\u001b[0m     sampler\u001b[38;5;241m.\u001b[39msample(\n\u001b[0;32m      4\u001b[0m         env\u001b[38;5;241m.\u001b[39mgrammar\u001b[38;5;241m.\u001b[39mstart_symbol,\n\u001b[1;32m   (...)\u001b[0m\n\u001b[0;32m      8\u001b[0m     \u001b[38;5;28;01mfor\u001b[39;00m _ \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mrange\u001b[39m(N)\n\u001b[0;32m      9\u001b[0m ]\n\u001b[0;32m     10\u001b[0m steps \u001b[38;5;241m=\u001b[39m [random\u001b[38;5;241m.\u001b[39mrandint(\u001b[38;5;241m1\u001b[39m, \u001b[38;5;241m5\u001b[39m) \u001b[38;5;28;01mfor\u001b[39;00m _ \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mrange\u001b[39m(N)]\n\u001b[1;32m---> 11\u001b[0m training_examples \u001b[38;5;241m=\u001b[39m \u001b[43m[\u001b[49m\n\u001b[0;32m     12\u001b[0m \u001b[43m    \u001b[49m\u001b[43mforward_process_with_guards\u001b[49m\u001b[43m(\u001b[49m\u001b[43mexpression\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mstep\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43menv\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mgrammar\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43msampler\u001b[49m\u001b[43m)\u001b[49m\n\u001b[0;32m     13\u001b[0m \u001b[43m    \u001b[49m\u001b[38;5;28;43;01mfor\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[43mexpression\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mstep\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;129;43;01min\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[43mtqdm\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mtqdm\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mzip\u001b[39;49m\u001b[43m(\u001b[49m\u001b[43mtarget_expressions\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43msteps\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mtotal\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mN\u001b[49m\u001b[43m)\u001b[49m\n\u001b[0;32m     14\u001b[0m \u001b[43m\u001b[49m\u001b[43m]\u001b[49m\n\u001b[0;32m     15\u001b[0m tokenized_forwards \u001b[38;5;241m=\u001b[39m [\n\u001b[0;32m     16\u001b[0m     (tokenizer\u001b[38;5;241m.\u001b[39m_tokenize_one(expr), tokenizer\u001b[38;5;241m.\u001b[39m_tokenize_one(m\u001b[38;5;241m.\u001b[39mreplacement))\n\u001b[0;32m     17\u001b[0m     \u001b[38;5;28;01mfor\u001b[39;00m expr, m \u001b[38;5;129;01min\u001b[39;00m training_examples\n\u001b[0;32m     18\u001b[0m ]\n\u001b[0;32m     19\u001b[0m lengths \u001b[38;5;241m=\u001b[39m [\u001b[38;5;28mlen\u001b[39m(expr) \u001b[38;5;241m+\u001b[39m \u001b[38;5;28mlen\u001b[39m(a) \u001b[38;5;241m+\u001b[39m \u001b[38;5;241m1\u001b[39m \u001b[38;5;241m+\u001b[39m \u001b[38;5;241m1\u001b[39m \u001b[38;5;241m+\u001b[39m \u001b[38;5;241m1\u001b[39m \u001b[38;5;28;01mfor\u001b[39;00m expr, a \u001b[38;5;129;01min\u001b[39;00m tokenized_forwards]\n",
      "Cell \u001b[1;32mIn[60], line 12\u001b[0m, in \u001b[0;36m<listcomp>\u001b[1;34m(.0)\u001b[0m\n\u001b[0;32m      2\u001b[0m target_expressions \u001b[38;5;241m=\u001b[39m [\n\u001b[0;32m      3\u001b[0m     sampler\u001b[38;5;241m.\u001b[39msample(\n\u001b[0;32m      4\u001b[0m         env\u001b[38;5;241m.\u001b[39mgrammar\u001b[38;5;241m.\u001b[39mstart_symbol,\n\u001b[1;32m   (...)\u001b[0m\n\u001b[0;32m      8\u001b[0m     \u001b[38;5;28;01mfor\u001b[39;00m _ \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mrange\u001b[39m(N)\n\u001b[0;32m      9\u001b[0m ]\n\u001b[0;32m     10\u001b[0m steps \u001b[38;5;241m=\u001b[39m [random\u001b[38;5;241m.\u001b[39mrandint(\u001b[38;5;241m1\u001b[39m, \u001b[38;5;241m5\u001b[39m) \u001b[38;5;28;01mfor\u001b[39;00m _ \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mrange\u001b[39m(N)]\n\u001b[0;32m     11\u001b[0m training_examples \u001b[38;5;241m=\u001b[39m [\n\u001b[1;32m---> 12\u001b[0m     \u001b[43mforward_process_with_guards\u001b[49m\u001b[43m(\u001b[49m\u001b[43mexpression\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mstep\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43menv\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mgrammar\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43msampler\u001b[49m\u001b[43m)\u001b[49m\n\u001b[0;32m     13\u001b[0m     \u001b[38;5;28;01mfor\u001b[39;00m expression, step \u001b[38;5;129;01min\u001b[39;00m tqdm\u001b[38;5;241m.\u001b[39mtqdm(\u001b[38;5;28mzip\u001b[39m(target_expressions, steps), total\u001b[38;5;241m=\u001b[39mN)\n\u001b[0;32m     14\u001b[0m ]\n\u001b[0;32m     15\u001b[0m tokenized_forwards \u001b[38;5;241m=\u001b[39m [\n\u001b[0;32m     16\u001b[0m     (tokenizer\u001b[38;5;241m.\u001b[39m_tokenize_one(expr), tokenizer\u001b[38;5;241m.\u001b[39m_tokenize_one(m\u001b[38;5;241m.\u001b[39mreplacement))\n\u001b[0;32m     17\u001b[0m     \u001b[38;5;28;01mfor\u001b[39;00m expr, m \u001b[38;5;129;01min\u001b[39;00m training_examples\n\u001b[0;32m     18\u001b[0m ]\n\u001b[0;32m     19\u001b[0m lengths \u001b[38;5;241m=\u001b[39m [\u001b[38;5;28mlen\u001b[39m(expr) \u001b[38;5;241m+\u001b[39m \u001b[38;5;28mlen\u001b[39m(a) \u001b[38;5;241m+\u001b[39m \u001b[38;5;241m1\u001b[39m \u001b[38;5;241m+\u001b[39m \u001b[38;5;241m1\u001b[39m \u001b[38;5;241m+\u001b[39m \u001b[38;5;241m1\u001b[39m \u001b[38;5;28;01mfor\u001b[39;00m expr, a \u001b[38;5;129;01min\u001b[39;00m tokenized_forwards]\n",
      "File \u001b[1;32mc:\\Users\\shrey\\Documents\\Projects\\tree-diffusion\\td\\samplers\\mutator.py:445\u001b[0m, in \u001b[0;36mforward_process_with_guards\u001b[1;34m(expression, num_steps, grammar, sampler, full_intersection, max_tries)\u001b[0m\n\u001b[0;32m    443\u001b[0m attempts \u001b[38;5;241m=\u001b[39m \u001b[38;5;241m0\u001b[39m\n\u001b[0;32m    444\u001b[0m \u001b[38;5;28;01mwhile\u001b[39;00m \u001b[38;5;28;01mTrue\u001b[39;00m:\n\u001b[1;32m--> 445\u001b[0m     mutation \u001b[38;5;241m=\u001b[39m \u001b[43mrandom_mutation\u001b[49m\u001b[43m(\u001b[49m\u001b[43mcurrent_expression\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mgrammar\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43msampler\u001b[49m\u001b[43m)\u001b[49m\n\u001b[0;32m    446\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m \u001b[38;5;28many\u001b[39m(\n\u001b[0;32m    447\u001b[0m         s \u001b[38;5;241m<\u001b[39m\u001b[38;5;241m=\u001b[39m mutation\u001b[38;5;241m.\u001b[39mstart \u001b[38;5;241m<\u001b[39m e \u001b[38;5;129;01mor\u001b[39;00m s \u001b[38;5;241m<\u001b[39m\u001b[38;5;241m=\u001b[39m mutation\u001b[38;5;241m.\u001b[39mend \u001b[38;5;241m<\u001b[39m e\n\u001b[0;32m    448\u001b[0m         \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m full_intersection\n\u001b[0;32m    449\u001b[0m         \u001b[38;5;28;01melse\u001b[39;00m _intersects((mutation\u001b[38;5;241m.\u001b[39mstart, mutation\u001b[38;5;241m.\u001b[39mend), (s, e))\n\u001b[0;32m    450\u001b[0m         \u001b[38;5;28;01mfor\u001b[39;00m s, e \u001b[38;5;129;01min\u001b[39;00m guards\n\u001b[0;32m    451\u001b[0m     ):\n\u001b[0;32m    452\u001b[0m         \u001b[38;5;28;01mbreak\u001b[39;00m\n",
      "File \u001b[1;32mc:\\Users\\shrey\\Documents\\Projects\\tree-diffusion\\td\\samplers\\mutator.py:109\u001b[0m, in \u001b[0;36mrandom_mutation\u001b[1;34m(expression, grammar, sampler, selection_max_primitives, replacement_max_primitives, max_attempts_difference)\u001b[0m\n\u001b[0;32m    100\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21mrandom_mutation\u001b[39m(\n\u001b[0;32m    101\u001b[0m     expression: \u001b[38;5;28mstr\u001b[39m,\n\u001b[0;32m    102\u001b[0m     grammar: Grammar,\n\u001b[1;32m   (...)\u001b[0m\n\u001b[0;32m    106\u001b[0m     max_attempts_difference: \u001b[38;5;28mint\u001b[39m \u001b[38;5;241m=\u001b[39m \u001b[38;5;241m100\u001b[39m,\n\u001b[0;32m    107\u001b[0m ) \u001b[38;5;241m-\u001b[39m\u001b[38;5;241m>\u001b[39m Mutation:\n\u001b[0;32m    108\u001b[0m     tree \u001b[38;5;241m=\u001b[39m grammar\u001b[38;5;241m.\u001b[39mparse(expression)\n\u001b[1;32m--> 109\u001b[0m     \u001b[43mAddParents\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mvisit\u001b[49m\u001b[43m(\u001b[49m\u001b[43mtree\u001b[49m\u001b[43m)\u001b[49m\n\u001b[0;32m    111\u001b[0m     candidates \u001b[38;5;241m=\u001b[39m nodes_with_max_primitives(\n\u001b[0;32m    112\u001b[0m         tree, grammar\u001b[38;5;241m.\u001b[39mprimitives, selection_max_primitives\n\u001b[0;32m    113\u001b[0m     )\n\u001b[0;32m    115\u001b[0m     candidate_primitive_counts \u001b[38;5;241m=\u001b[39m [x\u001b[38;5;241m.\u001b[39mprimitive_count \u001b[38;5;28;01mfor\u001b[39;00m x \u001b[38;5;129;01min\u001b[39;00m candidates]\n",
      "File \u001b[1;32mc:\\Users\\shrey\\.conda\\envs\\jax\\Lib\\site-packages\\lark\\visitors.py:364\u001b[0m, in \u001b[0;36mVisitor.visit\u001b[1;34m(self, tree)\u001b[0m\n\u001b[0;32m    362\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21mvisit\u001b[39m(\u001b[38;5;28mself\u001b[39m, tree: Tree[_Leaf_T]) \u001b[38;5;241m-\u001b[39m\u001b[38;5;241m>\u001b[39m Tree[_Leaf_T]:\n\u001b[0;32m    363\u001b[0m     \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mVisits the tree, starting with the leaves and finally the root (bottom-up)\u001b[39m\u001b[38;5;124m\"\u001b[39m\n\u001b[1;32m--> 364\u001b[0m     \u001b[38;5;28;01mfor\u001b[39;00m subtree \u001b[38;5;129;01min\u001b[39;00m \u001b[43mtree\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43miter_subtrees\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m:\n\u001b[0;32m    365\u001b[0m         \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_call_userfunc(subtree)\n\u001b[0;32m    366\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m tree\n",
      "File \u001b[1;32mc:\\Users\\shrey\\.conda\\envs\\jax\\Lib\\site-packages\\lark\\tree.py:147\u001b[0m, in \u001b[0;36mTree.iter_subtrees\u001b[1;34m(self)\u001b[0m\n\u001b[0;32m    145\u001b[0m     subtrees[\u001b[38;5;28mid\u001b[39m(subtree)] \u001b[38;5;241m=\u001b[39m subtree\n\u001b[0;32m    146\u001b[0m     \u001b[38;5;66;03m# Reason for type ignore https://github.com/python/mypy/issues/10999\u001b[39;00m\n\u001b[1;32m--> 147\u001b[0m     queue \u001b[38;5;241m+\u001b[39m\u001b[38;5;241m=\u001b[39m \u001b[43m[\u001b[49m\u001b[43mc\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43;01mfor\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[43mc\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;129;43;01min\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[38;5;28;43mreversed\u001b[39;49m\u001b[43m(\u001b[49m\u001b[43msubtree\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mchildren\u001b[49m\u001b[43m)\u001b[49m\u001b[43m  \u001b[49m\u001b[38;5;66;43;03m# type: ignore[misc]\u001b[39;49;00m\n\u001b[0;32m    148\u001b[0m \u001b[43m              \u001b[49m\u001b[38;5;28;43;01mif\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[38;5;28;43misinstance\u001b[39;49m\u001b[43m(\u001b[49m\u001b[43mc\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mTree\u001b[49m\u001b[43m)\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;129;43;01mand\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[38;5;28;43mid\u001b[39;49m\u001b[43m(\u001b[49m\u001b[43mc\u001b[49m\u001b[43m)\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;129;43;01mnot\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[38;5;129;43;01min\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[43msubtrees\u001b[49m\u001b[43m]\u001b[49m\n\u001b[0;32m    150\u001b[0m \u001b[38;5;28;01mdel\u001b[39;00m queue\n\u001b[0;32m    151\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mreversed\u001b[39m(\u001b[38;5;28mlist\u001b[39m(subtrees\u001b[38;5;241m.\u001b[39mvalues()))\n",
      "File \u001b[1;32mc:\\Users\\shrey\\.conda\\envs\\jax\\Lib\\site-packages\\lark\\tree.py:147\u001b[0m, in \u001b[0;36m<listcomp>\u001b[1;34m(.0)\u001b[0m\n\u001b[0;32m    145\u001b[0m     subtrees[\u001b[38;5;28mid\u001b[39m(subtree)] \u001b[38;5;241m=\u001b[39m subtree\n\u001b[0;32m    146\u001b[0m     \u001b[38;5;66;03m# Reason for type ignore https://github.com/python/mypy/issues/10999\u001b[39;00m\n\u001b[1;32m--> 147\u001b[0m     queue \u001b[38;5;241m+\u001b[39m\u001b[38;5;241m=\u001b[39m [c \u001b[38;5;28;01mfor\u001b[39;00m c \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mreversed\u001b[39m(subtree\u001b[38;5;241m.\u001b[39mchildren)  \u001b[38;5;66;03m# type: ignore[misc]\u001b[39;00m\n\u001b[0;32m    148\u001b[0m               \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28misinstance\u001b[39m(c, Tree) \u001b[38;5;129;01mand\u001b[39;00m \u001b[38;5;28mid\u001b[39m(c) \u001b[38;5;129;01mnot\u001b[39;00m \u001b[38;5;129;01min\u001b[39;00m subtrees]\n\u001b[0;32m    150\u001b[0m \u001b[38;5;28;01mdel\u001b[39;00m queue\n\u001b[0;32m    151\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mreversed\u001b[39m(\u001b[38;5;28mlist\u001b[39m(subtrees\u001b[38;5;241m.\u001b[39mvalues()))\n",
      "\u001b[1;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": [
    "def sample_forward_process(N=10000):\n",
    "    target_expressions = [\n",
    "        sampler.sample(\n",
    "            env.grammar.start_symbol,\n",
    "            min_primitives=2,\n",
    "            max_primitives=8,\n",
    "        )\n",
    "        for _ in range(N)\n",
    "    ]\n",
    "    steps = [random.randint(1, 5) for _ in range(N)]\n",
    "    training_examples = [\n",
    "        forward_process_with_guards(expression, step, env.grammar, sampler)\n",
    "        for expression, step in tqdm.tqdm(zip(target_expressions, steps), total=N)\n",
    "    ]\n",
    "    tokenized_forwards = [\n",
    "        (tokenizer._tokenize_one(expr), tokenizer._tokenize_one(m.replacement))\n",
    "        for expr, m in training_examples\n",
    "    ]\n",
    "    lengths = [len(expr) + len(a) + 1 + 1 + 1 for expr, a in tokenized_forwards]\n",
    "    print(\"Max length\", max(lengths))\n",
    "    print(\"Mean length\", np.mean(lengths))\n",
    "    print(\"Median length\", np.median(lengths))\n",
    "    print(\"Min length\", min(lengths))\n",
    "\n",
    "\n",
    "sample_forward_process(N=10000)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 61,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(- (+ (Circle 4 6 0) (- (Circle 7 9 1) (Quad C E D F I))) (- (Circle 7 B 5) (+ (Circle B 8 A) (Quad 5 5 0 D L))))\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(- (+ (Circle 4 6 0) (- (Circle 7 9 1) (Quad C E D F I))) (- (Circle 7 B 5) (+ (Circle B 8 A) (Quad 5 5 0 D L))))\n",
      "                                                                                         ^ --> B\n",
      "(- (+ (Circle 4 6 0) (- (Circle 7 9 1) (Quad C E D F I))) (- (Circle 7 B 5) (+ (Circle B B A) (Quad 5 5 0 D L))))\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(- (+ (Circle 4 6 0) (- (Circle 7 9 1) (Quad C E D F I))) (- (Circle 7 B 5) (+ (Circle B B A) (Quad 5 5 0 D L))))\n",
      "                                                             ^^^^^^^^^^^^^^ --> (Circle F 9 8)\n",
      "(- (+ (Circle 4 6 0) (- (Circle 7 9 1) (Quad C E D F I))) (- (Circle F 9 8) (+ (Circle B B A) (Quad 5 5 0 D L))))\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(- (+ (Circle 4 6 0) (- (Circle 7 9 1) (Quad C E D F I))) (- (Circle F 9 8) (+ (Circle B B A) (Quad 5 5 0 D L))))\n",
      "                                                                            ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ --> (Circle 8 8 8)\n",
      "(- (+ (Circle 4 6 0) (- (Circle 7 9 1) (Quad C E D F I))) (- (Circle F 9 8) (Circle 8 8 8)))\n"
     ]
    },
    {
     "data": {
      "image/png": "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DL2jg/E1Dv9w/V1cHO5jUVC3sc5Hw3aKI/kIH97yMYL7orR/vxd3fBI7kdATgc/0pCD+c93jp08np6jdOklpT5wvdX2QQgQcPSjMS2YBzDzUH7EpGUrjjE3gzswoCHOEoiDnn2PjZGzY2UloJzT83O+B5ELSBvLcojsKRETEoyiIiIhHURAREY+iICIiHkVBREQ8ioKIiHgUBRER8SgKIiLiURRERMSjKIiIiEdREBERj6IgIiIeRUFERDyKgoiIeBQFERHxKAoiIuJRFERExKMoiIiIR1EQERGPoiAiIh5FQUREPIqCiIh4FAUREfEoCiIi4lEURETEoyiIiIhHURAREY+iICIiHkVBREQ8ioKIiHgUBRER8SgKIiLiURRERMSjKIiIiEdREBERj6IgIiIeRUFERDyKgoiIeBQFERHxKAoiIuJRFERExKMoiIiIR1EQERGPoiAiIh5FQUREPIqCiIh4FAUREfEoCiIi4lEURETEoyiIiIhHURAREY+iICIiHkVBREQ8ioKIiHgUBRER8SgKIiLiURRERMSjKIiIiEdREBERj6IgIiIeRUFERDyKgoiIeBQFERHxKAoiIuJRFERExKMoiIiIR1EQERGPoiAiIh5FQUREPIqCiIh4FAUREfEoCiIi4lEURETEoyiIiIhHURAREY+iICIiHkVBREQ8ioKIiHgUBRER8SgKIiLiURRERMSjKIiIiEdREBERj6IgIiIeRUFERDyKgoiIeBQFERHxKAoiIuJRFERExKMoiIiIR1EQERGPoiAiIh5FQUREPIqCiIh4FAUREfEoCiIi4lEURETEoyiIiIhHURAREY+iICIiHkVBREQ8ioKIiHgUBRER8SgKIiLiURRERMSjKIiIiEdREBERj6IgIiIeRUFERDyKgoiIeBQFERHxKAoiIuJRFERExKMoiIiIR1EQERGPoiAiIh5FQUREPIqCiIh4FAUREfEoCiIi4lEURETEoyiIiIhHURAREY+iICIiHkVBREQ8ioKIiHgUBRER8SgKIiLiURRERMSjKIiIiEdREBERj6IgIiIeRUFERDyKgoiIeBQFERHxKAoiIuJRFERExKMoiIiIR1EQERGPoiAiIh5FQUREPIqCiIh4FAUREfEoCiIi4lEURETEoyiIiIhHURAREY+iICIiHkVBREQ8ioKIiHgUBRER8SgKIiLiURRERMSjKIiIiEdREBERj6IgIiIeRUFERDyKgoiIeBQFERHxKAoiIuJRFERExKMoiIiIR1EQERGPoiAiIh5FQUREPIqCiIh4FAUREfEoCiIi4lEURETEoyiIiIhHURAREY+iICIiHkVBREQ8ioKIiHgUBRER8SgKIiLiURRERMSjKIiIiEdREBERj6IgIiIeRUFERDyKgoiIeBQFERHxKAoiIuJRFERExKMoiIiIR1EQERGPoiAiIh5FQUREPIqCiIh4FAUREfEoCiIi4lEURETEoyiIiIhHURAREY/huq7r9xAiIjI76ElBREQ8ioKIiHgUBRER8SgKIiLiURRERMSjKIiIiEdREBERj6IgIiIeRUFERDz/F1j0gjeTJn+SAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(- (+ (Circle 4 6 0) (- (Circle 7 9 1) (Quad C E D F I))) (- (Circle F 9 8) (Circle 8 8 8)))\n",
      "                     ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ --> (Circle 8 8 0)\n",
      "(- (+ (Circle 4 6 0) (Circle 8 8 0)) (- (Circle F 9 8) (Circle 8 8 8)))\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAYUAAAGFCAYAAAASI+9IAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8g+/7EAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAxT0lEQVR4nO3dd5Rcd33//+e9d3rZ2V61u9KqW7244Rpwt7GBgAFTwpcQHAL5wiEQyo8SSIBQwy/kSwgkkAC2qQZjwDQb27jITbZlq3dt7zu702fuvd8/JN2YfIWRrDIzu6/HOT4+KgfeEod97ufzmc+9huu6LiIiIoBZ7gFERKRyKAoiIuJRFERExKMoiIiIR1EQERGPoiAiIh5FQUREPIqCiIh4fMf7Gy83X3U65xD5o4xgEHN+J6X6KIXaANkmH8Uo5BoMinGXUtwm2JglHsmzoHacukCGGl+OOl+GiJUnbuaotTJEzTz1VoqoUSJu2jSaASJmAICMU2DMKTDjWKRdHxN2jLQTZMqOkLQjZJwAyVKYqWKEZDHE/qkGZjJB8mNhfDMW/hmD0JiLPw2R0RKBqTzWRBrnYB9uPl/mv0GZ637tfP+P/p7jjoLIaWNaWHUJjHAYNx7Bjodwwj4KNX5KYZNi2KAUATtgUEiAHXaxwy5OrIgZtAlHCzSE8sSDebqik9T6M3SGJoibOaJmnoiZJ2DYRMw8UaOA33CIGyX8BkQNE79heaP4DYu4YWKaNhHXJmTYFF2TtC9AxglScC1mnDBpJ0jGCdAZnmSqGOFQQx0z+SCpXJCpVBA3b2GmLKxcFCsbI5Bswcq7+LLgz7r4sg6B6SJmtoQ1k8OYyeCmM9jJaXDsMv6PIXOdoiDlZRiYAT9OVxu51gipeT5muqFQb9Myf4Ll9cNcUruTyyL7aLHCv/cF/IUJPO+v+g2LOitC3TF/9eh3+pnn/NyB4/pvLbo2g3aWX6cXcd/UEnZNNtO7r4HARIjYoRjx3iKhoTTmzixO3gE9fUbKRFGQ086qqcGd3052XpxcncVMl0mxxqXYWKKmKUVjLE1DaIgOX57GYIq2wBRxM8f8wCgNZoZGq0iTFTwFQSgfv2HRYgW5PLqHtaFDTDWHObCgiaQdYbCQYKwQY6YYZDLfzFgqyvRoDP+oj8C0QfyQQ3DSJtw7g3FoAHt6utx/HJnFFAU5eYaBYVkY4TCGz4cRDIDfDz4LNxSgVBdhuifMTKdJrsmhZuEEi2snubhhNzfEt7DQH3ue//DgkX+qX9Dw0+Xz0+UDsCE8dMzft7eY4o6Z1dw7voQDk3WMNiYIjvuJ19dSUxPEN5nByBWgZEOxiJsv4JZKOOkMuFplyMkxjvcpqTpolj/Eqk1g1NUyeW4b6VaT6SUlatpnWN40zF+3/YZ6M0fIcPAbEDAM/Bj4j+zl+7CwDH0I7rls16GETdG1KboOOdfBBvIuFFyTUSfCP/dfxq7xJlL9NcR3WUSHHOo29eNOJrWSkD9IB81yypnxOGaihmJXI4VEgHSbj1yjQb7WpTQvT7wmy/lNQ6yMD7A0NMi6QImIGSn32FXFMkwsTIKG/5i/3u0UuLHlMXYl2tjW1MazHa0MTYcZXzWPwFQnoQmX6GCJQLKA/+AozvQMzszMGf5TSLVSFOT4GQZmUwO5+Q0MXBwk11Xgtese5NrEU1wQ+kPf7T//wa6cuIgZ4MZYEmJJaNwBC/771+7PwZ1T6/jB5g2EDkXp+J1F8IAfJ5XStpIcF20fyTEZPh/m/E5y8+sZXxUk2+wSWj5FV+0Ui+KjnBUZoNU3xYrACI2WRcIMl3tkAZJOljHb5plCK0PFBDuybeyabqZ3qpb89gThYYPGZ/KEDoxjH+rDLZXKPbKcQdo+kuN2+IA4iFmbgGAAuz5GckGU6QUW2TVZFrSO8f75P2exP0mX77kHw893SCxnWsIMkzBhoT8FpKC2n0NNKXYU6vh0/GoODDbg+MMk6lqI1cfwTaQgm8NJTuMWCoqEaKUgh/m6O8ktaubQ5QF8C1N8bt336fRN0W7ZhAwLE5Og4dOhcJWyXYe8W6LI4cPrAdtiX7GR921+Be7eKF2/yhPcM0ypr7/co8pppJWCHJPh8x2+Pby0m1xTmKlFfrItLoX2ImsW7mNdbS/rA2MkzAARM1TuceUUsAyTiPHf5zsRo0C9OcRrlj3BEy1dbG3uIDjQRXikm9rdBYIjWYw9h3CzWa0e5hhFYa4xDMxIBKO+loFza0gucXjLn9zNhbGdXPx7X/+1LTSbRY487+ljTVuhaSssgXuzJr9NLefbd19EYlcNbZN1OOPg6pB6TtH20Rzh65lPoaOOg1eHcOZnuWLxDlZHe2n3T7I+OEKt6SOmVcGclnSyJB2bzflWegsNPJtu5ze7luHbH6L75xn8AxOUDvaWe0w5Cdo+mssMA6u+DiMSwamPM7ksQarDpHn9ENe0b+Vd9c8854xAqwL570PqLl8KOzJNPrGbTwen+WX9ckZGm4kNhIgnopiTKdxUGntqSiuIWUgrhVnKjERIvnQ146sM1ly6i5vb7mVjMEXI8OkWsRy3o7erc26Jh3O1fKX/Unbes5CGZ23iP3saJ5cr94hyArRSmEMMfwCruZHs8jamFgfItBpYK5Osah7mppZHOMufJGFqRSAn5rm3q1cGxrmpbRPffrHBrhVNjK9cT2TIpXZ3gdD2fuyxCdxiodwjy0lSFKrdkYfRmTUxivObGbwwyIJLDvDWefdzdWTyOY9KUBDk5MzzxbgxluTGxb8gv6jIHWsb+VrvxfTf28m8bAu+QuHwfQfb1rZSFdP2UbUyDIxAgPyfrGZ8hZ+Wa3pZW9fHpTXbWewfp8UydctYTqukk2Wg5LK72Mh908vYPNHJxE87aNiaJ3DfM7ilouJQYbR9NBsZBr7WFtzaOLnOBMMb/RTOyvD++T/nLH+SNl8MiJZ7SpkDEmaYRACWBzKsD97P04lG/nrDTZQiIVpYRag3iZGcoTQ8ojhUEa0UqolhYIbDjN60hok1Dp+98jbWBgfo8p2KN5KJnLyia7O/lOOxXBcf+dmraNhiUH/bZtxCQWGoAFopzBJWTQ2llQuYWBFhugea1g1zfdMh1gcHaLJ8CoJUDL9h0W5ZnBc6yJUXPcXmZfPYv3g9NXuh/tkU5vYDeox3hVMUKpVhgGFihoLQ2sTouijFlyT56IpfcEO0/8hFMx0eS+WJmSFiJny5YxPJtiw/6unmU09fheOL0TrWAMUiTqGot8RVKG0fVShr6SLSi+vpe3WRFV2D3NxxLz2+Cdp9BjEjqHsGUhVs1yHl5ukrwd5iA1/tv4RtB9rpvs0gsnuM0r4D5R5xTtH2UZUxgkHMeIzSknmMLouQXAzXLn+SK2uf4dpIDtAbzKS6WIZJwjh8IL0ikIGO+/hFdDX3XLCOREsbiZYEvt19ODMp3Hy+3OMKWilUFN+8DjIr2rDfPc7rux7hz2v6tCKQWcl2Hb48tYBvHTiH6D8niGwbotTbV+6xZj2tFKqBYWAtX0xyRR2DF0OiK8n75t/LmmA/lqGVgcxOlmHy4ugOEgvTfP5tl3PwUAcd93YQ3zqOvWuvzhrKSFEoF9M6fBM5EWdmWR3D5xl85PLbuS66n0YriraKZLZbEQizIjDGVRv+nduXLOHz2esxi/VExyZwplOHb0Y7drnHnHO0fVQm1oqlTC+rJfpX/VzStJvrap6mxwdhI6AtI5lTbNch6xbYUzT40fR67h5civHVJuI7p7C37iz3eLOKto8qjOHzYbW2UOxqZGBDlJkeh0/Me5A1wX6WB7QykLnJMkxiRoi1QTATT9ARmOTTF11PvKOB5po1+A+NYg+P6A1wZ4hWCmfKkfcbTF2+hJGX5vmXc2/lJeGMLp6JHEPRtflZJsG7H7qR9jv91Ny9S+9vOAW0UqgQ1qIF5Lvr2fUak/nzB/nbeY+xJjCO39DlM5Fj8RsWG4ND/M3Zv+a2lrPZftliFn63RPDghO42nGaKwmlk+AOY4RCZxY1MnOXnny79JuuDQ3T5Yug2ssjzm+eL8fbaXi6P7mDT/G4+v/1G6gONhMcmcDIZbSedJto+Ok0Mnw/7glUcuDrEtZc/xhvqH2JlwNBbz0ROkO065N0S24vwjbGL+PWv1tP98yzmI88qDCdI20dlYEYimDVxkhfMZ3yFxbwN/byi7nHWBnyKgcgLYBkmESPA2oDDy+sf58kNHfRlW2lo2UD8wf24yWm9FvQUUhROMbOpgdyiZha/dxt/2/AE10czR3+lrHOJVDvLMHlJ2OYlq2/nhz01/HB0AyMTCwjuGcbp6y/3eLOGonCK+DrnMb2xg77LXZYsHeDm5nvp9mXR2YHIqXduaIDWtik+8qGXsX37PDp/2UHsyT5K/QPlHq3qKQonyzCwGurJL2xmdJ3JK8/bxHubHqDZiqIgiJwe83wx2iyHby+5lU8kXsI9IxsITDXjz+exxyf00dWToIPmk2FaWA317PhoD0vO6uNfF36XJst35F0HInImpJwco3aJN+96HYe2trH0H3ZjT0zpERnHoIPm08W0MENBiucuY2xJkNWr9vHS5qdZ4NfKQORMO/pSn1d2bOZnvlUMv2IJdbvy+B/dgZPLKw4nSKefL4AZ8GM21LPvFX6uftsDfHvhHfx5YqjcY4nMaW+v7eW2xT/ksrc9zP4bAph1tRh+fd97ovQ3diIMA2txD5MbmijcNME7Fvya6+LPEDbC5Z5MRICIEeBNdQ9Rf1maWxZsJH5LJ4knR7D3HtA5w3FSFI6TGQphxKLMrGhkbI3Bt1bcxiJ/7shjrkWkEvgNi+WBCA21T3Huqr3cvOZmfLkmopNJnFRab3c7DjpoPk7ui9YwvirCX7zzJ1we3cl8X0SX0UQqmO067Crm+GlqFd/9pyto2JqGTVvKPVZZHc9Bs76q/RFWYwOFKzdy6MoouSumuSiyhy5fWEEQqXCWYdLt83F5dBupK1P0Xh6jeNkGrIb6co9W0fSV7fkYBu68Fg7cYPLql93H1vNvYUUgrMddi1SJiBlgbTDIjgu/xbUvf5iD1/lx25rLPVZF05nCsRgGZiTCwXetobAiw+c2fof1wQF0GU2kev1lw+/YcPUBPlj7CqJbX8S8L23WM5OOQVH4nwwDX2sLdkcj5oYkr+nZwp/GplEQRKrbQn+Mhf5JHlq5hV8Ez4KlC7CGxrFHRvXJpOfQ9tFzGQZmMMih1/VgfnaC2zd8lY82PVXuqUTkFPp82yZ+dP5XmP50noEbF2EGg2AY5R6rYigKRxj+AL4F3Qz+xXrcFyV5TdujtFimzg9EZhm/YdFuudzU9RjZC1MMvWU9vvldGP5AuUerCNo+OsKMhsksbuTs1z/Naxo38ZKwDehSmshsVGdFeHttLws3fJfvdp/DwR1LCE1OYU8Vyj1a2c35lYIZCmG1NLPjH5YRfN8g72/9JRuDqXKPJSJnwPmhKT7Q9gusD4yw42PLsZqaMENz+4GWc3qlYPgDGN3zyCysY+3qfbyp7UEW6qF2InNGwgyTMOEvO+/jG9YFpDbOJ7J3AmPfwTn7qs85u1Iw/AGs9hb2vqGJ6z9zN19Z8GOujWiFIDIXvSw6xVd6vs9ln/4d+1/bgtXSjOGbm98zz8koGMEgZk8Xe/5iHvPO6+fF0e0kzIBuKYvMUZZhUm8GuCL+DE0vGmTvW+djLuiak1tJczKFZk0NmZ463vKyX/Hi6HbWBoPlHklEyixiBjgnCJ9b8j1+1baKux++iPB0CubYBbc59a2x4fNh1dSw48M91Ly/l9fXPM3KgD6fLCL/baXf5c9rHyf8vn52fqAHq6ZmTm0lzZ0oGAZWZwe585aweEU/b25/gGYrQtDwl3syEakgETNAsxXhTR0P0rVikNw5i7Hmtc+ZC25zIwqGgRkOM3BtB/M+tosvLfouL4umdIYgIsdkGSY3xpJ8efFttH5sH4NXdWAE5sblttn/VfHIs4z63rGW4ouT3NT0CC3W7P9ji8jJa/cZvK7lYdJ/kmbwbRuwWppn/Yph1m+UmZEI9rwmuq/Zz2vbHuWqSB7dVBaR45Eww1wbydG76m5urT0H995GzJkUTiZT7tFOm9kdBdPi4LvX4NswyQ8W/IB2ywLm3kfMROTkvLZmDxct3cMrPnAz/s1r6fjMw7P2yaqzdh/FamrCWLeM0llpXr5gCwt8IWKmgiAiJy5hhlnk93H1om1kzsphrlmO1dhQ7rFOi1m7UshsnM/BGwz+aeOtvCyaAvS0UxF54YKGny+2Pc55sb188C03Mv/HYfy/GS/3WKfcrFspmKEQzoVrGTrfx5vOf4A1gaFyjyQis8jZoV5uuvAhBi8M4ly0btbdep5dUTAtjHicsdURfCun+WjTNhboAXcicgot9Mf4h+ZnMFclGVsdxohFZ9UnkmbV9pG1fBHJFXXc/I47uCSyG4iUeyQRmaW+t+HfuXf5Er4zeA3xrePYO/eUe6RTYnasFEwLMxJhfEM9w+fAldGdLPLreUYicvos8Qe4KrqdoXNMJtc3Ht5GMqv/7HJWRMGMRjBbmzFfO8Ltr/wiXb6IXqMpIqeV37CY74vwrRv/hcyrk5hNjZih6v9mtOqjYPh85M9fyra/beb13Y/S7XP1+AoROSMsw2SRP8frFz3K9vd1UDx3WdU/PK+6v3qaFlZTI5NLAvz1hb/hmuh2EqZuK4vImdNoRbkhvoU/u+h3TCwNYjU1VvXBc1VHwWqoZ9vHOln+2u28tXYbXT4dLIvImbfQF+ZvGjaz6PW72PbRLqyG+nKP9IJV7TrHt6Cb7MJG1i07wLWNW3RbWUTKxjJMYkaIaxqfIbfcT2ZVN8F9EUoHe8s92gmr2pXC5Llt9F4R4Ivzf8Tr4rPvVqGIVJ831Yzw/8//Ab2XBUlubC/3OC9I1UXBjEbx9cxn8MU2r7/yPhrNufGMcxGpDk2Wj1dd+wD9l7v4euZjRqPlHumEVF8UahNkljSxYkkf72p4goiiICIVJGaGeG/joyxd0k96aRNmTbzcI52QqoqC4Q8wdUEXK/5+Cx/rvkOfNBKRipQww3x8wY856+PPMH1+d1V9TLVqomD4AxQvWsX4SoPr6zbT6SuWeyQRkT9ovq/A9XVPMrbSwr5gVdWEoWqiYEbDHHipn/kXHuKKSJFmq7r26URkbmm2olwVydN6UT8HrglhhKtjZ6Mq0mUtXUR6ST2vvvQhrk88id6NICLV4mML7+D2+o08/dt1hPeMYe/ZX+6RnlflrxRMi+yCOsZX+HhN3aOcF1IQRKR6XByCN9Q/xPhZfvLd9RX/0LyKjoLhD2A1NXDg1Q5feMvXWOqv7L9MEZFjWe6Hj/7lt9l3o4nVUI/hr9xPTVZ0FKyOVsauXsjy+YOsCkzi07aRiFShoOFjfXCA+T0jjF+9CKu1udwj/UEVHYVCdwPFl0/yxvaHafPF9PRTEalKlmGywB/jzzofJnXDNMWuxnKP9AdV5ldZ08JYt4LhjWH+ZdWtXBbpK/dEIiIn7Zrofv7P2tsYPieCsW5FRZ4vVOSnjwy/j+SyOKkemwtCJqCPn4pI9Wu2ojSHHWZ6bCJDcRLPWriOXe6xfk/lrRRMC7M2QcPNB/ns5beVexoRkVPuH6/8DjVv6cOsiVXcuxcqLgrmWYtJXryASxp3sTY4UO5xREROubXBAS5p2s3MxYuxli8u9zi/p+KikFxZy+CFBlfGtrLQHyv3OCIip9wSf5Qr488weIHB9PK6co/zeyomCkYwiK+7k8GLHT551XfpqcjTDhGRU2Op3+Ej1/2AgYsNfB3tFXN3oWKiYMZjpM9qpb5ziqujA3qTmojMamEjwEujh4h2TZNd3nb4fKECVEwUSkvmUXznOO9ecrceiS0is55lmNRZEd617B6S75zBXthR7pGACooCgN908Bulco8hInLG+I0SAZ8NZmV8CqlyouBCpugn5/jLPYmIyBnjYFKyLXDcco8CVFAUfLt6iX+hhs/vuJykk8V2nXKPJCJy2tiuw4id5nPbLqfmC3Gs3ZXx5IaKiYIznSK0rZ+ZQzX8KNVN1i2UeyQRkdMm5ea5fWYJ2QNxQtv7cVLpco8EVFAU3GKB0uAQ7ffBJ37yp+wpVsb+mojI6bCz6OMLd1xPx/0OpcEh3GJlfCNcMVE4qmbbBG0P2twxvY6thWy5xxEROeWeyBf4/uQ5tD9QIrZtvNzj/J6KuyJm79hDbLSe3w4voTs4xorASLlHEhE5pR7KLObegcU0/W4n9vR0ucf5PRW3UsB1cZIzlP6thb+/6xU6cBaRWcN2HfJukS/deQ3Rf03gZDLlHun/UXlRAFzbpmbHFPF9Jj/LxOgrpco9kojISeu3M9yRbiS+DyK7xnHtynpsNlRoFHBs7K07aX48zbseeg13pJaXeyIRkZP2g+nVvO93r6LpyRT27n3gVsbdhOeqzCgc4T80SvtP/HzrwLlsKeQoupVXVRGRP6bo2mwp5Lhl/0Y67zSxeiv3rLSio2APj1Dz6+2M7Gvgd5nF5N1iuUcSETlhebfI7zKLmdxXT/RXz1IaGSv3SH9QRUfBLZWwZ2ZYdGuOr3/xOrYVLR08i0hVKbo2mwshvvGF61j0nSxONgsV9grO56roKADguvh7x6nfmeNrI5dydzZY7olERI7bXZk4Xx26lLodWfy9YxV5jvBcFXdP4VhKvX34ppI88PM1PHV2B1es/165RxIROS4f2noD9qN1dD35NKV0ZTzK4vlU/krhCDebpeuXKTIPNnLLTAOH9DFVEalg+4spvppsx32gjs5fTePm8+Ue6bhUTxRKJXjkGRq2lfj+0Eb2FWt0viAiFcl2HXYWG/j+wAYathZxH3/28NewKlA1UQDAdYk9tJ/MB9v4u73XM2hnFAYRqSiHH4md4e/3XIv1/loij+4t90gnpLqiALgzM/j3DdG7rZWPDV6pR2yLSEVJuXk+NHAVw1ubMQ8MVcwjsY9X1UXByeUoDQ7R9Qubh29fw7BdHUsyEZkbhm2HR3+wmq67itijo1VzlnBU1UXhqMjTvbTfn+aN29/IFyZ6yj2OiAifHFvK6575X7TfP0P42cp4k9qJqtoolAaH8O3qZWRLCz8bWknSyeoxGCJSFkXXJulk+fnACpJbGrB291EaGi73WC9I1UYBwJ6YZPGndpD8TgcfH76APcXqWqaJyOywq1jgH0ZeRP57LSz67A7sqalyj/SCVXUUcF3sqSnqdue4495z+NH0Okbs6jrUEZHq1ldK8YPkBn7023Op3ZU9HIQKv7X8fKriRvPzcl18j2xn6a5avrNwPS9atZuGUAnLqO7eiUjlK7o2TxcauW3nBpZ+dj/OVBKnioMA1b5SOMLJ57EnJonfUsPNt93M3lJWT1QVkdMq5eTYWijxnm++meZbw9jjkzhV9kmjY5kVUcB1cfN5Ek+O0PqIzU9nVrG9oEttInL6bCta3DG9jtZHCsSfHMQtFqp62+gow3WP709xufmq0z3LyTMMrPo6Rl6+lPSVKbZf8K1yTyQis9TCe/4XTb8IUv/T7djJ6aoIwq+d7//R3zM7VgpHuS5OKk3Ds2mMLXHeO7SOvUU9OE9ETp1dxTT/e+Bsws+Gqd06g5POVkUQjtfsigIcvj24aQttD+W5/b5zeSzXWe6RRGQWeSi7gLvu3kjrphzuk1sPbxvNItX/6aM/IPjUfpYONfHB+pezaeUWPtv6CH7DKvdYIlKl8m6Rd/Rdym+2nMXy/xqHwVFm43XZWbdSOMoen8DeupPIthA/330Wu4oFkk623GOJSBWatDNsLzjcs2sJNdv82Nt2YU9Olnus02J2HTQfgxkKwbIesp/JcFPno7w1MVDukUSkynxhoodb9m+k+QMm7u79VfeQu6Pm3kHzMTj5PObQOJM/a+fzWy7jZ5mQVgwiclwm7Qw/TNXw5acvwbmrEWN4vGqDcLxmfRRwXUpDw7R95Qki98W4Zfh8Bkqz55MCInL69Nom/zX4Imp/G6Llq49ij46We6TTbvZH4Qi3UKD9530Mf7iHv9r9Wr6XSuitbSJyTLbr8J2ZOt6x87VkP9JGyy97q+Z1midrzkQB18XuGyD4+G4ObW3j630XMmJnyDiz6+NkInJyMk6BQTvDf/a/iP6tLfge30Wpf7DcY50xcycKgFsqYU9Ps/QTe8l9tp1vTG3g2aJR7rFEpII8WzT4z6mN5D/dxtJP7cVJp8GZjR8+PbY5FYWj3JkZInsn+OaPXsLf7n4Vj+aLWjGIzHEZp8Cj+SLv3vlqbrn9xYT3TeBMT5d7rDNuTkbByeVw9vey8Ou9DD7Sxm9mVpJ0CjpjEJmjbNdhwinwq5lVjG9qpec/DuEc6Jv1nzQ6lll/T+H5GP4A5oJOsj31xD/Qy5+1PcSfxubedwYic92P0zH+vf8i0p/qOLxCONA36x5fAcd3T2HWPubieLjFAu6hfsLJGZ59qocvFwOsWnwbLZZJwgyXezwROc0m7QxDNnzp4PXs39rOss17caanZ2UQjtec3D56LieXwx4eYdmHt+F+pplPDV7FI7maco8lImfA73KNfKz/OvhUE8s+vA17dHRObhk915yPwlFOJkN49yhPfXMV73zyNfxHslU3n0VmqUk7w1emOnjP469i9zeXEtozgpPJlHusiqAoHOGWSpQOHKL165vxPxzn1v5z6Cuh13qKzDJ5t8iAbXBr7zlEHo7S9PUnKB3qmzOX0/4YReG5XBcnn6fz1r343xvn5Q//Je8dfFG5pxKRU+h9Q+fz8k03E31vkPbbds+a12ieKnP6oPmYXJfS8Ajm9AzBzWu5M7eac+P7OC90kIX+WLmnE5EXaFcxzaZsNz95Zg3RbUHYtRk7lyv3WBVHUTgW18XJZGj/7EOYq5fx4ZlX8pqLHuaTLVvKPZmIvED/NnYRP77vHJb9xyT21ifQraRj0/bRH2H0j7DgjhI/+vGFnPXQ69layOqcQaRKZJwCT+QLLLn/jdx1+3nM/2kRBkbKPVZF00rhj7DHJ/DdPcG8/FrGR+Lcs3wZ/tg2FvosLENNFalUtuuwv2Tzm9Rqan4dpeGZGXj0mVn5Cs1TaU7faD4RRjCIGY8xc9Eihs41+f6rv0i3z6bOipR7NBH5H8bsNDuLYd58y9tpfcQm+uBunExmzt9B0JvXTiE3n8cenyC+bZzGLS5/sfUNfHt6OXuLKT0zSaRC2K7DrmKabyRX81dbXkfjFofYthHsqak5H4Tjpe2jE+G62Dv3UNs7gPFoC194z5XkLvTxttqtxIxQuacTmfNSbp7bps7mv+6/iOWfG8AZ3k9JnzA6IYrCC+Dk8pjDYyz4QT23brmSJ17XzTUNW3hjzVi5RxOZs76abOenI6vpv2UBPbvzOKPjOAV9KOREKQovhGPjzMwQuO8Z2rc38siaheSW+bko/EOaLB8xU6sGkTMl5eQYtkv8cGA9u7d3sOyOvTjjEzi6ofyC6KD5ZJkWVn0t+dXz6b08wI1XP8DfNDyiA2iRM2DETvPpkYv46V3n0vXrHP5nD2JPTOqG8h+gR2efCY6NPTZOaG+E5vp2bqk7n8eXdPF3839Cty9Lm0+3oEVOtb5SioOlCB/aexMHd7bSsdkhsG+U0vhEuUereorCKVI62Ev0YC9nbZ5Pemk73/jYhVxf/yTX+nTIJXKqPZ5v5fax9ZifamT53gFKB3vRZtGpoSicYs7wKNFsjm3/uJr7V67jCxf38fGeH3NBSJ/+FTkZtuvwu5yPv9t7PSP3tdO4tURs6wGcmVS5R5tVFIVTzEmncdJpondOEJxYyYF4B7fXbSRUv4nlfggaPt2EFjkBtuuQd0s8WzT4wcT59G9uY/59GaxNz2LrMPmU00HzaWT4A5jhEJkLlzJ+lp+/e+u3WR8cYIGetipy3PYXU2zOt/PxL7+e+u0FQg/txM1m9f6DF0AHzWXmFgvYxQKR3eOYhTrec89r6Fwwyk2dj3JDbKcOoUWeR18pxR2p5dx26Gz6DzTSsyVP6MA4pZmZco82qykKZ4C9ex/+PQbLn6xj8oolfO6llzP/nDGaray2kkSOoejabM4384XNl9H6kyDL796FPTFJSR81Pe20fXQGGT4fVmsLxc5GRs6OMtPj8Imrv8uaYD/LA7rXILK9kGFzvpOP3vkq4vtMmjen8PWOYQ+PaLvoFND2UYVxSyVKff0Yff20pZcRHUzwtZUX8SfNu7g6voWlfoewEdDqQeYU23VIuXn2FC1+Or2Re0cW0/agS3znGPa2Xfqo6RmmlUK5mBaGZWHWxJi5eDGDFxh86LrbuS66n0YrWu7pRM6YETvNj1OL+cxPb6D9AZvYA3twplO4paJuJp9iWilUMsfGdWzsiUniOydxzTo+Hng5/9Q1zf931s9ZGxxgiV9xkNlrayHLE7lOPrv9JjIHamh/2CG2a1KPqSgzRaHcXBd72y6i22D5452kV7bytXdfzOs6HmGhb0hbSTIrFV2be9LLuOXg2TT9nzDh7Yco9Q/orWgVQNtHFeTo291Ki+cxtTRCchFcffVjXJN4misiegSwVL9fZIL8ZHIdv71zPYm9Dondaay9/Ye3i4qFco8362n7qMq4+Tx2Po8xNk7j+EIio/Xc2b2KfV2NZNofYLF/lHk+iBlBrSCkKhw9RO4rwe5iE//efxHbDrSz4ME8oX1jlPYf1OqgwmilUKkMAwwTM+DH6Gxn6LIW8pdN88GVv+CGaL/e2SBVIelkuSPVyT8+exXBe2po/8UAdv8gbrEEjnJwpmmlUM1cF1wbJ2djDY/R+FSciUINH99xI/+yfoSzmw7xnubfUm/qpT5SWVJOjlG7xOdHXsLjo51Mbm6iZh/UbU/jjI7rXckVTlGoAvb0NMbDT9PwiEVTOMToTav56ZpGXnzldlYFhvAbRXxY2lKSsrJdhxI2A7bNM/l2fr5pLfVbTHq+9SROPg+ui1PuIeWP0vZRtTEMfC3NuIk4uc4EIxuCpBcV+cCFP+Oc0H7WBoPlnlDmoCfyBTZlF/K5B64itttP8xN5gv1JjOQMpeERfcS0Qmj7aDZyXUpDwzA8QuBAgFZ7BRPTQb7UeCkrmpdxTcMWVgQH6PQVaTDDWj3IaWG7DiN2hgE7wNZ8Oz8ZXcv2kRbqn/BRtzOH78FnsXX5rCpppVDtjhxIG34f5sJuhi5poHhFkjcufoS3124nYgbKPaHMQiknxz9PrOHbu84m9Js4rb8dxdl3CNe2wXUUgwqllcJccORA2s3bMDxG8+NBJlMJ/qv9Sv5t9cUsaB3jDR0Pc2nkAF16VLechEOlFPdkevhW33kcGGogsiVMzYBD7c4UjIzrnsEsoSjMIvb4BIxPUPsY1AWD5F6ymoE1ndxymUttV4a4MUbQ8OE3LPyGVe5xpQoUXZuia5N3S2zOt3JL/7kM/3oenVuKhO55AjefxwXdNZhFtH00WxkGVjyOEY3g1idILUqQbrVIXpLlnO6DvL/jLnp86OOsckwpJ8eeosEn+69h88Euan8bIjpkE92fxJhI4qYz2NPT5R5TTpC2j+Yy1z38f9rpaRgcIj7VTqS5jmIswabxJbwzU8M5jQdZEBzl7PB+WqwC87S9NKf1lVIM2wEeyy5gb66Zx8a6ObS9ldhBk6bHJjFHJikNDpV7TDnNtFKYawwDw7IwfD4yl61mapGPumsGuL59C39dt1vbSnPQ0fsFX5w4izv7VzFzVyt1u4uE7t5y+OaxDo5nDa0U5P/lurilEq5tE9s2Qng4ztRYK//Z3M6/tl5JbOkkyxuH+d9tv6Hbl9V7pGepvlKKg6UIXxq8jO2jLWR21BIZMgiPOrTtmsEam6ZUKCgGc5CiMFe5LqV9B2AfJB6Dungcs6GO/pfOY9OiWnouGeOc6F7OZoSQYeI3TIKGHxNDdx+qjO06OLjk3SJF1yHnOjyeb+XRVA+PPbaEml0mS+48hDMxiZNO44LedjaHaftIDjuyrWQmajBCIdyaKLnOBDPz/IxvcIh1TPPOpb/l7PABVgd0OF1NthRybMr28KUdl5Luj9Ow2SR+qECofxpjOo2by+Ekp4/cMdDKYDbT9pEcvyPbSvb4xOEf90N4soXAaCNQQ6a/jk+mrqa1Mcmy2hFWxfto90+yKjBIqwV1VqSs48thk3aGAdvgmXw7A8U6tqba2TbZwshYDaFtYRqGXBqemsYcGMUeHin3uFKBFAX5g0pDwzA0TP2TUH/k53xtrfS3d/PI5atJzy9x7YaneWX9Y1wa1qPOys12HR7LJ/jx5Abuenw10QM+5v0mSf3AODWDe73fp3sF8nwUBTkhTnIas1hk3t0mhUSAxx5az/1NG8jXuRS78iQSGda29HNJ7U7WBntZEfDpE02nWNG12VKweSrXxQPJxTw13EEyGSFwIEhw0iA05tIzUMCfzGAeHMJJpcs9slQRRUFOiJPJQCYDY+P4gVqgoa4OGmqZOLeFdFuIe5fEGViYoK+pnkL8GWrNHCHDJWQY+DGImH496vs4FF0bB4eMU6SIS851ybkGU06Au6bX8sDYQvbsbSW+00/LoEP9Q/24U0nsqSSgFYG8MDpolpN39O5DOIzh82EEAxAM4Pp9lFoSZNqCTCyzyPYUaGpJ8uYFD7E2dIjzQlpBPJ/7c7A5O59b9p/D2FAN4f0B6nfYRIby+IaTGPkiFIu4+QJuqYSTzuhOgTwvHTTLmXH07sPMzO//vGHgm6yjZqweK1fPzJifdH0Tnx64hkBNnvb6JIlAjoivQGMwRY0vR70vTaNvmhorR6uVJGHmabBc6sxQ1W9D5d0iE3aeKcck6QQZshNM2yHGSjWMFWOk7CBj+RipUpDpfIiBiRqKySCRA37qJ1zifUUiuydgdAJ7crLcfxyZpRQFOX1c13tIX2gnPPeDrIbPhxmLku9uZ6Yhwu7uAPk6g1yjS7G1SLgmx8rWQZbHhzgnupdzg+PUVfH7IWzXIekUeCzfzKPpheyYaWHrUBu56SD+YT/hUYPgpEv8UIHAWJrIgQF6Mn24pd+/MaDtIDndtH0k5XH0XkQ8jhEM4NbEcAN+nIgfO+rHDpiUohZ2wKAUNLBDYAcMCgmwwy52yMWO2Rghm3AsTzRUIBHKMS86RcKfpSs4QczKETXzxM0sAcMmYuYJGUVChk3cKOE3IGqYxMygtwopujYpJ0/adSi6MOP6yLkWOdfPjBOi6PpIO0HSTpCUHeJQvp6JQoTBTIJkLkQ6FyCbCuLmLKyUhZU1sHIGgSRYBRcrB768i1Vw8aVtrIKDlS5iZooY+QLGTBo3X8CZmdG9ATnltH0klevovYij2yBDw94vWUf++Z+vBzL8AczuDpxEhGIiRLbJTzHiI18fYiYGk1GXPfUtWJES9Yk0EX+RiL9ATSBHwCwR8xWIWnkiVoGElSVi5olbORqsFCGjCEDaDTFeamHGCZN3/CTtMBk7QNoOMl0MUXAsUsUgmWKAdCHA5HQEO+PDN+7Hlzbwp6Fx3MWfcQmNFfEnc5jJDM7B/ud934A+0CuVQlGQquEWC9j7DmGYBn7LImBZh988Z5oYlgmGCUf+bVhHtpkMg5kjjwcfP/Jjnvtv83/8+Oh35o77+z/2ft7BwCHqZomSpcmePHy4azvgOri2A87hw17XtsG2sR0XHG38SHVQFKS6ODauA5RKaGNF5NSrzlM7ERE5LRQFERHxKAoiIuJRFERExKMoiIiIR1EQERGPoiAiIh5FQUREPIqCiIh4FAUREfEoCiIi4lEURETEoyiIiIhHURAREY+iICIiHkVBREQ8ioKIiHgUBRER8SgKIiLiURRERMSjKIiIiEdREBERj6IgIiIeRUFERDyKgoiIeBQFERHxKAoiIuJRFERExKMoiIiIR1EQERGPoiAiIh5FQUREPIqCiIh4FAUREfEoCiIi4lEURETEoyiIiIhHURAREY+iICIiHkVBREQ8ioKIiHgUBRER8SgKIiLiURRERMSjKIiIiEdREBERj6IgIiIeRUFERDyKgoiIeBQFERHxKAoiIuJRFERExKMoiIiIR1EQERGPoiAiIh5FQUREPIqCiIh4FAUREfEoCiIi4lEURETEoyiIiIhHURAREY+iICIiHkVBREQ8ioKIiHgUBRER8SgKIiLiURRERMSjKIiIiEdREBERj6IgIiIeRUFERDyKgoiIeBQFERHxKAoiIuJRFERExKMoiIiIR1EQERGPoiAiIh5FQUREPIqCiIh4FAUREfEoCiIi4lEURETEoyiIiIhHURAREY+iICIiHkVBREQ8ioKIiHgUBRER8SgKIiLiURRERMSjKIiIiEdREBERj6IgIiIeRUFERDyKgoiIeBQFERHxKAoiIuJRFERExKMoiIiIR1EQERGPoiAiIh5FQUREPIqCiIh4FAUREfEoCiIi4lEURETEoyiIiIhHURAREY+iICIiHkVBREQ8ioKIiHgUBRER8SgKIiLiURRERMSjKIiIiEdREBERj6IgIiIeRUFERDyKgoiIeBQFERHxKAoiIuJRFERExKMoiIiIR1EQERGPoiAiIh5FQUREPIqCiIh4FAUREfEoCiIi4lEURETEoyiIiIhHURAREY+iICIiHkVBREQ8ioKIiHgUBRER8SgKIiLiURRERMSjKIiIiEdREBERj6IgIiIeRUFERDyKgoiIeBQFERHxKAoiIuJRFERExKMoiIiIR1EQERGPoiAiIh5FQUREPIqCiIh4FAUREfEoCiIi4lEURETEoyiIiIhHURAREY+iICIiHkVBREQ8ioKIiHgUBRER8SgKIiLiURRERMSjKIiIiEdREBERj6IgIiIeRUFERDyKgoiIeBQFERHxKAoiIuJRFERExKMoiIiIR1EQERGPoiAiIh5FQUREPIqCiIh4FAUREfEoCiIi4lEURETEoyiIiIhHURAREY+iICIiHsN1XbfcQ4iISGXQSkFERDyKgoiIeBQFERHxKAoiIuJRFERExKMoiIiIR1EQERGPoiAiIh5FQUREPP8XIHL0X/FyDgcAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(- (+ (Circle 4 6 0) (Circle 8 8 0)) (- (Circle F 9 8) (Circle 8 8 8)))\n",
      "                                                  ^ --> B\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  5%|▌         | 529/10000 [00:17<00:43, 217.33it/s]"
     ]
    }
   ],
   "source": [
    "current_expression = sampler.sample(\n",
    "    env.grammar.start_symbol, min_primitives=2, max_primitives=7\n",
    ")\n",
    "\n",
    "for _ in range(5):\n",
    "    print(current_expression)\n",
    "    current_image = env.compile(current_expression)\n",
    "    plt.imshow(current_image)\n",
    "    plt.axis(\"off\")\n",
    "    plt.show()\n",
    "\n",
    "    mut = random_mutation(current_expression, env.grammar, sampler)\n",
    "    print(mut.pretty(current_expression))\n",
    "    current_expression = mut.apply(current_expression)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Find confusing mutations.\n",
    "\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "jax",
   "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.11.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
