{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Topological regularization of circle embedding\n",
    "\n",
    "In this notebook, we show how a topological loss can be combined with a linear embedding procedure, as to regularize the embedding and better reflect the topological---in this case circular---prior. \n",
    "\n",
    "We start by setting the working directory and importing the necessary libraries."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Set working directory\n",
    "import os\n",
    "os.chdir(\"..\")\n",
    "\n",
    "# Handling arrays and data.frames\n",
    "import pandas as pd \n",
    "import numpy as np\n",
    "\n",
    "# Pytorch compatible topology layer and losses\n",
    "import torch\n",
    "from topologylayer.nn import AlphaLayer\n",
    "from Code.losses import DiagramLoss, pca_loss, ortho_loss\n",
    "\n",
    "# Ordinary and topologically regularized PCA embedding\n",
    "from Code.topembed import PCA\n",
    "\n",
    "# Plotting\n",
    "import seaborn as sns\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# Quantitative evaluation\n",
    "from sklearn.svm import SVR\n",
    "from sklearn.multioutput import MultiOutputRegressor\n",
    "from Code.evaluation import evaluate_embeddings\n",
    "\n",
    "# Representative cycle analysis with alpha-filtrations\n",
    "import networkx as nx\n",
    "import diode\n",
    "import dionysus\n",
    "\n",
    "%matplotlib inline"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Construct data and view restriction to first two coordinates\n",
    "\n",
    "We now construct a high-dimensional data set sampled from a circular model. The model occurs only in the first two dimensions, and the high-dimensionality of the data is directly caused by random noise."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Seed for reproducibility\n",
    "np.random.seed(420)\n",
    "\n",
    "# Sample parameters\n",
    "npoints = 50 # number of points\n",
    "ndim = 500 # data dimensionality (with noise)\n",
    "sigma = 0.45 # magnitude of noise per dimension\n",
    "\n",
    "# Sample data\n",
    "t = np.random.uniform(low=0, high=2 * np.pi, size=npoints)\n",
    "X = np.concatenate([np.transpose(np.array([np.cos(t), np.sin(t)])), np.zeros([npoints, ndim - 2])], axis=1)\n",
    "N = np.random.uniform(low=-sigma, high=sigma, size=[npoints, ndim])\n",
    "data = X + N\n",
    "\n",
    "# Plot the first two dimensions of the data two illustrate the circular model\n",
    "fig, ax = plt.subplots()\n",
    "sns.scatterplot(x=data[:,0], y=data[:,1], s=50, hue=t, palette=\"husl\", ax=ax)\n",
    "ax.get_legend().remove()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Conduct ordinary PCA embedding\n",
    "\n",
    "We now explore how well the ordinary PCA embedding is able to recover the model from our data."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Time for embedding: 00:00:00\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Conduct PCA embedding\n",
    "Y_pca, W_pca = PCA(data, random_state=42)\n",
    "\n",
    "# View PCA embedding (flip for consistency)\n",
    "fig, ax = plt.subplots()\n",
    "sns.scatterplot(x=-Y_pca[:,0], y=Y_pca[:,1], s=50, hue=t, palette=\"husl\", ax=ax)\n",
    "ax.get_legend().remove()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We see that due to the noise in high dimensions, while the overall ordering points is good, PCA is no longer able to effectively recover the circular hole."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Apply topological regularization to the embedding\n",
    "\n",
    "We now show how we can bias a linear embedding using a loss function that captures our topological prior.\n",
    "\n",
    "The model we will use for this learns a linear transformation $W$, which is optimized for the following three losses:\n",
    "- the reconstruction loss between our data and its $X$ low-rank approximation $XWW^T$;\n",
    "- a loss function encouraging orthonormality of $W$, i.e., such that  $WW^T$ is a projection matrix;\n",
    "- a loss function that captures our topological prior.\n",
    "\n",
    "As a topological loss, we will use the persistence of the most prominent cycle in our embedding. It is important to multiply this by a factor $\\lambda_{\\mathrm{top}} <0$, since we want this persistence to be high. To obtain this loss, we require an additional layer that constructs the alpha complex from the embedding, from which subsequently persistent homology is computed."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define topological loss\n",
    "def g(p): return p[1] - p[0] # function that returns the persistence d - b of a point (b, d)\n",
    "TopLayer = AlphaLayer(maxdim=1) # alpha complex layer\n",
    "CircularPersistence = DiagramLoss(dim=1, j=1, g=g) # compute persistence of most prominent cycle\n",
    "lambda_top = -1e1 # scalar factor that trades off embedding and topological loss\n",
    "\n",
    "# Construct topological loss function\n",
    "def top_loss(output):\n",
    "    dgminfo = TopLayer(output)\n",
    "    loss = lambda_top * CircularPersistence(dgminfo)\n",
    "    \n",
    "    return loss"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We can now conduct the topologically regularized linear embedding as follows."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[epoch 1] [emb. loss: 0.063185, ortho. loss: 0.001478, top. loss: -1.539055, total loss: -1.474392]\n",
      "[epoch 50] [emb. loss: 0.067497, ortho. loss: 3410.821045, top. loss: -0.782473, total loss: 3410.105957]\n",
      "[epoch 100] [emb. loss: 0.066744, ortho. loss: 1016.062500, top. loss: -3.505133, total loss: 1012.624146]\n",
      "[epoch 150] [emb. loss: 0.066747, ortho. loss: 1201.616699, top. loss: -5.466840, total loss: 1196.216675]\n",
      "[epoch 200] [emb. loss: 0.066591, ortho. loss: 555.281982, top. loss: -8.052293, total loss: 547.296265]\n",
      "[epoch 250] [emb. loss: 0.067223, ortho. loss: 6603.874023, top. loss: -2.147141, total loss: 6601.794434]\n",
      "[epoch 300] [emb. loss: 0.066911, ortho. loss: 1616.312256, top. loss: -5.034025, total loss: 1611.345093]\n",
      "[epoch 350] [emb. loss: 0.066875, ortho. loss: 1885.948242, top. loss: -3.005395, total loss: 1883.009766]\n",
      "[epoch 400] [emb. loss: 0.066544, ortho. loss: 377.594055, top. loss: -7.798023, total loss: 369.862579]\n",
      "[epoch 450] [emb. loss: 0.066521, ortho. loss: 311.904297, top. loss: -6.428357, total loss: 305.542480]\n",
      "[epoch 500] [emb. loss: 0.066493, ortho. loss: 310.406158, top. loss: -8.081623, total loss: 302.391022]\n",
      "Time for embedding: 00:00:05\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Learning hyperparameters\n",
    "num_epochs = 500\n",
    "learning_rate = 1e-1\n",
    "\n",
    "# Conduct topological regularization\n",
    "Y_top, W_top, losses_top = PCA(data, top_loss=top_loss, num_epochs=num_epochs, \n",
    "                               learning_rate=learning_rate, random_state=42)\n",
    "\n",
    "# View topologically regularized embedding (flip for consistency)\n",
    "fig, ax = plt.subplots()\n",
    "sns.scatterplot(x=Y_top[:,0], y=-Y_top[:,1], s=50, hue=t, palette=\"husl\")\n",
    "ax.get_legend().remove()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Compare with ordinary topological optimization\n",
    "\n",
    "For comparison, we also conduct the same topological optimization procedure directly on the initialized embedding.\n",
    "We observe that the results were much worse than when we accounted for the embedding loss."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[epoch 1] [emb. loss: 0.000000, ortho. loss: 0.001478, top. loss: -1.539055, total loss: -1.537577]\n",
      "[epoch 50] [emb. loss: 0.000000, ortho. loss: 3397.123535, top. loss: -0.791807, total loss: 3396.331787]\n",
      "[epoch 100] [emb. loss: 0.000000, ortho. loss: 543.927490, top. loss: -6.307951, total loss: 537.619568]\n",
      "[epoch 150] [emb. loss: 0.000000, ortho. loss: 1442.463501, top. loss: -3.729107, total loss: 1438.734375]\n",
      "[epoch 200] [emb. loss: 0.000000, ortho. loss: 1797.159668, top. loss: -5.906724, total loss: 1791.252930]\n",
      "[epoch 250] [emb. loss: 0.000000, ortho. loss: 1037.598145, top. loss: -6.311464, total loss: 1031.286621]\n",
      "[epoch 300] [emb. loss: 0.000000, ortho. loss: 1997.365601, top. loss: -3.227956, total loss: 1994.137695]\n",
      "[epoch 350] [emb. loss: 0.000000, ortho. loss: 3691.520752, top. loss: -3.611770, total loss: 3687.908936]\n",
      "[epoch 400] [emb. loss: 0.000000, ortho. loss: 1454.978638, top. loss: -7.833032, total loss: 1447.145630]\n",
      "[epoch 450] [emb. loss: 0.000000, ortho. loss: 6961.562500, top. loss: -2.408313, total loss: 6959.154297]\n",
      "[epoch 500] [emb. loss: 0.000000, ortho. loss: 1272.623657, top. loss: -2.707522, total loss: 1269.916138]\n",
      "Time for embedding: 00:00:05\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Learning hyperparameters\n",
    "num_epochs = 500\n",
    "learning_rate = 1e-1\n",
    "\n",
    "# Conduct topological optimization\n",
    "Y_opt, W_opt, losses_opt = PCA(data, 2, emb_loss=False, top_loss=top_loss, num_epochs=num_epochs, \n",
    "                               learning_rate=learning_rate, random_state=42)\n",
    "\n",
    "# View topologically optimized embedding (flip for consistency)\n",
    "fig, ax = plt.subplots()\n",
    "sns.scatterplot(x=Y_opt[:,0], y=-Y_opt[:,1], s=50, hue=t, palette=\"husl\")\n",
    "ax.get_legend().remove()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We observe that the results are highly similar."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Quantitative evaluation\n",
    "\n",
    "First, we evaluate the different losses (embedding and topological) for all final embeddings."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[1mLosses for pca embedding: \u001b[0m\n",
      "Reconstruction: 0.06318476796150208\n",
      "Orthonormality: 1.7184031883525453e-15\n",
      "Topological: -0.15390551090240479\n",
      "\n",
      "\u001b[1mLosses for topologically optimized pca embedding: \u001b[0m\n",
      "Reconstruction: 0.06680672810017917\n",
      "Orthonormality: 0.21647799015045166\n",
      "Topological: -0.34679036140441893\n",
      "\n",
      "\u001b[1mLosses for topologically regularized pca embedding: \u001b[0m\n",
      "Reconstruction: 0.06649286039989714\n",
      "Orthonormality: 0.028079602867364883\n",
      "Topological: -0.7508018016815186\n"
     ]
    }
   ],
   "source": [
    "print(\"\\033[1mLosses for pca embedding: \\033[0m\")\n",
    "print(\"Reconstruction: \" + str(losses_top[\"embedding\"][0])) # PCA initialization gives first embedding loss\n",
    "print(\"Orthonormality: \" + str(ortho_loss(torch.tensor(W_pca)).item()))\n",
    "print(\"Topological: \" + str(top_loss(torch.tensor(Y_pca).type(torch.float)).item() / np.abs(lambda_top)) + \"\\n\")\n",
    "\n",
    "print(\"\\033[1mLosses for topologically optimized pca embedding: \\033[0m\")\n",
    "print(\"Reconstruction: \" + str(pca_loss(torch.tensor(data - data.mean(axis=0)), \n",
    "                                        torch.tensor(W_opt), torch.tensor(Y_opt)).item()))\n",
    "print(\"Orthonormality: \" + str(ortho_loss(torch.tensor(W_opt)).item()))\n",
    "print(\"Topological: \" + str(top_loss(torch.tensor(Y_opt).type(torch.float)).item() / np.abs(lambda_top)) + \"\\n\")\n",
    "\n",
    "print(\"\\033[1mLosses for topologically regularized pca embedding: \\033[0m\")\n",
    "print(\"Reconstruction: \" + str(pca_loss(torch.tensor(data - data.mean(axis=0)), \n",
    "                                        torch.tensor(W_top), torch.tensor(Y_top)).item()))\n",
    "print(\"Orthonormality: \" + str(ortho_loss(torch.tensor(W_top)).item()))\n",
    "print(\"Topological: \" + str(top_loss(torch.tensor(Y_top).type(torch.float)).item() / np.abs(lambda_top)))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We also compare the magnitudes of the new projection weights with those from the ordinary PCA embedding."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 1080x216 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Compute the magnitudes of the weights by dimension\n",
    "magnitudes = pd.DataFrame(np.array([np.sum(np.abs(W[2:,:]), axis=1) for W in [W_pca, W_opt, W_top]]).ravel(),\n",
    "                         columns=[\"magnitude\"])\n",
    "magnitudes[\"type\"] = (data.shape[1] - 2) * [\"pca\"] + (data.shape[1] - 2) * [\"top. opt.\"] + \\\n",
    "                        (data.shape[1] - 2) * [\"top. reg.\"]\n",
    "    \n",
    "# Compare the magnitudes\n",
    "fig, ax = plt.subplots(figsize=(15, 3))\n",
    "sns.kdeplot(data=magnitudes, x=\"magnitude\", hue=\"type\", ax=ax, linewidth=4)\n",
    "ax.set_xlim(0, 0.2)\n",
    "ax.set_ylabel(\"density\", fontsize=20)\n",
    "ax.set_xlabel(\"feature importance (irrelevant features only)\", fontsize=20)\n",
    "ax.tick_params(axis=\"both\", which=\"major\", labelsize=15)\n",
    "plt.setp(ax.get_legend().get_texts(), fontsize=\"20\")\n",
    "plt.setp(ax.get_legend().get_title(), fontsize=\"25\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We see that by adding a loss for topological regularization, the linear embedding model puts less emphasis on the majority of features that are irrelevant for capturing the topological prior.\n",
    "\n",
    "Finally, we compare if the topologically regularized embedding improves on the ordinary PCA embedding for predicting data point labels."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<style type=\"text/css\">\n",
       "#T_2b9a9_row2_col0 {\n",
       "  background-color: lightgreen;\n",
       "}\n",
       "</style>\n",
       "<table id=\"T_2b9a9_\">\n",
       "  <thead>\n",
       "    <tr>\n",
       "      <th class=\"blank level0\" >&nbsp;</th>\n",
       "      <th class=\"col_heading level0 col0\" >mean</th>\n",
       "      <th class=\"col_heading level0 col1\" >std</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th id=\"T_2b9a9_level0_row0\" class=\"row_heading level0 row0\" >pca</th>\n",
       "      <td id=\"T_2b9a9_row0_col0\" class=\"data row0 col0\" >0.562144</td>\n",
       "      <td id=\"T_2b9a9_row0_col1\" class=\"data row0 col1\" >0.472218</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th id=\"T_2b9a9_level0_row1\" class=\"row_heading level0 row1\" >top. opt.</th>\n",
       "      <td id=\"T_2b9a9_row1_col0\" class=\"data row1 col0\" >0.774822</td>\n",
       "      <td id=\"T_2b9a9_row1_col1\" class=\"data row1 col1\" >0.244247</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th id=\"T_2b9a9_level0_row2\" class=\"row_heading level0 row2\" >top. reg.</th>\n",
       "      <td id=\"T_2b9a9_row2_col0\" class=\"data row2 col0\" >0.849050</td>\n",
       "      <td id=\"T_2b9a9_row2_col1\" class=\"data row2 col1\" >0.137003</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n"
      ],
      "text/plain": [
       "<pandas.io.formats.style.Styler at 0x7fc5383bfeb0>"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Machine learning model to be used for label prediction\n",
    "Ys = {\"pca\": Y_pca, \"top. opt.\": Y_opt, \"top. reg.\": Y_top}\n",
    "model = MultiOutputRegressor(SVR())\n",
    "scoring = \"r2\"\n",
    "\n",
    "# Hyperparameters for quantitative evaluation\n",
    "ntimes = 100\n",
    "test_frac = 0.1\n",
    "params = {\"estimator__C\":[0.01, 0.1, 1, 10, 100]}\n",
    "\n",
    "# Obtain performances over multiple train-test splits\n",
    "performances = evaluate_embeddings(Ys, X[:,:2], model, scoring, params=params, ntimes=ntimes, \n",
    "                                   test_frac=test_frac, random_state=42)\n",
    "\n",
    "# View resulting performances\n",
    "pd.concat([pd.DataFrame({\"mean\":performances.mean(axis=0)}),\n",
    "           pd.DataFrame({\"std\":performances.std(axis=0)})], axis=1)\\\n",
    "            .style.highlight_max(subset=\"mean\", color=\"lightgreen\", axis=0)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Topological regularization for different shape prior\n",
    "Finally, we study how the tologically regularized embedding varies over different forms of (potentially wrong) prior topological information. In particular, we study the topologically regularized embedding when the topological loss function is designed to ensure that:\n",
    "\n",
    "1) The sum of persistence of the two most prominent cycles is high.<br>\n",
    "2) The persistence of the second most prominent cycle is high.<br>\n",
    "3) The topology resembles a connected component with at least three flares away from the center.\n",
    "\n",
    "All other hyperparameters will be kept equal."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define different topological loss\n",
    "def g(p): return p[1] - p[0] # function that returns the persistence d - b of a point (b, d)\n",
    "TopLayer = AlphaLayer(maxdim=1) # alpha complex layer\n",
    "CircularPersistence = DiagramLoss(dim=1, j=1, g=g) # compute persistence of most prominent cycle\n",
    "CircularPersistence2 = DiagramLoss(dim=1, i=2, j=2, g=g) # sum of persistence of two most prominent cycles\n",
    "CircularPersistence2Total = DiagramLoss(dim=1, i=1, j=2, g=g) # sum of persistence of two most prominent cycles\n",
    "ComponentPersistence = DiagramLoss(dim=0, i=2, g=g) # compute total (finite) persistence\n",
    "Component3Persistence = DiagramLoss(dim=0, i=3, j=3, g=g) # compute persistence of third most prominent gap\n",
    "lambda_top = -1e1 # scalar factor that trades off embedding and topological loss\n",
    "tau_flare = 0.25 # sample fraction for which the flare topological loss is computed\n",
    "\n",
    "# Construct the topological loss functions\n",
    "def top_loss_other1(output):\n",
    "    dgminfo = TopLayer(output)\n",
    "    loss = lambda_top * CircularPersistence2Total(dgminfo)\n",
    "    \n",
    "    return loss\n",
    "\n",
    "def top_loss_other2(output):\n",
    "    dgminfo = TopLayer(output)\n",
    "    loss = lambda_top * CircularPersistence2(dgminfo)\n",
    "    \n",
    "    return loss\n",
    "\n",
    "def top_loss_other3(output):    \n",
    "    # Loss for connectedness\n",
    "    dgminfo_connected = TopLayer(output)         \n",
    "    loss_connected = ComponentPersistence(dgminfo_connected)\n",
    "\n",
    "    # Loss for flare\n",
    "    f = torch.norm(output - torch.mean(output, dim=0), dim=1)\n",
    "    f /= torch.max(f)\n",
    "    dgminfo_flare = TopLayer(output[f > tau_flare,:])\n",
    "    loss_flare = -Component3Persistence(dgminfo_flare)    \n",
    "        \n",
    "    # Total loss\n",
    "    loss = -lambda_top * (loss_connected + loss_flare)\n",
    "    \n",
    "    return loss"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We now obtain the topologically regularized embeddings for the different loss functions."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[1mConducting topologically regularized embedding for topological loss 1\u001b[0m\n",
      "[epoch 1] [emb. loss: 0.063185, ortho. loss: 0.001478, top. loss: -3.069539, total loss: -3.004876]\n",
      "[epoch 50] [emb. loss: 0.067590, ortho. loss: 4030.990967, top. loss: -1.291014, total loss: 4029.767578]\n",
      "[epoch 100] [emb. loss: 0.066726, ortho. loss: 678.191345, top. loss: -8.951507, total loss: 669.306580]\n",
      "[epoch 150] [emb. loss: 0.066657, ortho. loss: 784.648376, top. loss: -10.140961, total loss: 774.574097]\n",
      "[epoch 200] [emb. loss: 0.066653, ortho. loss: 1258.971802, top. loss: -7.716102, total loss: 1251.322388]\n",
      "[epoch 250] [emb. loss: 0.066619, ortho. loss: 548.612061, top. loss: -5.368172, total loss: 543.310486]\n",
      "[epoch 300] [emb. loss: 0.066566, ortho. loss: 355.967926, top. loss: -7.132659, total loss: 348.901825]\n",
      "[epoch 350] [emb. loss: 0.067129, ortho. loss: 3932.833740, top. loss: -3.193539, total loss: 3929.707275]\n",
      "[epoch 400] [emb. loss: 0.067130, ortho. loss: 4167.404297, top. loss: -4.837134, total loss: 4162.634277]\n",
      "[epoch 450] [emb. loss: 0.067381, ortho. loss: 5485.577148, top. loss: -2.801000, total loss: 5482.843750]\n",
      "[epoch 500] [emb. loss: 0.066741, ortho. loss: 480.503693, top. loss: -7.617016, total loss: 472.953430]\n",
      "Time for embedding: 00:00:05\n",
      "\n",
      "\n",
      "\u001b[1mConducting topologically regularized embedding for topological loss 2\u001b[0m\n",
      "[epoch 1] [emb. loss: 0.063185, ortho. loss: 0.001478, top. loss: -1.530484, total loss: -1.465821]\n",
      "[epoch 50] [emb. loss: 0.067736, ortho. loss: 1249.523560, top. loss: -0.778874, total loss: 1248.812378]\n",
      "[epoch 100] [emb. loss: 0.066756, ortho. loss: 685.630188, top. loss: -1.517200, total loss: 684.179749]\n",
      "[epoch 150] [emb. loss: 0.066748, ortho. loss: 386.324554, top. loss: -1.305436, total loss: 385.085846]\n",
      "[epoch 200] [emb. loss: 0.066631, ortho. loss: 408.183807, top. loss: -2.597678, total loss: 405.652740]\n",
      "[epoch 250] [emb. loss: 0.066589, ortho. loss: 471.743835, top. loss: -2.147955, total loss: 469.662476]\n",
      "[epoch 300] [emb. loss: 0.067160, ortho. loss: 2124.012451, top. loss: -1.638885, total loss: 2122.440674]\n",
      "[epoch 350] [emb. loss: 0.066761, ortho. loss: 959.783936, top. loss: -2.546629, total loss: 957.304077]\n",
      "[epoch 400] [emb. loss: 0.067828, ortho. loss: 9674.896484, top. loss: -0.521681, total loss: 9674.442383]\n",
      "[epoch 450] [emb. loss: 0.067338, ortho. loss: 2702.017822, top. loss: -2.920576, total loss: 2699.164551]\n",
      "[epoch 500] [emb. loss: 0.066865, ortho. loss: 753.225403, top. loss: -1.828310, total loss: 751.463989]\n",
      "Time for embedding: 00:00:05\n",
      "\n",
      "\n",
      "\u001b[1mConducting topologically regularized embedding for topological loss 3\u001b[0m\n",
      "[epoch 1] [emb. loss: 0.063185, ortho. loss: 0.001478, top. loss: 197.619308, total loss: 197.683975]\n",
      "[epoch 50] [emb. loss: 0.068041, ortho. loss: 5166.160645, top. loss: 55.678978, total loss: 5221.907715]\n",
      "[epoch 100] [emb. loss: 0.067140, ortho. loss: 1323.296387, top. loss: 85.885590, total loss: 1409.249146]\n",
      "[epoch 150] [emb. loss: 0.067050, ortho. loss: 577.588562, top. loss: 73.926880, total loss: 651.582520]\n",
      "[epoch 200] [emb. loss: 0.066967, ortho. loss: 454.377075, top. loss: 79.713654, total loss: 534.157715]\n",
      "[epoch 250] [emb. loss: 0.067389, ortho. loss: 2904.738525, top. loss: 100.876320, total loss: 3005.682129]\n",
      "[epoch 300] [emb. loss: 0.067422, ortho. loss: 1593.473511, top. loss: 71.410591, total loss: 1664.951538]\n",
      "[epoch 350] [emb. loss: 0.067249, ortho. loss: 524.616699, top. loss: 68.545914, total loss: 593.229858]\n",
      "[epoch 400] [emb. loss: 0.067267, ortho. loss: 200.631836, top. loss: 63.663490, total loss: 264.362579]\n",
      "[epoch 450] [emb. loss: 0.067247, ortho. loss: 443.747040, top. loss: 74.482666, total loss: 518.296997]\n",
      "[epoch 500] [emb. loss: 0.067320, ortho. loss: 407.586548, top. loss: 59.356419, total loss: 467.010284]\n",
      "Time for embedding: 00:00:11\n",
      "\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA3IAAADdCAYAAAD+dUBHAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjMuNCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8QVMy6AAAACXBIWXMAAAsTAAALEwEAmpwYAAB5WUlEQVR4nO3dd3hcZ5nw/+9zzpleNBr16irbcYud2OnFqSQBEgglkLAJISGUhPKy/CDAvgsLL2zg3X0XWGDZAIHAhl5CIIEQAmlOc4rjxCXuRVbvZTT1PL8/JMuWNbJlazRFuj/X5cvSc87MuWckPXPupyqtNUIIIYQQQgghCoeR6wCEEEIIIYQQQpwYSeSEEEIIIYQQosBIIieEEEIIIYQQBUYSOSGEEEIIIYQoMJLICSGEEEIIIUSBkUROCCGEEEIIIQqMlesAjqW0tFTPnTs312EIITLoxRdf7NBal+U6jqmQukmImUfqJiFEvpqofsrrRG7u3Lm88MILuQ5DCJFBSql9uY5hqqRuEmLmkbpJCJGvJqqfZGilEEIIIYQQQhQYSeSEEEIIIYQQosBIIifyTjxl0zSQoGkgQTxl5zocUcCUUvcopdqUUq9NcFwppb6plNqplNqklDot2zHONB2ROAf6Y3RHk7kORQghMsLWmpbBGI39UQbjUreJ/JHXc+TE7NM0kOBHr3Xx5739KAVXzQtw49IwVX5HrkMThelHwLeAH09w/EqgYeTfmcB/jfwvTlAkkeK5ln6+8XITbZEE84Mu/nFNLStKfViGynV4QuQdpdQVwDcAE/i+1vquCc5bCzwLXKe1/nUWQxRA51CCP+3t5N6tLQwmbNZWBPjoqloWhDy5Dk0I6ZET+aM/luJrz7fx4J5+UhqSNjywq59/f7GdgXgq1+GJAqS1fgLoOsYp1wA/1sOeBUJKqarsRDezbO2K8Ln1+2iLJADY3Rfjo3/fxd7eaI4jEyL/KKVM4NsMNyYtBd6tlFo6wXlfBR7OboQCQGvNw/u6+M6mJgYTwyOENrT287HHd9A8EMtxdEJIIifySEskyYbWoXHlzzRFaBmUoQxiWtQAB474vnGkTJyAWDLFfdvax5WnNDxxsDcHEQmR984Admqtd2ut48DPGW5YOtpHgN8AbdkMTgxrG0pw75aWceVd0SQHJJETeUASOZE3YseYD3esY0JMQboxfzrtiUrdppR6QSn1Qnv7+KRlNktoPeGcuPahRNbiiCZt9vbG2NgWYW9vXObYinx23EYkpVQN8Fbgu8d6Iqmbpk/K1gwk0o8IGkrKSCGRe5LIibxR6rEodptpyk3KvDKdU0yLRqDuiO9rgaZ0J2qt79Zar9FarykrK+g9gzPO77B407zitMfW1YayEkNPNMVPtnRz458OcPujTdz4p/38z5ZuemPSmy/y0mQakb4OfFprfcyMQeqm6VPssji/pmhcuQJq/e7sByTEUSSRE3mj0ufgS+dU4DIPf755LMUXz6mk3CuLnYhp8QBw48jqlWcBvVrr5lwHVYjOrSlidblvTNk1C8I0hLJzs7OlK8qPNneTGrkVTmn4wWvdbOuS4U8iL02mEWkN8HOl1F7g7cB3lFJvyUp0AgCPw+SDK2uoOOIexFDw6TX11PidOYxMiGHSzSHyyqnlHn50RR1NA8PDsWoCDmplxUpxkpRSPwPWAaVKqUbg84ADQGv9XeAh4CpgJxABbs5NpIWv0ufkS+fM4eBAnJ5YkjKPg2q/k6Bz+j9mUrbm/h3p5+I9sKuPMyq9KCUrZ4q8sgFoUErNAw4C7wKuP/IErfW8Q18rpX4E/FFrfX8WYxTA3KCb7168iMaBGENJm2q/i2qfE7c1fgSRENkmiZzIK4ZS1Aed1AelpUtMndb63cc5roHbsxTOjBd2Owi7c9PwYk4wvsSSBE7kIa11Uil1B8OrUZrAPVrrzUqpD44cP+a8OJFdFT4XFT5XrsMQYhxJ5IQQQhQ001BcuzDEE42RcceuXhiU3jiRl7TWDzE8KuDIsrQJnNb6vdmISQhRWGSOnBBCiIK3pMTFx08rxT0yx9ZjKf5xTRmLi6UV/VgSAzb9rydp/FWU1j/HGTqYwk6mXbhVCDGBRMpmeICHENklPXJCCCEKXsBp8taGIs6q9tIftwk6DSp9Dixj6r1xPYM2vRGNw4Sw38DtnBk9fMkhm7ZH4nQ/e2hRxBTtjyeYe4sL/0K5PRCz20DcpmkgRXfUpthtUO038TvH9n8c7I+zvmmQJxoHWFTs4o3zi5hX5MSQUQAiS6SmFkIIMSNYhqIukLn5tVpr9rTZ/OhvMXoGh1vbV84xufYsJ+FA4Q9oSXTpI5K4ETY0/S7OvA8aOGbAaxTiZHQOpfjBK4M8sPPwqrdXL3Rxy6k+SjzDi5w0DyT4xGMHaRxZnO3ltiHu39nLdy+rY1GxbE0gskNqaSGEECKNjj7Nf/05OprEAWzal+KRV+IkU4U/jCrRl/41xDs0djTLwQiRR15rT45J4gAe2Bljc/vhfSm3dEZHk7hDYinNfVu6iafsrMQJEBuy6WxJ0XogSX+3jW0Xft0kJi8jPXJKqXuANwFtWuvlaY4r4BsML/MdAd6rtX4pE9cW+a9/yOZAh81Lu5MEPAanzTepKjawTBl6IITIXx39NrHE+PLntqe49FRNSaCw6zDLmz5+K6gwZOFgMUulbM0fd6Vvyfjjrijn1joxDcW2rvTnbO2KMpiwcU60lG4GDfTaPPNwjLYDw4mj5YCzr3RRM8/EtAq7fhKTk6nfsh8BVxzj+JVAw8i/24D/ytB1RZ6LxGz+9FKc7z4c4/kdKR7dlODffh9lV0vq+A8WQogcmmjtAltPfKyQOEsUvkXjbwOq3uzEUSQDdsTsZCgITLCLSsCpODTt9pSS9MMnV5S68Tmm/+8nldJsezExmsQBJBPw1B9i9PfMgApKTEpGftO01k8AXcc45Rrgx3rYs0BIKVWViWuL/NbZr3lq69ikTWv4xfo4fZHsDT0QQogTVRo0cKTZ83fNQpMiX+G3dlt+g5q3uai+1omn1sC/xGTubS78i2SjYzF7KaV4yyJP2mPXNHhGtzNZGnYzr2hs17XHUrx7SXFWeuOig5qdrybHlWsNPR3SWD5bZGuxkxrgwBHfN46UNR99olLqNoZ77aivr89KcGL69EXStwp19GmG4hD0ZjkgIYSYpNKA4rbLXfzobzEGR6bLLKw0uGK1E8cMGRruDBmEzzQoWmWhDDAcM+N1icIXSw7RG+3AUAYhTwWWkb31+RqKLf7lvABff2GA7qgm7FZ8fK2fhuLDjRyVfgdfu6CaDS0RnmwcYFHYzaX1gdHkrmfQpntAYygo9iuC3swmd0qBaUJqfC6HMUPqJ3F82fqrSPcblfYOX2t9N3A3wJo1a6RvuMD53Okrk6BX4Zpg6IIQQkzGwf44G9uH2NYV5dQyL8tL3FT6M1exGIZicY3FJ99iHLH9gMLnnnnDDk2X3PiJ/NHav49fb/4mG5sfwzKcrJv3di5b+B7C3oqsXN/jMLhkrpvlZQ4GExqfQ1HhG99TXe13cs1CJ1cvKBrtqQPY357i+389vNptVUhx86VuKkOZqzs8fsWS0x1senrsRF7TgqIS+XueLbKVyDUCdUd8Xws0ZenaIodKA4qldQZbDowdRnntWU5Cvpl3MySEyI4D/XE++rdG2iLDzdG/3dHL/CInX7ughqoMJnMAJQGDkkBGn1IIMYHeaAffeu4TNPfvASBhx3hk131EUxGuX/kpHKYra7GkS97SOTKJ6x6wufuR2JgRSc09mvsej/HBN7gy1hCklGLBcovIgM2uV1NoDb6A4pyrXASL5f5qtshWIvcAcIdS6ufAmUCv1nrcsEox8/g9Bted62JbY4pnXk/icysuXmFRVypzMITIR62RIbqicbyWRaXPjcvMv79VrTV/3dc/msQdsrs3zsa2CFX+ohxFJoSYqraBxtEk7khP7/sDly64gZrg/BxENXldAzrttJJ97TY9gxpfBreY8wYMTl/nYslpNqkkuH0Kr1+SuNkkU9sP/AxYB5QqpRqBzwMOAK31d4GHGN56YCfD2w/cnInrisJQ7Dc4e4nBaQssDIMZM7dEiJkkYdu80NrFlza8RlcsjqkUb19Qx3uWzKXMk1+b2w6lNM80DaY99nTzIFfOl0ROiEIVSab/207pJJFk5LiP748niac0RS4Lyxh7vxFJpNjdE+fRAwPYNlwyx8+CkBNfulWNTpI6xi3OsY6dLMuhKCqZOP5UVBPvsdFJcAQVjqAkejNJRhI5rfW7j3NcA7dn4lqicLlkEr0QeWt//yCfXP8yqZF19VNa84ud+6n1e3lHQ34tPOU0FIuKXWzuHL+P09JwfiWdQogT43eV4DTdxFNj/75rgg1gTPz3PZhI8Ur7ID94rYWuaJKL64p468JSagPDQzGTtubR/QPc9Xz76GN+vaOXfzy9lKsXFo1L+k5WiV8R9iu6Bsb2yi2sMghlebXbeLdN0x/j9G4eXsXSGVbUv9uFry7/RlqIkyNpuRBCCDZ39Y4mcUf66fa9dA7FchDRxCxD8ZaFRTiP6t0POA3OqfblKCohRCb4XaW8dcWnMNXhvga/M8RFi2/DY008WXVDSz+ffGIPW7uGaI0k+NnrHXzyid20ReIAtAwm+PqLHeMe958bO2keSIwrP1lFPoPbLndRETpcP82vMHjXeS68ruzddttJTevfDidxAPEuzZ57osQ6ZfunmSJ7a7kKIYTIW0k7/SLBSVunX2I4x+aHXPzg8jru39nLS61DrCr38LaGEGVei53dcbqjSYrdFtV+E28Gh00JIaZXhbeU0sAi3rzqC5h2FIVBvw3aDFLpLU37mK6hBN/aOH7phf39cfb1xSj3OumJpYimxtdm8ZSmO5YasyLfVFWHTT76Rjc9g8PbD4T8Cl8WkziARK+m68Xx+8mlhiDeZeMqkb6cmUASuTyWtDWtkTixpE2x20GxuzB/XN3RJM0DKeK2TbnXospnjVnhSQiRe8vDRSjG7wvztgW1lLid6R6SM4mUZl9fis3tmrmBAFfNC1MfMInaNv/9Sg+/3t43eu47FgW5cXkR4QKtP4WYjZaHFxFwBtjSs4uknWJlyQLm+qqwjPSNMrGUpmWk5+1ondHh3raA08QyIHlUZ5SphnvzMy3gMQik31c8OzQTbPQ1vGm4mBnkky1PdQ4l+N3OTu7b1kYspZlf5OYzZ9SyNOwtqCToQF+Cf17fxvbu4Qo24DT48nnlrK5wYxTQ6xBippsT8PKFM1fwry9sIZoabsVdV1POFXOr867O2dAc5zOP93GocV0Bnz8vQJHLHpPEAfxqex9rKj2cVysfd0IUCodhsTBYx8Lg5PrJgk6T08r9vNA6MO5Y/cgcuSqfxXtOKeZHm7vHHL9ucYjqDG9Zkg+soKJouUnvq2N75QwHOMPSGzdTyCdbnvrr/h7u2dw6+v3u3igf+/tu7rl8EfXB7O2hYmvN3t4oz7X20RFJcFZVkIUhD8Xu41d6kaTNt17uGk3iAPrjNp96vJV7r6qhNpCdinMgYtPepdnfkiIUUNRWmJRkcFNOIWYCl2VxSW0FS8NBOoZi+CyLar8HvyO/bnBaBlJ85Zl+jhwhpYGvPtvPp8/ypn3MA7v6OLvag5mhxQyEEPnF5zS5Y1UVH350F5EjutyuXVhCrX/4nslpGrx9URENxS5+ua0HDbx9URGrKzy4zJl3T2A6FVVXOIn3xBga2cvX9MKc6124wlIXzhSSyOWh9kicn2xtG1ceSdrs6Y1mNZHb2hnh9se2Ex+5a/rZ9jaumV/Kh1ZWE3Qd+9enM5Jk/cHxSwVHU5qmgURWErnBiM3D6+Ns3nW4RcrjgpuucVMelnkzQhzJMgzq/D7q/Pm7YEh31KYnNn5c0FASknb6mxOvZZDrHK5/0Katy6a53aYkZFBVahCSZcCFyJhFxV5+cHkDr3YM0h5JsLrcz9yge8y9SrHbYl2dnzOrhht9PNbM/ht0lRjMf6+LeLfGToAzpHCEVN6NshAnTxK5PGRrGDp6EPeIicqnQ188yX+8fGA0iTvk97s7uHpByXETOaUUhoI0c4uzNqyyo0ePSeIAhmLw1MsJ3nyhgcM6fhyphCbRbaNTw3uwWL6ZXfELkc88lsKcoF6p8qdvnHlrQ2DMjYutNd3RJIZiUqMLpqpvwOb+v8fYe/Bw/R0KKN7zJjfhIqlPhMiUOUE3c4LjtyjQWtMcGaAjOoTPclDl9ePNs9EG08XyGVjT1DaXGLQZ6tAMNKZwFSl81SZuGbaZVZLI5aGw2+KKuSF+t7Nr3LEFoeztkTQQT7GlK/3mmy2DcU4JH7tmKPeaXDHXz4N7xo5ZL3IZVPuz86vX3D5+xSaAXftTDEU1Dv+xE7l4j03bX+L0vJwCG1yVitp3uvDUSG+eELlQ6Te5psHNb7eP3WPq4jku6gIm/3RWKd98qYu+uE3QafCx08M0FB9erKVlMMZDe7t4YFcHLtPgH06p4JzqIsLTmNA1tdtjkjiAnn7NqzuSXHC6Q1rHhZhG8VSSp1sP8sUX1zOQiKOAK+vn86Glp1Phzd/RB/kuMWiz/69x2o9YGdPywdKbPXjLc5vM2Uk93AuZ1DgCCod/5iaXksjlIYdpcP2ScjZ3RtjePXyzYir49NpaagPZWz3OZRpUep1pV4IKHac3DobHo9+8IsRg0ubxAxE0UBew+MK55VmbWBwKpP/jDQUNHMd5CXZK0/F4gp4jKqlYi2bv96Ms+IhHJgsLkQNuS3Hjci91QZNfbB3C1nDtIg+Xz3MR9phcMc/PqnI3AwmbgMOg4ohVcntjCe56YT/Pt/SPPt9XNuzn1mVV3Li0MmMbAh9tx/5k2vJte1KcucLC7ZJETojpsn+gnzuffQx7ZAlHDTy0fzcLgsW8p2G5NKScpKFOPSaJA0gOwsEn4sy/xoXpyM37Gu+1aX0iQftzyeEG+FLF3HdMzybo9kAC3ZmApI0KOTBKsjf16RBJ5PJUjd/F/7twPgcH4kQSKSq9Tqp8TpxZHM9d4nHwkVU1fO7pPWPKlxR7qQtM7pe1yu/gc2eVceuKJAlbU+oxCXuy92tXUWoQ8Cn6B8eOw7rkTAce97Hfy0SPpvu58TdgqQjEO21J5ITIkVKvyTuWeLlkjgutIewxRm/GlFJUTdBQ1DQYH5PEHfKTbS1cPqeY2sD0jHgoKzaA8aMDwkUKaxLDu/OVPWhDHJRfoXJ00ybE8bzW1T6axB3pF7u2cmXdAko96RdJmu2SgzbxLk2yX2MFFM7w2Kklg03pp/r0bE+RjGjMouzXCdrWdL6QpP2Zw/dusQ7NjnuiLLndg7s0c/dtdnuU+I/2Y+8YGXUWsHB9YB7GQj8qi5OyJZHLY2G3Y1qH+0zGmZVBvrluIfduaaEzmuTKuWEuqSum1HO4Z7AnarOzO8lj+2MEnIp19S7mhSyc5vAvstdhMC+Um32oioMGN77ZzfOvJXh9b4oiv2LdWgc15ZNomdGgJ5iSOFH5dIkP2gy22rRvSeHwKkpPMfGVGRgFfBMoxFSFPSfWwjoYTz/UOpbSRKdx/vGCOgunI0E8cbhMKTjnVAeWWXh/wzqqSe1KEf1VAt1hY640cb3RiVkjjVtCzASJXpvmP8bp23S4zgyuMKm62oljZJEmVzB93eUsUhg5unWN92lan0qMK7djEOuwM5bI6XiKxP3Nh5M4gP4ksf/chfuflqAqsjcNShI5cUxeh8maiiBLS3wkUjZB59jNvCMJm59tiXDflqHRsvu2DPG1dUHOqsl+F3M6JSGDy892cv5pGoelcDknd+NkBRSBU0z6t4y9+VMWOLK4dG8ypjmwPkHjES1M+59KsOJ6F+GF8icsxGSVeR1pF0qp9DonNVz8pK9bbHDzW9w8+VKCfU0pyooNLjrDSUVJYSY+qV0pIl+LjX6fXJ8itWkI7//2YFYU5msSM9fycBkGalyv3DsXnEKJO5c7duevyL7UmCQOoO/VFMEVKUKnDv+NeysNHAFFon/s+1p/qROHN0f1gAadfiQ7dvp2vJO7TE+C1Avd4w/EbHRHHLKYyEmNKybFa5kUucZPym8asMckcTC86ub/fX6AjkgG/2qOkkpp+ntsejpSRIeO35Jumgq/15h0EgdguhSVb3Tiqjj8GMMJ9Te6cGVxWGW02x6TxAGgYceDcWL9We4aPEKsy6bj+QS774vS+mScaHvuYhFiMqq8Lj5x2tgNhi1D8dkz6in1Tu+ogYoSk2vWubjt7R6uu8JNXaVZkMMq7YhN7DfjW7x1P6R2T1+dL8TJqvcH+OpZFxFwDP+NK+Cq+vlcUTtf5seloW1N9wvp/5Z7XkihR1rC3GGDpe91U3aaieUFb5XB4ve4CM7N3WJwjoAivCrN9Q0yOqwS1MQZVJZ/paQ5X0xJ51D6P/bWQZv+uKZ0GoaeRwZstm9Msu3FBKkkhEoVZ73BRUll5isPV6nB3Pe7SXRp7CQ4Qgpnscrq+OdYf5p11oFojyYZ1bgCWQtlVKzLZscPo8Q7h2Pr2Zyi9YkEi97vwZ3j1apmm3gqRftQDA2UeVy4TFlRdSJOy+ANc8IsCXt5pX0Aj2WyosRHfZrlyqeDw6FwFPpcsjjYHenrJD1BuRC55DQtLqiq4ycXv5nO6BDeWbb9wIlShsKa4L7CCjAmgfGUGcx7k4vaizWGAxye3H7+G5aicp2TSEuMoZGVgpUFc9/hwlWSubpXFTswzwyTevqo1eU9Bqo0u6PRJJETUxJ0pv+jLXIpPNPQ2qy1Zu/WJJufO9wi3NOhefRXUa54j4dgceYrEUfAwJGDZGn0+t7076PDC+YJ9DBmUv/u1GgSd0hyEDpfSlJ9uSOrie5s1jw4xI+27ubBvU1o4LK6St6/bAE1fpm8PxGvw+SUsO+426eI9JRPYa0ySDw+vhHPXCiNCCI/KaWo9gWo9uXww7yAhM90DK/YfeTHvILwWeNHZhmWmnC+XC64SgwW3uQi1qWx4xpnyMBVrFAZnI+sHAaON1Wh+5PYr/YNl4UdON8/D6NcEjlRQKoDJhfUOXniwNgtCj602kfFNGycHRnQbH5+/LCeRBz6uuxpSeRyzVOsKFlk0Ll97NDF+Zc5cedoM+H+CYZQ9e9KYSccmPkxPXJGG0wk+Y+N23iiqX207M/7m2mNDHHXOasocuVmgSExsymHwnmlk+QrUXTP4bs862wToyZ/buYKgVLqCuAbgAl8X2t911HHbwA+PfLtAPAhrfUr2Y1SzEbuKoM573XRdH+cRLfGUayousaJuyqDqz4O2thNmuTmFMqvsJaYGFWZSbgcfgOHPwNBHoNR6sJ161zszjgk9fD2AzlY2E8SOTElRS6Dj6/xc3Z1nN/vjOJ3KN691MOy0mna5FZDcnweB0AyOTOH9Ti8Bgvf6KJkUYrml5JYbqg9x0Egh5uS++pMujeOT+Z89QZGntUqk7hZWgf8Hji0z8ZvtdZfzGaMJ6MlMjQmiTvk5Y4eWiJRSeTEtDGrDLyfc2Pvt7Hbbcx5Jka1wgjOvIa06aKUMoFvA5cBjcAGpdQDWustR5y2B7hQa92tlLoSuBs4M/vRitnGcCgCSyzmf9jAjoHhHh6dlCk6qon/NUn8t4dv6GIWeD/lwlqcZzcRx6A8FmZtbuMtnHdL5K1yn8mbGzxcPMeFZYBrGve6c3sVc5eY7N48PokoCs/cYT3uoEHV6QZlyy2UQc422jwk2GBiBRTJI+bvGU4oXWtldPjCVE3yZgngSa31m7Ie4BTEUhMvLhNLyaITYnqZ5QbmNM2HTUZsUjEw3WDleM7NNDoD2Km13g2glPo5cA0wWjdprZ8+4vxngdqsRihmPcc0Nc7Y7Zr4745qlU9C9J443s8Z0ih0AiSRExnjm2C+XCaZlmL5mU66WqP0jEysNww48w1OAsX5k0BMF8uVH6/RXWqw6FY3XZuS9O9I4a0zKF1j5eNCJ8e9WSpUZR4XFR43rUPRMeVFTgdlnuwtfSxEpthJTaTR5sCDcYaabbw1BrVXOvHWzsg9M2uAA0d838ixe9tuAf6U7oBS6jbgNoD6+vpMxSfEtLG7bdLs0Y7dotEDGoLZj6lQSSKXR9qHouzo6efFti7qAz5WlxVTH5AJ+UcLFBtc/HY3Az2aRELjCyr8QQMzgx/09oANQ4AbjAwOJ5hJ3GUGVRc7qDzfgbLI1wVOJnuzdLZS6hWgCfik1nrz0Sfk281SmcfNl85ayceffJFIcrgHzmUafPnslVT5ZG8kUXiGWm22fz8KI53NkQPD3y/5sBtv9YwbcZGuwkw7P0ApdRHDidx56Y5rre9meNgla9asmZlzDMSMoiZYxA0vqBwt4laoJJHLEx1DMf71hS083dIxWhZ0OvjuurXML5rmGZsFyOMz8ExDjquTmtQum+hP49j7bIxahfsGJ+ZCE1Xoy4ZPA6UUKr+nYk3mZuklYI7WekApdRVwP9Aw7kF5eLM0P+jk2+tWcmAgAhrqA37mB6QpUxQerTUdzydGk7hRNnS+mJyJiVwjcOSmhrUMNySNoZRaCXwfuFJr3Zml2ISYVka5wlxskHp97B+86+0OVAa3CZgNJJHLE7v6+sckcQB98QQ/eX0Pnzl9KU7ZGyor7IM2kbsOtwjbBzSRr8bw/bMbc778DArQcW+WtNZ9R3z9kFLqO0qpUq312D/IPBNLxvlD4+P82+Yfj5aZyuDrZ3yKc8pPzei14qkUg8k4PsspdZGYHjbEu9K3j8Q6NdrW+drrf7I2AA1KqXnAQeBdwPVHnqCUqgd+C/yD1np79kMUYnoYQQPPbS4SzySJP5FEeRSuqx2YSwzZpP0ESSKXJ7Z09qUt39DaRV88SalHbp6yIf5kcnyLsIbY3xJ45hoz7UZiNpjMzVIl0Kq11kqpMxje7jTvW74PDrXx9S33jSlLaZsvvXI3PzrvS1R4wlO+htaaXX1d3LdzE5s6W1hWXMF7GlaysKgEQz5sRQYpU1G8yqJ/V3zcsfAqc8bVvVrrpFLqDuBhhlfUvUdrvVkp9cGR498F/hkoAb4zcnOb1FqvyVXMQmSSUWrgfJMDx4UWmGBMw5ZVs4EkcnmiPpB+A9/6gBePJUlctuj29C3CukNDiuFbfFEwJnmz9HbgQ0qpJMMzI9+ltc6LoZPH0hHtIanHr07ZFu2iJ96fkURuT383tz7xeyIje34cGOzj7817+OGFb2FhUcmUn1+IIwUWmPjmGQzuOdya5l9g4J83Mz8DtdYPAQ8dVfbdI76+Fbg123EJkS1KKVQebSZeiCSRyxOnhIOUe1y0DcVGyxRw27KF+BzyY8oW62yTZJr90RznWiTjmuh+m8E9KRwhhW+OgbtsZt5gzCSTuFn6FvCtbMc1VQFH+kmibtOF18rMjuyPHNw1msQdEksl+eO+1/nYirNlCIzIKFfIYN51LmIdmli3jSts4C5VYMFQt41hgUsWnxJCiFGSIeSJap+Xb124ht/vbuSJpnbq/F7ee8p8FhcHch3arGItMjFXGqQ2HW4RNpcYGIsMmh6M0/3S4STP9MCC29x4qiSZE9lX4y3nworTebz1xTHltzS8hSpP6ZSfX2vNlu7xG44DbOvtIGHbMl9OZJwzaOAMQgATrTX9TTY7/xSn/6CNK6iYf7mT8EIzb7ZiEUKIE2G3xEltHsQ+EMVc5sOY78EocZz080kil0fqAz4+tKKB9yyZh9s08Fjy48k2I2zgeb8Lu1ljd9oYYQOjShHp1mOSOIDUELQ8kqD+OgNTbipEFkWSUZ7reJW1pcuZF6jhb83P4zKd3LTgas4qW4FlTL3uUEpxYdVcnmk9MO7YhVVzJYk7SiKu0RqcUhdkTKRd88qPotjJ4e9jfZqtv46x4gYX4YXy+SiEKCx2U4zoV/fD4HBnQeqpPtRcF64P12CETy6Zk5owz1iGQbErv9dzn+mMoIERhOEpVcOiWxNpzx3YmSI1pCWRE+McGOhhT18XA8k4C4Ml1PqK8Doy87f9eu9e7nzxGwDU+So5t/xU4naSuf5qil2Z237gzPJa5vlD7BnoGS2r8wU5v3JOxq5R6IYiNm0HUmx9MQkaFp9mUVlv4pGJ+1PWsy81msQdad8TCYJ10isnhCgcOqlJ/K1nNIkbLd8bw94bzW0ip5S6AvgGw3e+39da33XU8XXA74E9I0W/1Vp/MRPXFiIbHEXpbxicYYU6+R5xMUNt72nnQ0/dT3/i8JzXT668gDfPOQWPNbVfmJS2+cWev4x+f2CwhZ/vaRn+RsPnQplbG6HGF+Tr51zFa91tbO5u45RQKSvCFVT7ZK86gFRSs/3lJK89e7ih5+mH4ixebbHqfCeW7D05JfH+9GsOxQc0dlKDJHJCiAKhh1LYWwfTHkvtiGCddnJTqaacyCmlTODbwGUM79m0QSn1gNZ6y1GnPqm1ftNUrydELngqDRzFikT32BuLqiucOKTlXRyhLx7la688MSaJA/j3TU9wWmnNlFd7tG2bnkT67Uq64unLp6LKF6DKF+Cy2gUZf+5CN9Bns+X58b312zcmWbjSIlQqw0+noni+yf4nx7+/FSstHF5J4oQQhUO5DYw6F6nW8XWaWec+6efNxB3oGcBOrfVurXUc+DlwTQaeV+TIYCLFrp4YG9si7O+Lk0zl/Urs085ZbLDgFjel51k4QgrvHIP5t7jwzdBlscXJ64lF2dTVPK5cA82RqSdaDtPi2vqL0x67uu6CKT+/mLxEDOyj950EtIZ4NPvxzDTeMkX1GWPbm30ViopTLVkxVWRcZ6yb13t3sndgP7FU7PgPEOIEKIeBdWXJkbN2hstDFsZCz0k/byaGVtYAR86GbwTOTHPe2UqpV4Am4JNa683pnkwpdRtwG0B9fX0GwhMnoiOS4PuvdvLH3X1owGkqPn5aGZfNCeB1zO6kxVVqUH2lk/ILNcqhsNxyIyHGc5gGfoeTgcT4jY19GZojt7rkFK6sOZc/HVw/WvbW+otZHmrIyPOLyXF5FJYTkkf9qE0L3L7c1g/xfpt4rwYFriKFw194IwecPoO56xxUrrSI9mocXoUnrHAFC++1iPyltea1nq3808a7ODB4EEuZXFN3Bbc2/AMVnrJchzeraa2JtGvatyQZbLUpPcWkaI6Ju6gw6wCj1oXrs3NI/rUbuzGGudKHdXYRRvnJ3xtkIpFL92l1dBfOS8AcrfWAUuoq4H4g7R2H1vpu4G6ANWvWSFdQlq1vGuQPuw/3GsRTmq9taGNhyMWy0pNvMZgplKlwBCSBExOr8AS4ZfFavvHa+jHl9b4i6nxFU3rutqEODkZaSegEtzRcw40L3kxHrJtiVxFlzhK6YnEODjZT6vZR5Q1Ir8U08xcp1l7s5Jk/j83kTrvQiS+Hm9wOtqbY8fMY0c7hj1BPmaLhOhfe8sJrjHN4DBw1EKjJdSQChm+sWyJRhpJJgk4HpZ6THxKWLw5EDvLh5z7NYDICQFKn+M3+Bwk5Q3xw8U2YqvD+brJNxyKAQrkye5840Gqz8Z4o9shoxI5tKQLVBsuuc02qQUenhodMKDM/Ej9lKsw5boybKiCuwW2gjKl9VmQikWsE6o74vpbhXrdRWuu+I75+SCn1HaVUqda6IwPXFxnSG0vx6+09aY892zQoiZwQk2AoxZV1i/FYDn6wbQN98SiX1y7ipsWnU+bxn/Tz7u7fz8ee/wJNQ60AeEw3/7bmnzizbCVtQ4N8eeMTrG/ZB0DQ4eJLay9jbVktliE3IdNFKUVdg0UwbHBgRwqtNXUNFqESA9PMTSIX77fHJHEAQ+2aHb+MsfRmt8zpFSetLx7n0QNtfOfV7fQnklR53dx5+lJOKwvjyJMb5ZOxf6BxNIk70k/3/Ia31b+RSm9FDqIqDHZfF3rny6TW/x5ME/P8t2HMW4HyT63REiCV0Ox/IjGaxB3S32Qz0GofM5GzB6Lo/Z0kn9wGgHX+EtScUgyfa8pxZYKyjIztG5CJp9kANCil5gEHgXcB1x95glKqEmjVWmul1BkMz83rzMC1p008lWL/QISdPf2YSrEwFKA+4MOcwS3cBhpzgpYBa4otBkLMJmG3l2vnLeeCynkktU3Y5Z3Svmv9iQH+9dVvjyZxAEOpKJ984f9w3/nf5Le794wmcQB9iRj/+MxD3LvuWnSqmaQdp8RTRZmvGkNalzPK4VSUVpmUVuXH+xrv1WOSuEOG2jSxXo3Dl4OgxIywqaOHr750eB275kiUTzz1Mj+85EwWFRfuSraJdHtcjJTb4waYiUN0bIjUo/dhP/vH0bLkvi2Yl9+EeeE7UVPcCzkZ1fQdSDMJGRhosimZYCaBTqZIPb2d5B83jpbFNx/EevNq1EVLUVZ+1NWZMuVETmudVErdATzM8BS+e7TWm5VSHxw5/l3g7cCHlFJJYAh4l9Y6b/86UlqzvrmDzz3zCod+hRyG4j/OP4015VNbcS6fBVwW715SzBefaRlTroCzquXTX4gTVerJzN9NR7SLl7peG1c+lIrSGGnhd3vHTzlOaptXO/fw4OaPkbQTuC0vd6z9GsvKz5RkLk/Ee22SfRrlGF5QSfajFPkqkkzyP9v2jisfvl9qL+hEbo6/FkuZJHVqTPkbay+l1BXOUVT5T3e3Yj/34Ljy1N9+irHyAlRZ7ZSe33Qp/JWKrp3j0wVfxcS9cbpzgOSfN40rT/7pFcyV9aiKqfcW5pOM9IVrrR/SWi/SWi/QWn95pOy7I0kcWutvaa2Xaa1P1VqfpbV+OhPXnS5NA0N88fnXOLIdIGFr/uX512iLzOylyNZWerlxaRhr5Dcj4DT44rlVzA3KJuVC5IqhTEyVvrp2KAcq7VRlsPXwcD+AaDLCN5//JG2DB6ctTjE52tYM7E6y49tRdnw7yvZvRGn8bYx4T/rW5xPhDCncpeN/HzwVCtcE+2EKcTxJWzOYTKU91hcfv5x6Ianz1vC10z+P2zw832950RJuWXgDTlPufSYU6R9eovdoyQTExg9VPVGWUzFnnZOjP/o8pQp/5TESucEYJNPUpUkbHZl5q5FmaITmzNIRjTKUGl9htQ/F6IzGKPcW/uTeiYTdFjcvD3PFvCCRRIqQy6TS55BFE4TIoQpPKVfVXMwfGv86przSXUadr5K3z1/OT3a8POaY0zDxmxFS+vCwoXgqSvvgQSr9siJwLsU6bXb/MIY+dP+roeeVFI5QkqrLHagpzK9z+g0WXedixy9jDLWPLHZSoWh4h8yPEycv6HTwlvk1/N+Xt407dkFNeQ4iyhyH6eC88rP4xfl30xJtw2O6qfZWEXaFch1afvOHwHRA6qhE3uNHeTPTQxuoMlh9q5umDQki7ZrSpSalSyzcoYnrMuVzgWWMT+YsY/jYDCOJXBreCcb1mkrhmWFja9NxmgZzpAcuY5IxTazLJtppY3kV7rCB6xiVkBBHc5suPrDoBmxt8+emx0hpmxWhxfzvlR+jylvOO+Z7aIn088jBnQCUuLx8dOkyHtv5tXHPZRknXu0nbZuDg/20RAZwGiY1/gDlGRo2OhvF2vXhJO4InU8nKD3bwhmaWsOZt8Jk6c1uYr3DiZwrpCSJE1N2XnUZTzd3sL5leJ06Bbz3lHksCJ78Ik75wjJM6vw11PlledTJUuFKzDfeSuqB/xpTbl1zOxRnZoEYZSgCVSaL3mRgp8B0HL9uVCV+rDetJnn/i2PjetNqVLjwf1ePJolcGlVeD2dVlPBs69j1WK6eV0OVLz9XbhxMDuFUFg7TketQxBGSQ5qW5xI0/u3wXZurWLHkH9x4SuXGSkxelbecz628g/c1XEdKpyhzlRB0Dn8oVXoDfHb1Rdy8eA3RVIISl4cn9txL88DuMc9R5CqlzHdi8xaSts3TLY189vnHidupkVj8fP2cS5kXDGXktc06E8wQ13riYyfK4TdwzLx7FpFDFV4P/3zGcg4ODtEXT1DqdlHt8+B1nPitZONALzv7OuiORVlUVEq9P0TAOfN6S2YyZTkwT7sMo6aB1KtPgeXAXH4uqqI+46O4lKGY7MKoyjKxzlqIURMm+cwOAKyzGzDqwjNuoROQRC6toMvBp09fym93HeD+3Y1YhsG7Guq5ck41rimsPDcdmgbbeaz1Bf7U+BTV3jJumH8Vi4NzcVnSo5YPop32mCQOINataV4fZ+4bXRiWDFkVk+cyXcz1p0/EfA4nC4sOL8Z06fx30Bfr5OnGh7B1ijlFp3DraZ+n1Ft1Qtc8ONjPZ55/jIR9eJhKc2SAf3vlOb521kUZ2+R8NnGVKpQJR62tQHithZXD/eeEOJ4il5Mi19T+5nf2dvKhp+6nN354zYFblqzhhoWr8DskmSskyuNDzV2GMXdZrkMZQ3ldmIurMBef2OddIZJEbgJVPg8fWL6Qty+sx1BQ6nbl3Tyx9mgXn3rhP9jatweALb27ebT5ef7rrM+ytmx5jqMTAIPN6Rcv6HwtRc06LYsPiGlT4q3iplWf5U2Lbialk4TcpfidoRN+nqZI/5gk7pAN7c10RIckkTsJrlKDuTe62P/zGKmh4TJ/g0HZ+Q6MHO0/J0Q2RJJxvrX56TFJHMAPtr3AhVXzWBIq7Pl2Yup0wkYPpVBuA+XMr86TfCSJ3DFYhkFFHi9ssqe/aTSJO0Sj+Y8t9/Gdsz9LyBnIUWTiEMs7QblPIfs0i+nmNF1UBeZO6TkcE/yiOg1zRu+rOZ2UqQgsMmn4iIdEn8ZwgLNYYXlluLWYeWytMUbqip5YlGda96c9b39/ryRys5zdOETi4XbsnRFUnRvHVeUY9R6U7GU8IUnkCljzUEfa8t0DjQwlo5LI5QFvpYnpYbTV/ZC6ix04/HLTJvJfrS9AmdtLe3TsctLXzltEhSx4ctKUUrjCCtmmSsxUjQO9rG/Zz/rW/awIV3Bp7QL8loNSt4/26OC484tkjtysZjdFid61G2LDI0B0R4LYqwO4PrsAsy4/16fIB3InWcDqfZVpy1eEFuJ3yA1WPvCUGCx9r5vg/OE/NYdfMf8aJ0ULpTtOFIZKr59vnnsZq0qGW8odhsE75i/hhoblOPJszrAQ4vj641G2drfxRPNuXulspis69T2/jnZwsI871j/Iv7/6NM+2NfK9bS/yoSf/QHusnXcuWDLu/AqPnzmB4ozHIQpH8uW+0STucKEm+VgX2s7QKlAzkPTIFbA5/iourDidx1sPL7HqNBx8bOkNBBwTjOkTWeerMln0bjfJiMYwwRmU9hORO12RVhJ2DJ8jiH+S+yQtKCrm38++hI7o0PCQc48PpyRxQhSc7liE/976HL/ds3m07LSSaj6/5jKqvJkbxfNiexNNkf4xZWdVVPH1zf9DlbeCjyw/nd/t2UlXLMJZ5TXcuGg1ld4Ag9FuUnYcryuEZUoP3Wyi9w+lLz8YhYQGlwyvTKfgE7ne+BBd0SGcpkmFJ4BlFM5Ncm98iMFEAp/DQZHzxLuNw64iPrPyfVzds46/NT1Pra+Ci6vWMi9wYsuLi+lnuRSWVEIihwbjfWxsfozfbP4mfbEu6otO4YZVdzKveBmGOn69GXC6ZHlwIQrc6z0dY5I4gGXFNbza0ccvug5S6/eypjzM3JPYG05rzYHBZg5GmjHNIT556ip+unMXTYPDCd2CYJCHW3bwYtcWytwbuaLmQgKOWjZ2vUZ/rIIt+1/j8Ve/zWC0i8U16zhn6S2UBOdm4mWLAmCeGiT1Ut/48tVBlKtw7u2zraATuW3dbfzry39na287btPiPQ2ruXbeckrd+TOscLhi66NxsBetocYXpNLjZ0tPK1/ftJ7tvR0sKirl4yvPZUVx5QkPVSpzh1lXGWZd5ZppegVCiJlge8eL/PClL4x+v793K//25G3874vuozo4P3eBCSGy5vHmsXtLvmv+aezsNrhv2+HkzmuZfPeitSwKBU/ouTd2b+Gjz32eyMikcI/p5lPL/pG7N++mLTpIcyTCHH8VO/r20x7t5ie77weg0lOCM9LE/Ru+MPpcr+57kL1tL/APl/yAkK+aWGKAZDKGxxXCkJXCZiRjsQ9V70bvP7yiqSp3Yq4+sd/DVLQPnUpgeotRk2ikLHQFm8gdGOjhw+vvZyARByCaSvL9bRvwWk5uWLgqb7YK2NTVyseefohIcngvMZdh8o1zruT29b8npYfH/G7rbef2p37PDy98O0uKZcUmIURmDcb7+OO2748rT9gxdnS+LImcKEhaa3T3ICRS4HNh+PN3lel8UXZEQ7epFHP9Ffxy+64x50SSKe5+bSdfOnMlnklu9t061M6dL941msQBDKWifOv17/Cu+R/iu1te48nmg9y5+gY+9vxdaA7Pebp57uWs3/zf456zf6iVjt7ddPXt4enX7mYw2sGSujewcsG1FMvIoxnHKHHiun0O9r4h7D1DGPVujLlejNLJbXGTHOohcuBFOp79PnZsgMrzP4FbF2NvexUjGMI4ZRWqoho1w6YFFGwit7e/ezSJO9K921/k8toGyj0nPiwg09qHBvmnDY+OJnEAi0Ol/H7fltEk7pCU1ty/dwt3SiInhMiwpJ1gINGT9lh/rCu7wQiRAfZAlNRLe0k+tBEicVRNGMd1Z2LMKc2bhtx8dH7VPL6/bQNJbRNyemiJjL+PAni5vYe+RHLSiVxbtJOONHVJZ6ybMo/JTYtWcWVdAzVeHz849/P8eOcfaB7q5IqaczivbAU/eqk97fP2R1p5fON/kEgOL8jy/LYfsbPpcd657jsEvBWTfNWiUBhhJ0bYCauLTuhxWmv6tz9Ky1+/AkBo8ZuxXt5K8qUXALABHnkAx23/H+b8RRmOOrcKts8xlkqmLR9MxEnp9JswZ1tHdJDWoYExZUGni65Y+gmdByN9eRO7mL30UBz7YBepbU3YB7vQ0cTxHyTyWsBVzNl1b0x7bHHZ2ixHI8TU2duaSf76eRhJRPTBLuLfegTd2pvjyPLbvECYb593DQsCYfoTMco96Xs76vwePNbkey5c5sS9Jg3BMm5fdibzg2FclotTw4v5yukf4+5z/jc3LnwzYW8FC6rOTftYh+keTeIO6erbQ1ffvknHJma+RH8L7U99e/T7ouqzYSSJG5VKkfzd/2AP9DOTFGyP3NxAMQo4ekHSy+saKHHlx4qNTtMaF+OW7nauW7CU59oOjDv/qrrFmLNgPK+YmI7b6NYY9v4oOBRGnRtV4craZph23xDJP79C6qnto2XmulOwLluOEZB9XAqVoQzOnXMNr7asZ1/v1tHyqxa9j+qADKsUhcUeiJJ8eNP4A/Ek9r4OjMpQ1mMqFJZhsLq0hu+c/1YGEjFStkmtr4nGwbENzHesXETQ6Zj081Z5yjm77DSeaX9pTPmZpauo8Y3vOXOZDlzm8PM7HR7Wrbydg52vEIn1jJ5z1pL3sqvpibTXG4x1Tjo2MfPpZIxU9HAjjhFN39mjWxohGgH/zNlnuWATuTpfiP992iV8+eW/j/ZizQ+EuWXxWpxmfrysKq+fi6vn8WjTntGyrtgQxU4va0preKHj4Gj52rJaTiutzkWYIk/opE3q5T7iPzhwOPt3KFwfn4e5KDsL+Nh72sckcQCpx7ZiLK7CWCZzEgpZma+Gj5zzDdoHDjCY6CPsqaTMV4PHkfth6EKckJSNHko/JFBPMFQwHymlrgC+AZjA97XWdx11XI0cvwqIAO/VWr807olOQrHLQ7FruHHu6xeczoN7D/L3xjYqfW7eu2Q+S4pP7EY34PDzmRW38/0dP+Ohxr8DcGXNRby/4d0EJlHHlBUt5KZL7qWtdweD0S4qi5cQ9Fbwx2c+mz5+f90JxSdmNtNdhKtkIbHOnQBo53BHyjhFxTDDVl/Oj4znJLgsi8tqG1hWXEHLUD8e00Gtr4hST/6sWOm1nHx0xdmUuL3cv3cbKa25qq6BteW1nF81l/0DPTRH+qnyBqj3hyhx50dPosgN3REn/qPGsV24CU38hwdwfXoBRmjyraMndf2UTerp7WmPpZ7egbm0ZnTuid0eR7cl0Akbo9yJKnOgHNKbnO9C7lJC7tJchyHElCi/G3PtfFKPbh53zJxfGPPMlVIm8G3gMqAR2KCUekBrveWI064EGkb+nQn818j/GVXr9/L+ZQu5rmEOLtPAY53crWGNt5I7l9/OLQvfBUCZu+SYQy6PVhyoozgwNkG7aNUn+Nnf3j9meOWqhe8k5K8/qRjFzGR5i6m87DPs//WH0ckYg/078dfPg/17xp539bsxgqHcBDlNCjaRA3CZFvOCYeYFw7kOZUJV3gAfW3421y9ciQbK3N7RHsMSt5fVuQ1P5BHdk4Tk0YOFQXck0P1JmOZEDgW4019DeRyjSVzqQJTYvzfCQGr0cc5bKjHXBFBW/iVzuWz1FkJknjINrHMXYW9vRh8YWWBDgfWm1ajyE1uqPIfOAHZqrXcDKKV+DlwDHJnIXQP8WGutgWeVUiGlVJXWujnTwRhKEXJNPumaiMt0UuurykBEw8qLl/APl/+Eg+0b6Y+0UV+xhpLgAjyugvk5iyzxVK1g7g0/Yejgy6SG+jDfdj3s3IX94nqUP4h50VUYdXNzHWbGFXQiVygcpkm1TyodcRwTbXhpKpRz+hMkZRhY5y0m/vL4SeTmOcOrPNmDKRI/aT2cxAFoiP+wBfccN6oqv4Ys5FOrtxAic4zSAM4PXILu6EMPJTCKfaiwHzVBY1QeqgGOnCzfyPh6J905NcCYRE4pdRtwG0B9/czqqVJKURKcR0lwXq5DEXlOGSbu0gW4SxccLqxegF57HpgmaoYNqTwk/5rPhZilVJkTY9H44bXmJSWokuzcnBh1YRw3nAPekZZZnwvHjedhVBcPf9+XxN4dHf/AFNgdebm65Wirt9Y6Dhxq9T7SaKu31vpZIKSUylyTshBiWhhBD+b8CqxltRjVxYWUxAFpp/AcPSRjMuegtb5ba71Ga72mrKwsI8EJMVMoj3fGJnEgPXJC5A3Db+G8uY7k3ztJPtUFDgPHG0ox14ayNmRRuZ2YZyzAWFgJ0Th4nBjhIyaqWwqcCuLjh4Bmo9fwJEirtxAiHzUCR04IqwWaTuIcIcQsJomcEHnEKHXiuLYS65ISMNS0L3CSjlIKVZJ+lTFV7MC6tJjkQ2M3flXlDlR5XraGZ7TVG7gbYM2aNeMzWSGEmLwNQINSah5wEHgXcP1R5zwA3DEyf+5MoHc65scJIQqXJHJC5BllKlT4xCad63gK3R7Hbo+hnAaq3IVRmvmhBMpSWBeHwGmQ/HMXxGyM1X6c15ZiFOdlIiet3idoMBlhMBHB7/DitWQl3ZlGx1LYLTHsbYNgKszFPlSVKy8XKprJtNZJpdQdwMMML8R0j9Z6s1LqgyPHvws8xPAiTDsZXojp5lzFK4TIT5LICVHgdNwm9UIP8Xv3He5HCli4P9GAUZP5TbyNkAPHVWGss4Nga1TQRLnMjF8nQ6TVe5ISdpJtvTv49uv38HrvTpaHlvChxe9lSVEDhpKb/JlAJzWpF/qGtzkZkVDg/PAcrFUzZ0EuHbexm+PYWyNggLnEi6py5t0WKVrrhxhO1o4s++4RX2vg9mzHJYQoHJLICVHgdEdsbBIH0J8k/osDuD40H+XJ/J+5MlTWFmCZCmn1nrxd/Xu45emPk9RJANa3P8+Gzpf5yXnfoSE4P8fRiUzQnXHi9x08qhDi9zZi1C3EKJn68vO5plOa1Iv9xH/QMlqWUOD8UDXmav/oNipCCDETSCInRIHTHfE0M7rA3jaA7ktOSyJXSKTV+/hsbfObfQ+OJnGHxO0Efzr4qCRyM4TuTUAiTWUxkEL3JWEmJHKdCeI/aT2qEOL3tuCun4MqLfzXKIQQh8zuOzwhZoKJVot0GWBK67M4voSdZO/g/rTHdvXvzW4wYvp4JhgCbSqUJ3fDDrWt0R1D6M7YcCylboyw++SeqzeZdlVdBm10XwpKpxisEELkEUnksqRtaJADA/0MJuNUevzU+gJ4Hfk/NE3kP1XuQoWd6K74mHLHFRUnvGiKmJ1cppPLq9bxYucr445dVr0u+wGJaWGUOjHPKCL1fO+YcuuSElSOeuO01tg7eoj/12aI2cOFQQeuO1Zg1KVfPfeYPObwOrRH53IGOU1WhRBiOmSkVlNKXaGUel0ptVMpdWea40op9c2R45uUUqdl4rqF4sBAHx958i986Mk/88ln/sZ7/vYAv9y1lYFE/PgPFuI4jLAT18cWYqwqGr6B8Zg43lqNeV4JypAeOTE555avZUlw4ZiylcXLWFNyao4iEpmmPCaOt1fiuK4KVe5EVbpw3liDdXlpzhYC0Z1R4v+95XASB9CXIH7vNuyBE/+MNEodmGePX7jFvDhUEPN6hRDiREy5R04pZQLfBi5jeBnvDUqpB7TWW4447UqgYeTfmcB/MX5T3hkpkUrxs51b2DMwtgX0O1te4ozyapaGZZyHmDqjyo3rlrnD81wMUMVOSeLECan2VvEfa/8PO/v3sGdgPwsCc1ngn0u5R+qomcQodmJcWop5RggUGIHcDszRXTEYSo0vPxiBnjj4T6ynULkNHG8txZjjJvm3bjAUjsuKMVb6UBMNQxdC5KXUYDeJrr0kmrdhFlXirFiEFarJdVh5JRM1+BnATq31boCRJbyvAY5M5K4BfjyyqMCzSqmQUqpqNizx3R2P8kjjnrTHtvR0SCInMka5TFRZ3m4DIApAhaeMCk8Z55afketQxDQzgnkys8KaoMFJcdJzfI1iB8YlxZhnBIa/z3GyKoQ4canBbnr+9nUir/5xtMz0l1F2w3dxlMzNXWB5JhPNUzXAgSO+bxwpO9FzAFBK3aaUekEp9UJ7e/uUAhtKxuiM9hJPJab0PFNhKYOgI32LYpEj8xs2CyGEEIVClbhRleP3uzROK0Wd5IIno88RsCSJE6JAJTp2jUniAFID7fQ//1N0Du/r800mErl0TWZHTzOezDnDhVrfrbVeo7VeU1ZWdlIBJewkr3Xv5DMvfoObnvon7nr1Hnb1Nx7/gceRtBMc7N3J+n1/4Km9v+dA73YSqdgxHxN2e7j1lFXjyn2Wg8Wh8JRjEkIIIQqVUeTC+YFlGKeERgrAOKscx1vno1wywkCI2Sp+cFPa8qHtj5OK9GQ3mDyWiaaqRqDuiO9rgaaTOCdjtvft433rP09KD0+e/v2Bx/h7ywZ+eO4XmRuoPunn3dr+PN965n+RGtlrSWHwwTO+yurqizDUxDnx2RU1fHrVWXx3y8v0xmOsDJfxyVPPoj5QdNKxCCGEEDOBUenF+f6l6N74yBxfF8ohSZzIT7ZOYSj5/ZxuZrAqbblVVInhmFpv/UySiURuA9CglJoHHATeBVx/1DkPAHeMzJ87E+idrvlxsVSCH+/8w2gSd0hfYpANHZtPOpHrjDRzzwv/PJrEAWhsfvjSF6grWky5v3bCx4Zcbt42fwnnVdYSt22KnC6CThlWKYQQQgAoj4XyyDBIkb+6h9rY3vkyT+3/A2FPBevmXkttcCEOU+7npoOzehmGN4R9VO9b0QUfwnAHchNUHppyram1Tiql7gAeBkzgHq31ZqXUB0eOfxd4CLgK2AlEgJunet2JxFIx9gwcTHtsZ//+Mec1RlpojrThNl3U+aqpOMbqbH2xLvrj3ePKo8lBemPtx0zkDqnwnsSeOEIIIYQQImf6Y938eNNXean576Nlj+/7PZ8851ssLz8rh5HNXI5wPWU33E3f+h8Q3f00VrCaoovuwFm7Mteh5ZWMNH9prR9iOFk7suy7R3ytgdszca3j8Tm8nF9+Wto5cWeWrQAglozxcNMTfGnTN7EZ7rmr8pTzrTO/xNwJEjKX6UFhoLHHHXNbvgy+gtzoHmqjPdKErVOUeCop9VajlCxfL4QQQojZrXXwwJgkDoZHZf1k01f57Hnfp8hdkqPIZjZn2QLCb/w89lAPyuHG9MiUpKPNuE1VTGVwdf06yt1jFxJZHV7CstACAA5Emvnipm+MJnEAzUNtfHvbvQwlo2mft8Rbxblzrh5Xflr1JZR6T37eXT442LeLLz95C19+8n3861Pv5/OP3cD2zo0M599CCCGEELNXT7QjbXnLwD6iyUiWo5ldDIcLK1ghSdwEZuSA9Dn+Kr53zj/zavdOdvcfYGV4EYuDcyn3DCd3Bwab0GkWzXys5Vm64j3UWJXjjrksD9ec8kEq/HP466770Nrmonnv5Jw5V+Nx5MeQyaSdZCA5iMf04DInt4nqQLyX7730Bdojh4ejDib6+MZz/4t/WfdTynyFnaQKIYSYmE6mUFZ2F26w+xLopiip1/sxwk6MhX6MKlm8QOSvkDv91JtK/5wZMSpLFK4ZmcgB1PoqqPVVpD3mNtN/YPgcXsxjrERU7CnnioYbObvuKjSaInfpMVerzKbd/fv41b4HeL7jJRYFFnDjgutoCM7HMo79Ad0dbWdPz+Zx5YOJPjqHmiWRE0KIGchu68Le9Dr263tR82owT1uKUTnxPPGMXbc/QeK3B0k93XW40GPg+uQizDrvtF9fiJNR7qtjdeWFvNzy+GiZQvGeFZ+iyC1bSWWajiTQHVH0QAIVdA7vNymLIaU1K9+Vel81xc4iuuO9Y8pvXvBOKiZodTlEKUXIc3L7202XvQMHuPXpj9OT6ANgz8B+/tb6FPee+58sKWo45mMNDBQqbQ+lLK87dTqVRA8NoJxulFNanIUQuWe3dxH/9s+gf3C4YNcBUutfxvmRG6Y9mdNtsbFJHMCQTeJ3BzFum49yy+eOyD9BVzE3nfoZzqp9A0/ue4BiTwUXz30btcFj32OJE2f3xEjcvwf7ubbRMvOSGqw31GEEJjfabDbJj+6kLKvxVfJfZ32Z08PDi594TQ8fXvwPvLH2ooJc4OPZ9hdHk7hDEnaCn+7+LYlU4piPLfFWsrpq3bjyUm81Zd6aTIY569jtjSQf+gGJ7/wvEj/5EvbezejEsTeQF0KI6WZv3X04iTtkKEbqhdfQ9vTOjdYt6eeh29sG0IOpab22EFNR7CnnrNor+MTZ/8ktq/+Z+eHlOC3ZeiDT7B29Y5I4gNSjB9H7B3IUUX6blT1yAA3Befz72n+iK9aLw7Co8JQec1hlPtvWuyNt+fb+XUTtGA7TMeFj3ZaX65d/AoCXmx9Do5lfvJxbV3+e4jzreSwkdmczie99GnqHJ0jrziYSO17E8YF/Q81bnuPohBCzmb3zQPry3Y2QTIJz4s+MqVKh9M+tSl3gLLyGVDH7mMeZsiJOno6nSD2Zfpvp5HOtmMtkGOvRZm0iBxBw+AnkyUIlU3F22RoeaPzzuPLzys/EZx1/zkGZr4YPnP4luoZaSNkpij3l+J2yOtBU6Mbto0nc4UJN8i/34rjpCyi3TI4WQuSG0TAH+7XxDYDGwjngmN7bAlXlRpW70G1jRyc43l6DEZi+BFIIUQAMBd70ibLyzuqUZUKzcmjlTLOyeCmnHDUXrtRVwptrL5/0Yixuy0t1YD51RQ2SxGWA7ki/Kb1ub0TH0w8tEkKIbDBOmQehwNhCnwfz9KXTPr3ACLtwfXQB1hUVqBInxkIfrv+1EHNR4TeqCiGmRlkG1kXpp/WYZ6ZfwHC2k/R2BqjyVvDvp3+R13q38VLnJpYULWR1eAV1PpnjliuqfsnYglAZ5pk3QHAButuBbSYwfNL6LITIPqO0GOeH3oW9ZRf263sw5tdhrGzAKM/OpsZGuRvHW6qxLi1HOZSsRifyTsrWdEZToKHEY2IaMuw3W4z6AI4bF5P4zS4YTELQieNdCzFqZCRTOlJ7zhCV3nIqveVcWnVBrkMRgKqYi1pyBnrb81BSjXn+p0j8tgcGWoAWjIVBHDcuxijz5DpUIcQsZJQVY1y4Bn3B6Vld5EtHY+hoHOVxYQRlBTqRf1oGk9y/o4/f7ujDQPG2RQGuXhigQhpfs0J5LKyzKzAWF8FQCrwWRrEsKjMRSeSEmAZGMIz1to+jm3ZDqoj43W0Qt0eP2zv7SPxxH84bGlBOmTgthMiNbCVxOplCN7aSePApdFM7xtxqrCvOQdWUo6S3Q+SJgbjNN17s5InGyEiJ5kebe2kaSPL/nVGK1yEzkrLFCMuWTZMhv5FCTBMjWIK5ZC1QOiaJO8R+oQ3dG8/4dXV3H/bBNuzOHrQ9/rpCCJFturmD+H/+Ar3zAESi2Ft2E//Pn6NbOo7/YCGypGUwcUQSd9gj+wZpGUzmICIhjk165ISYbhO1eGe4JVwnktjb95L45cPDe0S5nFhXnIu5ZjnKJ0M4hRC5oVM2yac2wtENS4kkqVd2YFTLVjciP0SS6fdR1MBQUhpGRf6RRE6IaabK3OA2ITp2s1vzzHJUKHNzRHRrB4l7fjv8iQMQi5P8/d9R4RDmioZjPlbkH61tegabGYx24nL4KPJW43RIQi6GadtGd7dCLILy+FHFebyiWyqFbu9Ke0i3dGY5GCEmVuoxCTgM+hNjk7aQy6DEI9MgRP6RRE6IaaZK3TjvWE7iB9vQ3cN7JxkrwlhXzUE5MvfBkHpt5+Ek7gjJx57DWDQH5ZKFBQpFMhljZ8tTPPj8vxBLDACKU+ddwwXLP0jAW57r8ESO6Ug/qVceI/WneyAWAV8R1ltux1hyJsqZf/NKlNOBuWoxyT1N446Zp0ojk8gfVT6Lfz6njDufaCU18nlqGfD5c8qolMVORB6SRE6IaaaUwlxQhPrUKnRPDOUwUGF35pfcjk8w3y6eHD+kSeS1zv69/Hb9pzicmWte2XM/ZUULOWPx9bkMTeQB+8DrpO7/1uGCwV6S930Fx+3fGL/1SZ4wls1HPfsquvnwnDg1vwZjXnUOo8oNpVQY+AUwF9gLvFNr3X3UOXXAj4FKwAbu1lp/I7uRzj5KKdZWerj3qhp29cRRwPyQk7oT2Kw+lYozGGkllUrg8ZTgdsnevGL6SCInRJYYIReEpm8JXXNZA6nHXhhffu5qlCf/WunFxBo7NpGue3XDjp+ybM4b8Lmzs9+XyD86ESf11G/THktt/DtGviZyJSGc738rdnMHuq0LVVWKqixFFc3KjcDvBB7VWt+llLpz5PtPH3VOEvhHrfVLSqkA8KJS6hGt9ZZsBzvbOEzFvCIn84pOfBTL4GArr26+l207foNtJygrXcG5Z32OcPGiaYhUCFm1UogZQ1WVYV11/phFVIxVizGXzM9hVOLkpJ9wj9boCQ6JWULbEI+mPxYbv9pePlHFQcyl87HWrcFcPBdjdiZxANcA9458fS/wlqNP0Fo3a61fGvm6H9gK1GQrQHHibDvJ5m0/Y8vrP8e2EwC0d7zKn//6YfoHxg8rFiITpEdOiBlCed2YF5yOsbwB3TeA8npQJUWztjduMsOXRs7bC/QDKSCptV6TvSjTqy09FVAcndCtXXQ9fo/0xuU7Oxkj1dNEqr8N5fJjhWowvaGMPLdyujHPehPJvZvHHTNWXZyRa4hpV6G1bobhhE0pdcyJr0qpucBq4LkJjt8G3AZQX1+f2UjFpA0OtrL19V+MK49Gu+jrP0DAP/uGEYvpJ4ncNOqPtNHSs40D7RspDcyltmwV4YBUsmL6KKcTVVkKlaW5DiUfTGb40iEXaa3zZkOrksBcrj3nqzy44Yuji52snHc1p9RdluvQxHHY8QiR1/5E91++BvbwvlOuOWsJv/HzWKGqjFxDLViFsfYK7A1/HikwMC5+N0b1gow8v5g6pdRfGZ7fdrTPneDz+IHfAB/XWvelO0drfTdwN8CaNWukzz5HbJ0ilUo/V/1QD50QmSaJ3DTpj7TxwHP/zL6250fLfK4wN1z8PUqD83IYmRCzxjXAupGv7wUeY+JELq9YlovFtRdTWbyEgWgXLstLka9Gth8oAMnOfXT/+StjymL7NjC46QGC578fpaY+o8EIhlFvug19ztXoSB/KX4wKV+blipWzldb60omOKaValVJVI71xVUDbBOc5GE7i7tNap58YOQvFEoP0DbWjlCLoKcdp5Ue96PWUUldzPgcOPjGm3DAsAn4ZFSumhyRy06S15/UxSRzAYKyLF3f8gktXfRLTlLdeiGk22eFLGviLUkoD/z3Suj0tkskoWikc5vEXvVHKIOSvJeSvna5wxDSINb6Stnxg42/xrb4WK5CZza+V24eSHrhC9QBwE3DXyP+/P/oEpZQCfgBs1Vr/v+yGl786+vfz4MZ/Y3vLUygMltddyuXLP0I4DxIlh8PL2tM/Tk/fHvr7DwBgGk4uPP8rBKQeF9NEsolpcqD95bTlu5qf5rxlffjMcJYjEmLmydDwpXO11k0jid4jSqltWusnjj5pKvNQBoY6OND2Ii/v/CWW6WbNouupKlmBxxU8oecR+U850/cOGA43ypANhQUwnMD9Uil1C7AfeAeAUqoa+L7W+irgXOAfgFeVUhtHHvdZrfVDOYg3LwxGu/n5M3fS3LsNAI3Nqwf+QiwR4bozv4LbmfvFc0JFc7nq8u/R33+QVCqK31dNIFCNYcgedGJ6SCI3TUqC6VcKLAnMxWHmxzAAIQpdJoYvaa2bRv5vU0r9DjgDGJfInew8lFhikGc2f4+NO381Wrav5RkuXv1JVjdchyE39zOKs3o5mA5IjZ0TEzj7Zkzf9DbgpaJ9kExgeEOSNOYxrXUncEma8ibgqpGvn2J4xSMxojvSNJrEHWl7y1P0DrXgdi7MQVTj+bzl+LzHXL9GiIyR7QemSW3pqfhcYz+0lTI4d9mtMs9FiOw4NHwJJh6+5BvZowmllA+4HHgtk0H0DTaxceevx5U/+eq36Ys0Z/JSIg84SuZS9q5vY4aGh3opy03w/A/gWXj+tF3TjvYRef0x2u/7EK333kTvE/9Nortx2q4nRC6kjrFgSFIWExGzlPTITZNwoI4bLv4eL+74Jbua11MSnMt5S2+lojg/N2sVk5eK9KDtJKa3WFq989tkhi9VAL8bno6CBfxUa/3nTAYxFOsh3b5wieQQsXh/Ji8l8oAyTNxzTqf8xnuwI90YDjdmsAo1jfOih7Y/QdcfPz/6ff/TPyC66ylKr/smll9WsBUzQ5GnAp+rmMHY2F1kSvx1BD2ZmXsqRKGRRG4alQbncemqf+S8ZbfhMN3SE1fgUpEeYnufp2/997GjA3hXvBHfqrfgCOV+krUYb5LDl3YDp05nHF53GKVMtE6NKXc7g7idRdN23WQyRv9AI/0DB7FMN8FgPX5fuumEYjpY/lLIQhKV7Gul5+/fHFeeaH2dZNcBSeTEjBHyVXH92f+XHz/1cWLJAQC8zhDXnfmvBNzyey5mJ0nkpplpWvjM4lyHIaZIa83Q1kfofviu0bL+p+8huudZSt/xdSy/bNIs0ivy1XDOsg+w/rXvjCm/5LRPE/RlZl+xoyWTMfbse5innvkiWtsA+LwVvOGS7xAKyfYnM4lOxrEHO9Mes6O9WY5GiOk1p3Q1t1/2U3oGD4IyCPtqKPbJRtti9ppSIqeUCgO/AOYCe4F3aq2705y3F+gHUkBSa71mKtcVIttSfS30Pv6dceWJ5i2kug9IIicm5LDcrG54BzVlK9m6909Ylpulc66ktGghI0M6M66//wBPPv0vHDmkczDSyoaXv8m6876Mw+GdluuK7DM8QRxVS0k0bxl3zApOT0OBELmilKLEX0uJLOcvBDD1Hrk7gUe11ncppe4c+X6iDXcv0lp3TPF6QuSETsawo31pj9kxmeckjs3jKmJOxRnMqTgjK9frGzhIunl5BxqfYCjaJYncDGJ6iii+/NO03/cBdDI6Wh44+2bMYhn2LYQQM9lUE7lrgHUjX98LPMbEiZwQBcvwhHCULSTRvnPcMTMo845EfrEsd9pyh8OHYciI+pnGWb2Mipt/QqzxFVKRLtz1p2OVzsd05X5fLSGEENNnqp/oFVrrZoCRvZom2jhDA39RSmngv0f2YxI5YNtJOvv2sK91A7F4P3Mqz6S0aD5up2xMfCymN0TxlZ+j/acfGtPqXXThh7FksRORZ4KBOjyeUoaGxg6CWLn8ZtnfaAZSSuEom4+jLP3+pUIIIWam4yZySqm/Aum6HD53Atc5V2vdNJLoPaKU2qa1Hrfh7sj1bgNuA6ivrz+BS4jJaGzfyK8e/zC2nQTg6c3/zfkr7+C0hnfhlOFWx+SsWU7F+35C7OCr2EO9uOpW4yiZi+GU903kl4C/misu/S+e2/DvNLU8i8PhY+Wym2mY/2aUku1DhRBCiJnguImc1vrSiY4ppVqVUlUjvXFVQNsEz9E08n+bUup3wBlA2kRupLfuboA1a9aMn+QhTtrAUCcPb/jiaBJ3yJObvs2C6gsoCy3MUWSFQSkDR+l8HKXS6i3yX3FoARdf+H+JRrswDAufr0KSuAKmbRvd3QGRAXB7UcWlKEuGyQohxGw21U+BB4CbGN549ybg90efoJTyAYbWun/k68uBL07xuuIkROM99Aw0pjmiGRhqk0ROiBnG6fThdPpyHYaYIh0bIvXKCyQf+BnEomA5sC67BvPMC1A+mQcnhBCz1VQTubuAXyqlbgH2A+8AUEpVA9/XWl8FVAC/G1lm2wJ+qrX+8xSvK06Cw/TgcviJJQbGHXPJHDkhxEmIxwdJJiO4XEFM05XrcGYk3dRI8lc/PFyQTJD8069RldWYS1flLC4hhDgROpFEd3RhN3egHBaqohRVVjxtW/EA2B37sTv2QyqGUToXVVKHspzTdr1sm1Iip7XuBC5JU94EXDXy9W7g1KlcR2RG0FfJuSs+zN9e+tqY8jkVZxDy5e+CHYmu/cQaN5HsacRdtxpH2UJM2bftpGitoacfnUiC14Ph9+Q6JFGgkskYnR2beeXl7zDQv5/K6rNYuvwmQqEFuQ5txkm+9Gza8tRTf8VoWIZyOLIckRBCnBidTJF6dTvJ+/54eHccpwPnB9+Jmjs996CpA68R/eGHITY4XGCYuK77CuYpF6LMmVFvygD7WUQpg6VzrsDvKeXZLfcQTwyycsG1LKm/HK+7ONfhpRVv30X7/9yGPdQDDO8q71lyCcVv+DSmT5K5E6EHh0i9sp3kg09BJIqqLsXxjstQ9VUoY/paw8TM1NW5hb8+/H4OfSLv2fVHDjY+yRuuupdgUBaqyqij5jUfLrdJt1+gEEIcEov2kEwO4XAGcDpzNxRbd/WS/PmfxlZZ8QSJn/0Jxx3vxghkdhqAPdBF7NefP5zEAdgpYr/+PJ47foYqnRmfU5LIzTIeV4jFdZcyp+JMbDuJxxWa1i7tqbATMfqevmc0iTtkaNuj+E9/pyRyJ8jesZ/kr/46+r1u6iD+nV/h/MQNqMrSHEYmCk0iEeG1Td/j6CQiHuulvXVjRhO5aLSHWKwXy3Lh883OPRut084m/vyT48rNcy5GOWbOECEhROYkk1Ha2zby0gv/QV/vPsrKV7H69I8SLlmSk4WvdE8fJFPjy9u7oD8CGU7k9EAnumPv+AOJKLq3BWZIIidLmM1SbmcAr3t6xyVPlR3tI7Yn/ZCieNPmLEdT2HRkiORf0ryXiST23qbsByQKWioZpb8/3cJJ0Ne3NyPX0NqmrW0Tj/z1Q9z/+7fyhz9ez/YdvyMW68vI8xcSVVWH9cZ3gGmOFCjMC9+AmteQ28CEEHmrq3Mrf3vkw/R078C247S2PM8jf76V3p49uQnINUGjk2mCI/P9SspygjnB8zpmzrQS6ZETectwuLFCtcQjPeOOWcW12Q+ogOmkjR6KpT84EE1fLsQEnK4gNTXn8/q2n407VlF5ekau0du7h7888gFSqeHf21ish2ee+RIuV4g59Rdl5BqFQnl9mOddhrF0FXqgH+X1oUMhUolB7Gg/ljuQ6xCFEHkkmYyx5bV7x5WnUlGaDj5FqDj7c5lVSRGqvgq9v3lMuXn+aahw5hfcU6EqrLVvI/nsL8aUGzWnoML5uy7EiZIeOZG3DHeAonW3jys3/WU4KxbnIKLCpfxezLXL0h4zFkpSLE6MYVgsWvIOPN6yMeU1tRcQKl6UkWu0tm0cTeKOtGnT3bOzV86yMMqrMOcvIm5FaXn0K+z5ybs58Ns7GNy/ATshDTJCiGGpVJzBwea0xyYaTTHdDL8Pxz+8GeOcVeB0gM+D9aYLMS9cgzo02iCDlOXEccF7cVz8AXD5wLQwT78G13V3YfjDGb9erkiPnMhrzuoVlN3w3/Q+eTepnoN4Gi7Ev+ad0iN3gpShMM9cjv36XvT+ltFy68pzUOUzp0IT2RMsmstlV/yArs6t9PXtp7R0GaHQQjzezMy3jEV70pfH+rDtREauUYhi3fvY/4v3k4r2ApAa6mH/rz5I/TvvxleXmd5QIURhczr9zJ13BRu7d4w7Vlt3QQ4iGmaUhHC89RL0JWcNT+0p8k/rFB+jqBzHRbdinX41aBsVKJ1RWw+AJHIizxlOD6760/G/+X8TH+omRhJlaYJ2AsOYGUvHZotRUoTjlregO3pgKIYqDqDCRSj3zKrURPYEArUEAtPTqFJZuSZt+cIFV+MwnETbd2AnY1j+UpyBE1sEJZWK0z/QRDzRj9tVjN9XhWFkvkV4Ogwd3DSaxB2p/Zm7cVf8P0zZAF6IWU8pRf3cy9m7+8/09BxO5urqL6E4vCSHkYEyTVRx9vYuVoaBCs3chbIkkZtlUnYSrW0ss3Bu3ts6NvGnRz9MMjkEDA/ruuzC/0dt9Tk5WXmpkBlBHwTlRk/kv6KieaxccSubXv3+aFlJyVIW1V/GwT//MwO7ngDA8pZQ88av4Ks9bVL1wVC0i83bfsYrm+9F6yQOy8s5az/N3PpLcDi80/Z6MiXek35YVLL3IDoRBUnkhBBAIFDDRZd+k97ePQwONBMsmkewqB63W0bhzCSSyM0SsUSE5p7XeWbnz4nEelgz/63MKzuNoKc816Ed02Cknb8/9dnRJA7AtpP87anP8tarfkpwmnoDhBC55XIFWbbsRubMuYTBwVaczgBFwTq6nvnBaBIHkIx0sv93H2PBe36KKzznuM/b3PICG1/7wej3iWSEx5/5PEXBuZSXrZiW15JJ3rrVdD43vty/4EJMd1H2AxJC5C2vrwKvryLXYYhpJIncLLG95Sl+/uydo9/vbt/Ayro38ObVd+J15e+H/9BQBwNpJuwmEgMMRtokkRNiBnM6/YTDiwmHhxc3ivc20fPq/ePO08ko8Z4Dx03k4vFBXt12X9pjexv/XhCJnLu0gUDDJfTveHS0zPKXU7zqnaiJltoWQggxI0mtPwv0Rlp5cOO/jSvfdOBhzl30D3mdyJmmC1AcvfEwgGW5sx7PVOhkAt3dCckkBIow/LJkuBAnbnxdcOzyo0+zJyhOX55vLF8JlZd8muLV1xFt2YIjVIO7fDHOopmznLYQQojJkURuFoglB+mPdqQ9NhjrynI0J8bvq2LBvKvYtefBMeUV5asJ+KpzFNWJ0z1dJB97iNSzj0MqhaqqxfHO92HUzs11aEIUDMtfRmj5NXS/8usx5cpy4QzVHffxTqePZUuu57H1nxt3bG79xRmLc7pZvhIsX4msUimEELOcrBQxC7gsP0We9GOkA+7MLBU+XRwOD2tX3c7yU/4B03RjGBaLF7yFC8/+F9zuUK7DmxSdSpF85u+k1v8NUqnhsuZG4nf/O3ZnW46jE6JwGKaD0jU34Z97zmiZ6Smm/pqvTyqRA6iuXMuyJdePLoximm7OPeOzFIfmT0vMQghR6OIDbfQ2vkDHjr8w0LqZZHT27eWZr6RHbhYo8pbz5tWf5r6n/xF9xPCjtfOupdiX/8Nx/L5Kzlj9EZYtfheg8XpKMQto1U3d203qyUfGHxgaRHe0QUl+LzgjRD5xFlVTc9VXSPQ1oZMxLF8ZzqKqST/e6yllzam3s2ThW4nF+/G4iwn4azAM+TgUQoijRXsbef3Bf2Soa+doWcWKd1J7xm04PMU5jEyAJHKzxsKKs/nAJffy0t4/MDjUyep5V1MbXobHWRjztAzDIuCf/M1aXtEaUsn0xyYqF0JMyHIHsNyLT/rxDoeH4tCCDEYkhBAzj9Y2bVv/MCaJA2h99ZeULLwUR40M7841SeRmCYfloi68nLrw8lyHMuuoQBBjxenYr2wYe8AwUdIbJ4QQs4pSKgz8ApgL7AXeqbXunuBcE3gBOKi1flO2YhQCIDHUQ+f2P6c91nPgOYKSyOWczJETYpoppwvrimtRlUcMY7UcON7zQVRJWe4CE0IIkQt3Ao9qrRuAR0e+n8jHgK1ZiUqIoximE4c3/QbiLp80ROcD6ZETGRWL9dLdvYMDBx7H4QxQV3sBodACTNOR69ByyiitwPH+T6K72iCRQIXCqHCp7PskhJg0rTW6+yC6pwWUgSquxghV5josceKuAdaNfH0v8Bjw6aNPUkrVAm8Evgx8IkuxCTHKcvmpXXsr2/7w0THlhsNDoOa0HEUljiR3kSJjEokIW7bcx6ZXvz9atmnT97jk4m9QU3POMR45OxjBIgjm7559Qoj8Zu/bSPQn/wui/QCoYDmuG7+JWdWQ48jECarQWjcDaK2blVITdW18HfgUcMzJ7Eqp24DbAOrr6zMYphDgrzyVhiu+yr71/0G8v4VA1SrmnPcJPMXzch2aQBI5kUEDAwfZ9OoPxpRpneKZZ7/MVVf+CK9XhhEKIcTJsLubiN73ydEkDkD3tRH79T/hft93MXyyelw+UUr9FUjXXTp+E8P0j38T0Ka1flEpte5Y52qt7wbuBlizZo0+1rlCnCjL5adk4aUEqlZhp2JYziCWuzAWypsNJJETGROJtAPjP0MGB5uJxXolkRNZpZR6B/AF4BTgDK31CxOcdwXwDcAEvq+1vitrQU7SYKyb9v69NHVtJeitoDq0mLC/NtdhiSzSvS0Q6Rlf3rIT3dcOksjlFa31pRMdU0q1KqWqRnrjqoB0G4qeC1ytlLoKcANBpdT/aK3fM00hiwIxGGmno2srjQfXEwzWU1t1NqGieSilpvW6Tl9+7zs8W0kiJzLG5Uo/bNDpDOBweLMcjRC8BlwL/PdEJ4ysCPdt4DKgEdiglHpAa70lOyEe32Csm4c3fZMX9/5+tCzgLuV9F/435UEZ2jJrHGufO9kDr9A8ANwE3DXy/++PPkFr/RngMwAjPXKflCRORIY6Wf/cl9l/8InRspcsL2+6/AeUhE9+SxZRuGTVSpExfn8NtbUXjis/7bSP4vPl1x5wOpXE7utBD0VyHYqYJlrrrVrr149z2hnATq31bq11HPg5wwsR5I32vr1jkjiA/mgHT73+Y5KpeI6iEtmmiqtRZXPHlZtLLsAoqsh+QGIq7gIuU0rtYLgR6S4ApVS1UuqhnEYm8lpP7+4xSRxAIhnhxU3fJZGQ+5nZSJrxRMa43SHOOvNOGmvOZfuO3+Bw+Fm+7EbKyk6d9i7/E2G3N5F86i/YW15ChcJYl78do34hyuU+oedJxgaJ9R8kPtCG5S7CXVSLwyPDmwpMDXDgiO8bgTPTnZirBQUauzanLd/a9DiXLPsQRV5ZAno2MAKluG/4d2IP/Cv27hdAKcylF+O84mMoty/X4YkToLXuBC5JU94EXJWm/DGGV7YUs1xXz8605S2tLxKL98nop1lIEjmRUT5fBYsXv525c9+AYZh5V6nYXe3E774LersA0L1dJL53F45bPoW5eOWknycx1EPTyz+m+aV7R8uK6s9m/kX/hCsgy4Fny7EWFNBajxuulO4p0pSlXSwgVwsKFHnT97aEvJU4LFe2whB5wCibi+v6f0P3t4MyMIIVKJcn12EJITIgGtP0D9qgIOgzcDnHfzwF/TVpHglFRfNwWPl1vyWyQxK5PJdKJTAMK696tCbD5crPFY108/7RJO5IyT/9AqN2Psrnn9TzRDp3jkniAHr3P0P3nieoXPnOjMQqju9YCwpMUiNQd8T3tUDTFJ8zo6qLl+BzhRmMjf29vWz57Xidsp1FLujBAXRXG3poEBUMoYrLTrhH/2QZngB48rN+FUKcnM4em4eejLHnoA1AQ73JG851Ei4aOwMqXLyIouAcevv2HVGqOGP1R3C5glmMWOQLSeTyVH//QQ4cXM/e/X8lHFrIooa3ECpagGGYuQ4t45LRPmLt2+ne/AeUMggtezPusgbMaUgGdXdH+vKudkhMfr5R957H0pa3bfkdpYuuHLM0r9Y2id5mUvF+TFcAR7C64BLzGWwD0KCUmgccBN4FXJ/bkMYq8ddx67q7eXzbD3m96UlC3kouW/ER6ktX5Tq0WUn3dJL43Y+wt748XKAU1uVvxzz7EpR3cg1BQggBYNspevoG+M0jDlo6Dw/02LE/RTwZ452Xu/C4Dydzfl8lb7jom+zc82f27H+EgL+GU5e9l5LiJbkIX+QBSeTy0MBAM4/8/eP09O4CoKX1BV7f8VveeMUPKS05JcfRZZadTNCz+Q+0Pv7/Rst6Nj9A5UWfIrzybSgzs7+iRs3c9OWLV4J38vNMLHf6nhDTGRwTcyo2QN/2R2h77OvY8QEMV4CKdf9IoOEiTJfc9E0npdRbgf8EyoAHlVIbtdZvUEpVM7zNwFVa66RS6g7gYYa3H7hHa51+UloOlQfn85bT/4nIil4cpkt64nIotfnFw0kcgNYkH/4VxvwlqHmyapwQYnLi8T527/4Tcb2Mls7xKxDva7LpHdB4jursDwbqWL3iVpYtvg7TcmGZMsR+NpvSqpVKqXcopTYrpWyl1JpjnHeFUup1pdROpdSdU7nmbNDZtW00iTskZcfZuOl7JBJDOYpqeiT6DtL21LfGlbc++U3ifdMwwq2sCuP088aWef1Yl7wF5Zx8ZVg89wJUmiW/q0+/EdNxeM5KtHUbLX/5P9jxAQDsWD/ND3+BaPv2k4tfTJrW+nda61qttUtrXaG1fsNIeZPW+qojzntIa71Ia71Aa/3l3EV8bA7TRZGnXJK4HNKRAVLP/i3tsdSRyZ0QQhxHR8dWnnv+q9j2xOdMdEwphcsVlCROTLlHbkbs05RvOrvT3+R3dG0hkRjE4Zg5k9tTQ73oNEuo62SUVLQ349cz/EGsq65Hr7kQvfd1KC7DqF+AUXpiC5R4wvNZ8ub/ZPdjXybW24jlDlF/7scIVB5eMEVrm+5XfpX28T2v/BZf7WlTei1CiCwzTJhgLpxyy0IDQojJse0kr7/+i+GvEzvxe+czEBm7hla4SBH0yzSMfKS7+9DRGLicqOJgTqfLTCmR01pvBY73Akb3aRo599A+TZLITWCiTR3LSlfgdM6s4XiGOzi8ma2dHFOuTMe0zJEDMAJBCARhwckPUzVMB0V1Z7Ds2ntIxvowHZ7xq1VqjZ2Ipn28nYygtZa5ckIUEOX2YK17E4kff33sAcPAWHJqTmISQhQerSFlJwDYsf3bXHH2qTy4vpyhkVsGn0fxtktd+L2y3XM+0fEE9tY9JH7zKPRHwOvGuuZCzBUNKE9uekez8RuSbp+m9OunCgBKwksIF49N5kzTzaoVt2BZ2VkZLVucRdWUnXHzuPLSs96PI5j/vyZOXwne8Ly0Ww4ow6R45bVpHxda8VZJ4oQoQMb8JVjX3AjukZERoRIc7/v/UOX5X18JIfKDaVosWvR2AKLRTjZv/CCXnf4UV19wkOuvtLn1WjdVZTNvcbtCp5s7SPzoD8NJHEAkSvJnD2M3tuYspuP2yGVzn6aR6+Vk09184vdVcsm6f6ep+fnhVSuLF7Fg3lUUh+bnOrSMMywX4dXX4alYSufGXwAGJauvw1O1HMNy5Dq8KfNUrSB8+g10vfhThn/tFeG1N+KpXJ7r0IQQJ0F5/ZhnX4pxymqIRcHrxygqznVYQogCU1a6gmXLbmLLlp8QGWpn08Z/YsWK91E7/9243dITl4+Sz6dfCy31+EsY82tQZvaT7+MmctnepylXm+7mm4C/msUNb2HRwqtRamb/QVueYgILLsA35yxQCsMs/ATuEMsXpvScD1K0/GpSkW5MTzGOompMp8ynEaJQKcNAhctyHYYQooB5PGFOXXkbC+a/iVisB7c7hN9fM+NGXs0o0VjaYh2Nga2H177OsmxsP5D3+zTls5mexB3JsJy5DmFamE4vZunCXIchhBBCiDzicHgoLl4wbc9vJ2PYqTjWNK05MNuYa5Ziv7RtXLl1zqkoR252dJvq9gNvVUo1AmczvE/TwyPl1UqphwC01kng0D5NW4Ff5uM+TSI/RKPddLS/RnPTs/T27CaZTN/6IYQQQgghxrOTMQYPbqTxj59h3y9vo33DvcR7DuY6rIKn6iow150+psxYuwy1oDZHEU191crfAb9LU94EjNmnCXhoKtcSM99AfxPPrP8X2lo3AKCUyelnfJL5C96EwzH5zbqFEEIIIWaryMFX2PebD3NoSYpo+3Z6t/6JOW/9Bo5ARW6DK2CG34t6w9mYa5eh+wZRfg+qpAh19K7t2YwpZ1cW4gha2+ze+cBoEjdcluKF575KX+/e3AUmhBBCCFEgkkM9tDz+Hxy9rmCsYwexrr05iWkmUW4XRnUZ5pK5GLUVOU3iQBI5kSeGhjrZufP+tMfa2jZmNRYhhBBCiEJkJ4aId+9NeyzR15zdYMS0y83MPCGOYhgWluVJe8xh5fcKjzqVQPe0gk6i/KUo98zatF2ImSI+2Eky1ofp8KTd+1EIIQqd6QrgrT6VwQMbxh1zFs8Z/dqO9pMa7ESZFmagEmVmJiXQQxF0dzs6HkMFQ6jiMtk3dxpJIifygttdzLIVN/Ps+i+MKTcMB6VlK3MT1CTYva0knvofks//GpJxjPlrcb75U5jlM2/PPyEKlZ1K0N+8kT1//zLR3gNY7hD153yM8Px1WO5grsMTQoiMMV1+Ks7/KHt++X50MjpaHlh4Ea7wcCIXb9tB95//lXjjKyjLjf+Md+M//TqswNS2VbG7O0je/2PsrS8NF3i8ON71YYyG5ShLUo7pIEMrRd6orjmPU1ffjjmyh4o/UMvFl32HYNHc3AY2AZ1KkHjyJySf/ikk4wDYuzcQ+9FHsHtbcxydEOKQoe49bHvgDqK9BwBIRnvY/bd/ob9lU44jE0KIzHNXLGH+9fdSft5HKF5xLfVv+TpVF9+J5Q2T6Gmi/acfIt74CgA6GaX/6R8y+PJv0XbqpK+pUylSTz9yOIkDGIqQuPc/0B0tU31JYgKSHou84fGEWbr8JubMewOp5BAuVzEeb2muw5qQ7mkd7ok7ury3Bd3ZCEWyMpQQ+aBn33q0nRxX3vTSjwhWn4bpzO/h20IIcSKUMnCXLsSdZg/bVPcB7Ej3uPL+5/8H38qrsUJVJ3fR/h5Szz46vtxOodsOQmXuluifySSRE3nFMCwCgQL5Y7eTkEqkPaRT8SwHI4SYSCLNTQtAMtaPbScxsxyPEELkik6mvz/R8SG0PvkeOVAw0Vw4mSM3bWRopRAny1+CMe/08eWmA3WyLVpCiIwrnndB2vLypW/BIXPkhBCziBmqBjW++cqz6EJMX8nJP3EwhHnu5WkuaKEqCqSBvgBJIifESTI8AZxv/jQqWH640HTgetddGMU1uQtMCDGGt2Qh1ae9d0xZsPYMwvMvyk1AQgiRI1ZxLeGrvwjG4UF5ZlE1RRd9BMOZfvXwyVCGgXXmxRirzzncA+cvwvG+T6JKZKrJdJGhlUJMgVmxAPcHfojuakQn46jiaoziGpTlyHVoQogRDk+I6tPfR0nD5cQH27HcRbiDtTi8xbkOTQghssqwXHgXX4yzYjHJ3maUw40VqsU6slH6JKlQCY5r34e+6GqIx1DBIlQof9c6mAkkkRNiioxQJYRkTyoh8pnl8mGVLcZXtjjXoQghRE4py4mjdB6O0nmZf26XGyULm2SNDK0UQgghhBBCiAIjiZwQQgghhBBCFBhJ5IQQQgghhBCiwEgiJ4QQQgghhBAFRmmtcx3DhJRS7cC+XMeRRinQkesgToDEO70k3hMzR2tdlsPrT9lJ1E25fs8zTV5PfpPXc3JmY900HQrp909inR4Sa+alrZ/yOpHLV0qpF7TWa3Idx2RJvNNL4hXHM9Pec3k9+U1ej8ilQvp5SazTQ2LNHhlaKYQQQgghhBAFRhI5IYQQQgghhCgwksidnLtzHcAJkninl8QrjmemvefyevKbvB6RS4X085JYp4fEmiUyR04IIYQQQgghCoz0yAkhhBBCCCFEgZFEbhKUUu9QSm1WStlKqQlXtlFKXaGUel0ptVMpdWc2YzwqjrBS6hGl1I6R/4snOG+vUupVpdRGpdQLOYjzmO+XGvbNkeOblFKnZTvGo+I5XrzrlFK9I+/nRqXUP+ciziPiuUcp1aaUem2C43n1/s4khVZnHE+h1CnHU2h1zrEUWn10PFJfFa5Cqu8KoS4rpHqqUOqhGV2/aK3l33H+AacAi4HHgDUTnGMCu4D5gBN4BViao3i/Btw58vWdwFcnOG8vUJqjGI/7fgFXAX8CFHAW8FwOfwcmE+864I+5ijFNzBcApwGvTXA8b97fmfav0OqMSbyevK9TJvEaCqrOycBryav6aBKvSeqrAv1XSPVdvtdlhVRPFVI9NJPrF+mRmwSt9Vat9evHOe0MYKfWerfWOg78HLhm+qNL6xrg3pGv7wXekqM4jmUy79c1wI/1sGeBkFKqKtuBjsinn++kaK2fALqOcUo+vb8zSgHWGcdTCHXK8RRanXMshfS7MylSXxWuAqvv8r0uK6R6Kl9+psc1k+sXSeQypwY4cMT3jSNluVChtW4GGPm/fILzNPAXpdSLSqnbshbdsMm8X/n0nk42lrOVUq8opf6klFqWndBOWj69v7NRIb3/hVCnHE+h1TnHMhPro+MplJ+NSC9ffn75XpcVUj01k+qhfHlPT5iV6wDyhVLqr0BlmkOf01r/fjJPkaZs2pYEPVa8J/A052qtm5RS5cAjSqltI60W2TCZ9yur7+lxTCaWl4A5WusBpdRVwP1Aw3QHNgX59P4WnEKrM45nBtQpx1Nodc6xzMT66HgK5WczIxVSfVfgdVkh1VMzqR7Kl/f0hEkiN0JrfekUn6IRqDvi+1qgaYrPOaFjxauUalVKVWmtm0e6htsmeI6mkf/blFK/Y7ibPFs3XZN5v7L6nh7HcWPRWvcd8fVDSqnvKKVKtdYdWYrxROXT+1twCq3OOJ4ZUKccT6HVOccyE+uj4ymUn82MVEj1XYHXZYVUT82keihf3tMTJkMrM2cD0KCUmqeUcgLvAh7IUSwPADeNfH0TMK61TCnlU0oFDn0NXA6kXc1nmkzm/XoAuHFkNaGzgN5DQyJy4LjxKqUqlVJq5OszGP776sx6pJOXT+/vbJRPdcbxFEKdcjyFVuccy0ysj46nUH42Ir18qe/yvS4rpHpqJtVD+fKenrhMr54yE/8Bb2U4W48BrcDDI+XVwENHnHcVsJ3hVXw+l8N4S4BHgR0j/4ePjpfhVYZeGfm3ORfxpnu/gA8CHxz5WgHfHjn+KhOshpVH8d4x8l6+AjwLnJPjeH8GNAOJkd/fW/L5/Z1J/wqtzpjE6ymIOmUSr6Og6pwpvpa8qo8m8XqkvirQf4VU3xVCXVZI9VSh1EMzuX5RIy9ACCGEEEIIIUSBkKGVQgghhBBCCFFgJJETQgghhBBCiAIjiZwQQgghhBBCFBhJ5IQQQgghhBCiwEgiJ4QQQgghhBAFRhI5IYQQQgghhCgwksgJIYQQQgghRIGRRE4IIYQQQgghCsz/D2HPYPrOLrdZAAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 1080x252 with 3 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Learning hyperparameters\n",
    "num_epochs = 500\n",
    "learning_rate = 1e-1\n",
    "\n",
    "# Conduct topological regularization for different topological loss functions\n",
    "top_losses = [top_loss_other1, top_loss_other2, top_loss_other3]\n",
    "Y_tops = []\n",
    "\n",
    "for idx, this_top_loss in enumerate(top_losses):\n",
    "    # Conduct topological regularization\n",
    "    print(\"\\033[1mConducting topologically regularized embedding for topological loss \" + str(idx + 1) + \"\\033[0m\")\n",
    "    this_Y_top, W_top, losses_top = PCA(data, top_loss=this_top_loss, num_epochs=num_epochs,\n",
    "                                        learning_rate=learning_rate, random_state=42)\n",
    "    Y_tops.append(this_Y_top)\n",
    "    print(\"\\n\")\n",
    "    \n",
    "# View topologically regularized embeddings\n",
    "fig, ax = plt.subplots(1, len(top_losses), figsize=(len(top_losses) * 5, 3.5))\n",
    "for idx in range(len(top_losses)):\n",
    "    sns.scatterplot(x=Y_tops[idx][:,0], y=Y_tops[idx][:,1], s=50, hue=t, palette=\"husl\", ax=ax[idx])\n",
    "    ax[idx].get_legend().remove()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.9.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
