{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Zebrafish lineages"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/ag16115/anaconda3/lib/python3.8/site-packages/scipy/__init__.py:138: UserWarning: A NumPy version >=1.16.5 and <1.23.0 is required for this version of SciPy (detected version 1.23.1)\n",
      "  warnings.warn(f\"A NumPy version >={np_minversion} and <{np_maxversion} is required for this version of \"\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as pl\n",
    "import scanpy as sc\n",
    "\n",
    "sc.settings.verbosity = 2  # verbosity: errors (0), warnings (1), info (2), hints (3)\n",
    "sc.settings.set_figure_params(dpi=150)  # low dpi (dots per inch) yields small inline figures"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "from scipy import linalg\n",
    "from scipy import sparse\n",
    "from scipy.sparse.linalg import svds\n",
    "import ot\n",
    "\n",
    "\n",
    "from numpy.linalg import matrix_rank\n",
    "import itertools\n",
    "import copy\n",
    "import time as time\n",
    "from sklearn.cluster import AgglomerativeClustering\n",
    "from matplotlib.pyplot import figure\n",
    "from scipy.cluster.hierarchy import dendrogram, fcluster, cophenet\n",
    "from scipy.spatial.distance import pdist,squareform\n",
    "import plotly.graph_objects as go\n",
    "from plotly.subplots import make_subplots\n",
    "\n",
    "from tqdm import tqdm\n",
    "from sklearn.metrics import pairwise_distances\n",
    "from sklearn.metrics.pairwise import cosine_similarity\n",
    "from sklearn.metrics.cluster import adjusted_rand_score, fowlkes_mallows_score\n",
    "from scipy.stats import rankdata,kendalltau,sem\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.ticker as mtick\n",
    "from sklearn.manifold import TSNE\n",
    "import seaborn as sns"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "import random \n",
    "random.seed(22)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "tags": []
   },
   "source": [
    "# Functions"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## General"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "def signif(x, p):\n",
    "    x = np.asarray(x)\n",
    "    x_positive = np.where(np.isfinite(x) & (x != 0), np.abs(x), 10**(p-1))\n",
    "    mags = 10 ** (p - 1 - np.floor(np.log10(x_positive)))\n",
    "    return np.round(x * mags) / mags\n",
    "\n",
    "def flatten_list(l):\n",
    "    flat_list = [item for sublist in l for item in sublist]\n",
    "    return flat_list\n",
    "\n",
    "def check_symmetric(a, rtol=1e-05, atol=1e-08):\n",
    "    return np.allclose(a, a.T, rtol=rtol, atol=atol)\n",
    "\n",
    "def is_pos_def(x):\n",
    "    return np.all(np.linalg.eigvals(x) > 0)\n",
    "\n",
    "def reverse_dict(dic):\n",
    "    for r in dic.keys():\n",
    "        if not isinstance(dic[r], list):\n",
    "            dic[r] = [dic[r]]\n",
    "    inverse = { v: k for k, l in dic.items() for v in l }\n",
    "    return inverse\n",
    "\n",
    "def pc_scores(X, r):\n",
    "    U, s, Vh = svds(X,k=r)\n",
    "    idx = s.argsort()[::-1]   \n",
    "    Vh = Vh[idx,:]\n",
    "    Y = X @ Vh.T\n",
    "    return Y\n",
    "\n",
    "def embed_cov(X, r):\n",
    "    if r == X.shape[0]:\n",
    "        U, s, Vh = np.linalg.svd(X, full_matrices=True)\n",
    "    else:\n",
    "        U, s, Vh = svds(X,k=r)\n",
    "    idx = s.argsort()[::-1]   \n",
    "    U = U[:,idx]\n",
    "    s = s[idx]\n",
    "    ## need to take square root as these are eigenvalues\n",
    "    Y = U @ np.diag(np.sqrt(s)) \n",
    "    return Y\n",
    "\n",
    "def find_rows_to_merge(F):\n",
    "    f = copy.deepcopy(F)\n",
    "    np.fill_diagonal(f, -np.inf)\n",
    "    i,j = np.unravel_index(f.argmax(), f.shape)\n",
    "    return [i,j]   \n",
    "\n",
    "def inner_products(X):\n",
    "    return X.dot(X.T)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def wasserstein_dim_select(Y, split = 0.5, rmax = 50):\n",
    "    n = Y.shape[0]\n",
    "    train = round(n * split)\n",
    "    rtry = int(np.min((train, rmax)))\n",
    "    if sparse.issparse(Y):\n",
    "        Y = Y.todense()\n",
    "    Ytrain = Y[:train,:]\n",
    "    Ytest = Y[train:n,:]\n",
    "    U, s, Vh = sparse.linalg.svds(Ytrain,k=rtry-1)\n",
    "    idx = s.argsort()[::-1] \n",
    "    s = s[idx]\n",
    "    Vh = Vh[idx,:]\n",
    "    ws = []\n",
    "    for r in tqdm(range(1,rtry+1)):\n",
    "        P = Vh.T[:,:r] @ Vh[:r,:]\n",
    "        Yproj = Ytrain @ P.T\n",
    "        n1 = Yproj.shape[0]\n",
    "        n2 = Ytest.shape[0]\n",
    "        M = ot.dist(Yproj,Ytest, metric='euclidean')\n",
    "        W1 = ot.emd2(np.repeat(1/n1,n1),np.repeat(1/n2,n2),M)\n",
    "        ws.append(W1)\n",
    "    return ws"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## IP HC"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "# def ip_metric(X,Y):\n",
    "#     return  np.exp(-np.sum(X * Y))\n",
    "\n",
    "# def ip_affinity(X):\n",
    "#     return pairwise_distances(X, metric = ip_metric)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "def ip_metric(X,Y):\n",
    "    return  np.sum(X * Y)\n",
    "\n",
    "def ip_affinity(X):\n",
    "    ips = pairwise_distances(X, metric = ip_metric)\n",
    "    return np.max(ips) - ips"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "def clusters_to_labels(clusters):\n",
    "    d = defaultdict(list)\n",
    "    for index, sublist in enumerate(clusters):\n",
    "        for item in sublist:\n",
    "            d[item].append(index)\n",
    "    labels = flatten_list([d[c] for c in range(len(d.keys()))])\n",
    "    return labels"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Ranking"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "def find_ancestors(model, target):\n",
    "    n_samples = len(model.labels_)\n",
    "    global ances\n",
    "    for ind, merge in enumerate(model.children_):\n",
    "        if target in merge:\n",
    "            if n_samples+ind in ances:\n",
    "                return [target]+ ances[n_samples+ind]\n",
    "            ances[n_samples+ind] = find_ancestors(model,n_samples+ind)\n",
    "            return [target]+ances[n_samples+ind]\n",
    "    return [ind+n_samples]\n",
    "\n",
    "def find_descendents(model,node):\n",
    "    n_samples = len(model.labels_)\n",
    "    global desc\n",
    "    if node in desc:\n",
    "        return desc[node]\n",
    "    if node < n_samples:\n",
    "        return [node]\n",
    "    pair = model.children_[node-n_samples]\n",
    "    desc[node] = find_descendents(model,pair[0])+find_descendents(model,pair[1])\n",
    "    return desc[node]\n",
    "\n",
    "def get_ranking(model, target):\n",
    "    rank = np.zeros(len(model.labels_))\n",
    "    to_root = [find_descendents(model, cl) for cl in find_ancestors(model, target)]\n",
    "    to_rank = [list(set(to_root[i+1]) - set(to_root[i])) for i in range(len(to_root)-1)]\n",
    "    for i in range(1,len(to_rank)+1):\n",
    "        rank[to_rank[i-1]] = i\n",
    "    return rank"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Dendrogram"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "def plot_dendrogram(model, rescale = False, size = (10,10), **kwargs):\n",
    "    # Create linkage matrix and then plot the dendrogram\n",
    "\n",
    "    # create the counts of samples under each node\n",
    "    counts = np.zeros(model.children_.shape[0])\n",
    "    n_samples = len(model.labels_)\n",
    "    for i, merge in enumerate(model.children_):\n",
    "        current_count = 0\n",
    "        for child_idx in merge:\n",
    "            if child_idx < n_samples:\n",
    "                current_count += 1  # leaf node\n",
    "            else:\n",
    "                current_count += counts[child_idx - n_samples]\n",
    "        counts[i] = current_count\n",
    "    \n",
    "    if rescale == True:\n",
    "        d_max = np.max(model.distances_)\n",
    "        d_min = np.min(model.distances_)\n",
    "        distances = (model.distances_ - d_min) / (d_max - d_min)\n",
    "    else:\n",
    "        distances = model.distances_\n",
    "\n",
    "    linkage_matrix = np.column_stack(\n",
    "        [model.children_, distances, counts]\n",
    "    ).astype(float)\n",
    "\n",
    "    # Plot the corresponding dendrogram\n",
    "    fig = plt.figure(figsize = size)\n",
    "    dendrogram(linkage_matrix, **kwargs)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def linkage_matrix(model, rescale = False):\n",
    "    # Create linkage matrix and then plot the dendrogram\n",
    "\n",
    "    # create the counts of samples under each node\n",
    "    counts = np.zeros(model.children_.shape[0])\n",
    "    n_samples = len(model.labels_)\n",
    "    for i, merge in enumerate(model.children_):\n",
    "        current_count = 0\n",
    "        for child_idx in merge:\n",
    "            if child_idx < n_samples:\n",
    "                current_count += 1  # leaf node\n",
    "            else:\n",
    "                current_count += counts[child_idx - n_samples]\n",
    "        counts[i] = current_count\n",
    "    \n",
    "    if rescale == True:\n",
    "        d_max = np.max(model.distances_)\n",
    "        d_min = np.min(model.distances_)\n",
    "        distances = (model.distances_ - d_min) / (d_max - d_min)\n",
    "    else:\n",
    "        distances = model.distances_\n",
    "\n",
    "    linkage_matrix = np.column_stack(\n",
    "        [model.children_, distances, counts]\n",
    "    ).astype(float)\n",
    "\n",
    "    return linkage_matrix\n",
    "\n",
    "def NormalizeData(data):\n",
    "    return (data - np.min(data)) / (np.max(data) - np.min(data))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Data load and preprocessing"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "The SCANPY version of this data can be downloaded here: https://kleintools.hms.harvard.edu/paper_websites/wagner_zebrafish_timecourse2018/mainpage.html"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "# load and sample data\n",
    "adata = sc.read_h5ad('WagnerScience2018.h5ad')\n",
    "sc.pp.subsample(adata, fraction = 0.08)\n",
    "adata = adata[~adata.obs['ClusterName'].isin(['NaN']),:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "labels = list(adata.obs['TissueName'])\n",
    "labels_id = list(adata.obs['TissueID'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "cats = np.array([cl.split('-',1)[1] for cl in adata.obs['ClusterName']])\n",
    "uni_cats = np.unique(cats)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "tissues = np.array(adata.obs['TissueName'])\n",
    "uni_tissues =  np.unique(np.array([adata.obs[adata.obs['ClusterName'].str.contains(c)]['TissueName'][0] for c in uni_cats]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "<ipython-input-15-627c8036739a>:1: ImplicitModificationWarning: Trying to modify attribute `.obs` of view, initializing view as actual.\n",
      "  adata.obs['Cat1'] = tissues\n"
     ]
    }
   ],
   "source": [
    "adata.obs['Cat1'] = tissues \n",
    "adata.obs['Cat2'] = cats"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "n = len(adata.obs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [],
   "source": [
    "# true ranking \n",
    "id_2cats = {i:list(adata.obs[['Cat1','Cat2']].iloc[i]) for i in range(n)}\n",
    "pairs = np.array(list(itertools.combinations(list(range(n)), 2)))\n",
    "n_inter =  [len(list(set(id_2cats[pairs[i][0]]) & set(id_2cats[pairs[i][1]]))) for i in range(pairs.shape[0])]\n",
    "\n",
    "upper = np.zeros((n, n))\n",
    "upper[np.triu_indices(n, 1)] = np.max(n_inter) - n_inter\n",
    "true_ranking = upper + upper.T"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "filtered out 1381 genes that are detected in less than 3 cells\n"
     ]
    }
   ],
   "source": [
    "sc.pp.filter_cells(adata, min_genes=200)\n",
    "sc.pp.filter_genes(adata, min_cells=3)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "normalizing by total count per cell\n",
      "    finished (0:00:00): normalized adata.X and added    'n_counts', counts per cell before normalization (adata.obs)\n"
     ]
    }
   ],
   "source": [
    "sc.pp.normalize_per_cell(adata, counts_per_cell_after=1e4)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "extracting highly variable genes\n",
      "    finished (0:00:00)\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAgoAAAErCAYAAAC/2atbAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8qNh9FAAAACXBIWXMAAAsTAAALEwEAmpwYAACaL0lEQVR4nOyde3wU1dn4vyckgWxCCCThFky4RAIiSpSLAQzQRsBoo1jfFrwUY1trK7UtLfRqA7X1bUmL7Vts+9O2qfWCtlY0ahSMAhFEhBoVkEQjgcg9CQmXLCG38/tjdobZze5mkuxmd5Pz/Xzmk52ZMzPPTvaceeY5z0VIKVEoFAqFQqFwR1igBVAoFAqFQhG8KEVBoVAoFAqFR5SioFAoFAqFwiNKUVAoFAqFQuERpSgoFAqFQqHwiFIUFAqFQqFQeEQpCgqFQqFQKDyiFAWFQhFwhBBhQghboOVQKBTtUYqCQqEICEKI64UQ/xRCHAKagLNCiAYhxFtCiJ8KIUYGWkaFQgFCZWZUKBQ9iRBiEfAbYCBQBLwLHAXOA0OAy4FrgQzgH8ADUsrqgAirUCiUoqBQKHoWIcRO4EGgSErZ5qVdEvAd4ISU8nc9JZ9CoXBGKQoKhUKhUCg8onwUFAqFQqFQeCQ80AIoFIq+hRDi71bbSinv9qcsCoWiY5SioFAoeppEl/VMoA3Y41i/HM3aWdKTQikUCvcoRUGhUPQoUsov6J+FED9Gi3bIlVI2OLZFA3/jouKgUCgCiHJmVCgUAUMIcQz4vJTyI5ftk4A3pJTDAyOZQqHQUc6MCoUikMQA7hIrjQBUpkaFIghQioJCoQgk/wEKhBCLhRCjHctitKmH5wMsm0KhQE09KBSKACKEiAJ+B9wNRDg2t6ApCj+QUtoDJZtCodBQioJCoQg4DgfGcY7VT3XHRoVCEXjU1INCoQgGohxLuVISFIrgosPwSCFEGDAXmAOMRuvM1cB7wCYp5Wd+lE+hUPRihBADgb8DXwQkcClwQAjxF+C4lHJVAMVTKBR4sSgIIaKEED8FPgNeAeajeSg3AWOAPKBSCFEkhLimJ4RVKBS9jt+gRT1chZZPQedlYFFAJFIoFE54syh8AuwA7kGzHDS7NhBCpAC3Ac8KIX4ppXzMP2IqFIpeSg6wSEr5vhDC7DC1HxgbIJkUCoUJbz4KC6WU/yOlfMWdkgAgpTwkpfxfNHPhls5cWAiRKYQoFEIcEUJIIcRdpn0RQojfCCE+FEI0CCGOCSGeFkIkd3DOuY5zuS4TOiObQqHoMQYDtW62DwRae1gWhULhBo8WBSnlXqsnkVI2oVkgOkMMsBf4p2MxY0MzRf4KeB8YhBZC9ZoQ4gopZUsH554EnDKtV3ckTEJCghw9erTXNhcuXKB///4dnapPou6NZ0Lx3vz3v/+tkVK61mTwB7vQrAq/d6zrVoVvAG/7++Kq33cPdW88E2r3xluf96goCCEyrV5AStnp4i1SyiKgyHGtf7jsOw1c5yLPN4B9wEQ6zgF/UkpZ0xl5Ro8eze7du7222b9/PxMnTuzMafsM6t54JhTvjRDiUA9d6ifARkfK5nBguePzdLRiUX5F9fvuoe6NZ0Lt3njr8958FLagaffCsa5r+q7rAP26KlwniHX8rbPQdrcQoj/wEfBLKeVm/4mlUCi6ipTybSHETOAHwKfA59EiqjKklKoolEIRBHhTFMwmiBnAb9GmAnY4tmWgvQ2s9I9oFxFCRKJNPbwkpTzspekx4Jto5sxI4E7gDSHEHCnlW/6WU6FQdB6HQrA00HIoFAr3ePNRMByMhBAPAt+RUr5uanJACHESWIMWPukXhBDhwJNAHNpcpkeklOVAuWnTDiHEaGAF0E5REELcgxbVwciRI9m/f79XWRobGzts01dR98Yz6t54RgjxJrBZSvmgy/bBwH+klJ8LjGQKhUKnw4RLDi4D3L3JHwH8FlHgUBLWA5OBuWblpRPsBBa72yGlfBR4FGDq1Kmyo/mkUJtz6knUvfGMujdemQtMEUJcAXxFSqnnUohES/KmUCgCjFVFYR+QJ4TI1Tuyo5jLzx37fI4QIgJ4BrgcTUk43sVTTUGbkgg6zpw5w8mTJ2ludht9GlI0Nzert2YPBOO9iY6OZtSoUYSFBUUW9yw0hX2bEOJGKWVQ9leFoq9iVVH4JlqmtCNCiA8d2yajxTnf0JULCyFigFTHahiQLISYghbWeBT4NzAN+AIghRDDHW1Pm5SVfwJIKb/iWP8ucBBNeYkE7gBuRksPG1ScOXOGEydOkJSURFRUFEKIjg8KYs6fP09UVFSgxQhKgu3etLW1ceTIEWpqahg6dGigxQHNWjkbeBzNETkH9xZMhUIRACy9Tkgpd6FlSfsRmkfye47PYxz7usJUoNSxRAGrHZ9/AYwCbkJL7fpfNIuAvnzZdI5kx6ITCeQDH6L5JMwGbpBSBl1d+5MnT5KUlITNZgt5JUERWoSFhTFs2DBOnz4daFHAET0lpWyUUn4ZzbKwBfhSIIVSKBQXsWpRwFHR7VFfXVhKuYWLoZbu6PDpKaWc67K+Bs25ssepqamhoKCA3NxcEhISOmzf3NwcVG+Zir5FREQELS0d5S3rEZz6uZRytRBiH/CPwIij6A3Y7XZKS0tJT0/HZrMFWpyQx/IEpRDieiHEy0KIj4QQlzi2fU0I8Xn/iRc6rFu3jpUrV7Ju3TrLxyhLgiJQBNFvbx7OWVSRUj4HXAPcHRCJFCFPaWkpxcXFlJaWBlqUXoElRUEIcTvwL7Q0zWOACMeufvRAHgWFZ0aPHs0zzzzjcf9DDz3EokXWi/DNnTuXX//61x73/+Mf/2DChOApnfHWW29ZsuDodHS/7rrrLu69915fiKawgJRyq7uU7FLKvVLKxwMhkyL0SU9PJysri/T09ECL0iuwOvWwEvi6lPIZIcTXTNvfQfMp6PMsW7aM6OhocnNzAy2KEz/5yU8CLYJfufbaa6mp6VS2bkWAEUIUAndIKc84PntESuk1d4pC4Q6bzcasWbO6dKyatmiPVUXhUi5mZDRzjouplfs0CQkJrFixItBi9Cmam5uJiIjouKEi2KjlYgr4ruRGUSj8hj5tAXRZ2ehtWPVROAqMd7M9Ey0/uyKAHDx4kHnz5hETE8MVV1zBO++8Y+xbtWoVCxcuNNbLysrIzMxk4MCBpKen8/DDDzNgwACn8506dYpFixYxcOBAUlNTeemll9xe989//jNXX32107Y9e/YQFRXFqVOn2rW/9dZb+cEPfuC0bd26dUybNg2ADz74gDlz5hAfH8+QIUO44YYbqKysNNreddddfOUrX+HOO+9k8ODB/OhHP2LLli1O8j/zzDNceeWVxMbGMnLkSL71rW9ht9udrvnJJ5+QkZFBTEwM06dP57///a/b7wdQVVXFrbfeyvDhw43zNTQ0eGzv7v4OHjzY2N/S0sJDDz3E+PHjiYuLIzMzkw8++MDYP3fuXFauXOn1/r/wwgtcffXVxMXFcdlll/Gvf/3L2FdaWsqsWbMYNGgQ8fHxzJo1i3PnznmUNxBIKXOllGdNnz0ugZZV0fdQ0xZukFJ2uKBNPewHZgFn0TKmLUUr33yflXME+3L11VfLjvjoo486bGMVX50rJSVFXnrppfKjjz6SLS0t8rvf/a6cMGGCsT8vL08uWLBASillc3OzTE1Nld/73vfk+fPn5SeffCInTpwo+/fvb7SfM2eOTEhIkDt27JCtra1y7dq1cvDgwfL8+fNSSikLCgpkWlqalFLK06dPS5vNJt9//31pt9ullFJ+5zvfkUuWLHEr68svvyyHDx8um5ubjW1XX321fOSRR6SUUn7wwQdy8+bN8sKFC7Kurk4uWrRIZmZmGm2XLl0qIyMj5fPPPy9bWlpkQ0OD3Lx5s5P8r776qty3b59sbW2VH3/8sRw/frz8+c9/7nS/kpKS5HvvvScvXLggV61aJYcNGybPnTtnXOMb3/iGlFLK8+fPy3HjxslVq1ZJu90ua2tr5YIFC+Q3v/lNt9/Pyv39yU9+Iq+55hp54MAB2dLSIv/yl7/I4cOHyzNnzli6/5s2bZIJCQly27ZtsrW1Ve7YsUMOGjRIvvPOO1JKKTMyMuQvfvEL2dLSIpuamuT27duNY13x9hsEdssg6Jf+Xnq63/c21L3xTKjdG2993nKHQisIZQfaHMt54EGrxwf7EsqKwtq1a431vXv3SsB48JkVhbfeektGRUU5PTj+8pe/tFMU7r//fmP93LlzEpB79+6VUjorClJKmZubK7/97W9Lu90uL1y4IOPj4+Ubb7zhVtaWlhY5fPhw+dJLL0kppdyzZ4/s37+/PHXqlNv2paWlsl+/frKpqUlKqT3E58+f79TGVVFw5eGHH5bXXnutsZ6SkiJXrVplrLe2tsrhw4fL5557zriGrij8+9//lpdeeqnT+d555x0ZHR3t9lod3d+2tjYZExMjt2/f7nTchAkT5L///W8pZcf3/4YbbpC/+tWvnI6/99575X333Wcc//Wvf10ePHjQ4z3RCZSiABRaXfwlg74oRaF7qHvjmVC7N976vOXwSCnlT4EEtDrx1wCJUsoHumTGUHilpqaG/Px8y056I0aMMD5HR0cDcPbs2Xbtjhw5wrBhw5xM9SkpKV0+H8DXv/51nnrqKS5cuMALL7xAXFwc8+bNc9u2X79+3HnnnfzjH/8AtAiKm266yTDNf/rpp9xyyy0kJSURGxtLZmYmra2tTtMY7uQ18/rrr3PttdeSmJhIbGwsP/nJT6iurnZqM3r0aONzWFgYKSkpHDlypN25KisrqaysJC4uzljmz59PS0uL2/9NR/e3pqaGc+fOcf311zud89ChQ3z22WdGO2/3v7Kykl/96ldOxz/xxBOG/AUFBbS0tDBz5kzGjh3L6tWraWtr83rPAkBtJ5Y+RWf7vkLRE1hyZhRCfAXYJaXcD+w2bR8AfElK+U8/ydcnKSgoYOVKLerUlw6SSUlJnDhxggsXLtC/f39Am4PvDhkZGYwcOZLCwkKeeuop7r77bq8x+nfddRfp6emcPHmSp556ir///e/GvnvvvZdLLrmEPXv2MGTIEN5//33S09P1t1AAr7UJmpqauPnmm/nd737H0qVLiYqK4ve//z1/+ctfnNodPHjQ+NzW1kZVVRVJSUntzpeSksJll13m5EPgjY7ub0JCAtHR0WzZsqXL858pKSl87Wtf43vf+57b/WPGjDHu6Ycffsj8+fO59NJLue2227p0PX8gle+BR/zV9xXWUBEP7rFqUfgH8K4jB7uZQUCBTyVSkJuby5o1a3weajljxgxGjhzJT3/6Uy5cuMCnn37KH/7wh26f9+tf/zq//e1v2bJlC3fddZfXtpdddhlTpkzhrrvuIiwsjPnz5xv7zpw5Q0xMDIMGDaK6upq8vLxOydHU1MSFCxcYMmQIUVFR7N27lz/96U/t2v31r3/lgw8+oKmpiYceeojW1lYWLFjQrt0NN9xAQ0MDv/nNbzh37hxSSj777DNefPFFt9fv6P4KIVi2bBnf//73+fRTzQf47NmzvPrqq5w4ccLSd7z//vv53e9+x/bt22ltbaWpqYldu3YZiWUef/xxjh3TaioNGjSIfv360a9fP0vnVgSenJwcsrOzyclRUaGBQCVqck9nSsc9BDwrhPihv4RRaOihlp1JJGSFiIgICgsLeeedd0hISODWW29l6dKlxttvV7nzzjv55JNPWLBgASNHjuywfW5uLq+++ip33nmn00Ps4YcfZvPmzcTGxjJnzhy+8IUvdEqOmJgY/vznP7N8+XIGDhzI/fff7/ZN+p577uHee+9lyJAhvPDCC7z88svExMS0axcdHc2bb77JBx98QFpaGnFxcSxYsIC9e/e6vb6V+/vLX/6S66+/nhtvvJHY2FjS0tL461//avk7Lly4kD/96U8sX76chIQERo4cycqVK43IjjfeeIOrrrqKmJgYZs+eTW5uLl/6UnCXTRBCzBNCPCqEeE0I8aZ5CbRsPU1hYSFFRUUUFnpNL6HwEyriwQOenBfMC5rz4lDgWuAEmoUhAhgGtFo5R7AvoerM2F3WrVsnJ0+e3K1ztLa2ypEjR8oXX3zRR1L1HtatWycvv/zyQIvhlmCIegDuAhqB9Y6/z6IVh6sH1vn7+t3t99XV1XLNmjWyurq6w/NYwdfn8zfBMo4FI6F2b7z1+U4Vo5dSvgXMANKBN4DEbmkpih7nrbfe4sCBA0gpef/998nPz2fJkiXdOmdBQQGRkZHccEOXKo73Ktzd3//5n/8JtFjBzA+AZVLKJUAz8GMpZTrwJFpCt6CmKzVevOEva6JC0R0sV4/UkVIeFELMBJ5GUxYUIcShQ4e47bbbqK2tZejQoSxZsoTvf//7XT7f4MGDsdlsPPbYY2ouHPf39zvf+U6gxQpmxgLFjs8XAH0OaB1auekfBUAmhUJhwqqi8Dha3gRAKzkthLgZWIWWnVERItxxxx3ccccdPjtfXV0dAOfPn++gZd/A3f1V98YrtcBAx+cjwOXAh0A8EPR12IO1xovCPSqqoWtYmnqQppSrpm1SSpknpXQfNN/HUPHPCkWXeAvQQ1/+BfyfEKIAzWfh9YBJZRE1VRBc2O12tm/f3i5tu46KaugaHi0KQohbgJeklM2Oz56QUsoNvhcteKmpqaGgoIDc3FxjgFDxzwpFl1gG6Bmq/hdoQUsV/y/gl4ESShGadFTQSY9mUFENncPb1MNzwHDgpOOzJyTQpyan3SkFuulRmSAVCutIKU+ZPrcBvwmgOIoQpyNFoDvlp/syHhUFKWWYu88K90qBKjOtUHQdIcQQtBBsp7FGSvlRYCTyH+4skgrfYFYElD+C7+h01INCKQUKha8QQqSjZXedrG9Cs1Lqf3uVtbKmpoalS5dSVFQEqGlKf9LRNITCOh35KFhCSvm8b8RRKBR9jL+jRTt8By2Zm/Te3D1CiBHAr4FstCiKA8A3pZRbfSSnTygoKKCoqIjs7Gw1TelnlD+C7+jIR8EKvU7r7+2MHj2aX//61yxevDjQoigUlwL/I6Ws6OoJhBBxwHZgG3ADUI2Wn+GkLwT0JbpykJOTo6Yf/IzyR/AdHn0PpJRhFhelJAQxo0eP5plnngm0GAqFJ7YBE7t5jpXAMSnlV6SU70opK6WUb0it2m1QoU9bFhYWsnLlSgoKVE09RfATMCdFIUSmEKJQCHFECCGFEHe57BdCiFVCiKNCiPNCiC1CiEkWzvtFIcRHQogLjr+L/PYlFAA0NzcHWgRF6PJV4FtCiO8IIT7vGBeMxeI5bgZ2CiGeFUKcFEK8L4RYJrzVOw8w/qoQq1D4A8uKghAiXAgxUwixWAjxFfPSxWvHAHvR5ibdpa5bCXwf+DYwDc2M+LoQYqCbtrqMGWhFZZ4Cpjj+/lsIMaOLMgY9tbW13HnnnQwfPpwRI0aQm5trZEtctGgRVVVV3HXXXcTExDhVYzx48CDz5s0jJiaGK664gnfeecfpvI899hiTJk1i0KBBXH311bz55sVCfnfddRdf+cpXuPPOOxk8eDA/+pH7LLuFhYVMnDiRmJgYcnJyuP/++1m4cKGT7HfffTejRo0y0h2bE1YJIfjLX/7C1KlTiYmJISMjg48//tjY39LSwkMPPcT48eOJi4sjMzOTDz74wNj/zDPPMGHCBAYOHMiwYcO45557uniXFX7kUrTaMQ+jJVjaYlo2WzzHWOBbaH4JC4A/oPkr3OdTSX1Ib0zUpJLO9V6EVjSqg0ZCTABeAsageSO3ovk3NAMXpJSx3RJCiHNohWH+4VgXwFG06nG/cmyLQlMWfiCl/H8ezvMsMERKeZ1pWzFQ7Sg645GpU6fK3bt3e5Vz//79TJzYXSupb8+1cOFCIiMjefzxx5FScttttzFgwABeeOEFwL0/wujRo4mMjOTFF19k/Pjx/OAHP+C1115j/37NUvvYY4+Rn5/Pf/7zHyZNmsQrr7zC7bffzp49e0hJSeGuu+5i/fr1PPPMM+Tk5HDhwgWEEERFXcy4W1FRwWWXXcaTTz7JLbfcwubNm1m0aBGzZ8/mtddeQ0pJZmYml112Gfn5+YSHh/Otb32L6upqXnnlFUBTFK655hr+9a9/MXToUO644w7OnTvHq6++CsBPf/pT3nzzTZ5++mmSk5P561//yqpVq/j444/p168fgwYN4vXXX2fu3LmcO3eO999/n9mzZ3f7nneW8+fPO92bYMHbb1AI8V8p5VR/yyCEKAd2oSVbaufMKKWstXCOJrTKdzNN2x4CFkkp231BIcQ9wD0AI0eOvFr3jPdEY2MjAwYMcLuvpaWFmpoaEhISCA/ve0Fk5ntz/PhxDh8+zMiRIwkLC+uz90TH2+8mGLnssss893lPZSXNC/Aa8AwQDZwFxgFXATuB66yco4PznwPuMq2PRRswprm0ewV43Mt5qoAVLttWAIc6kiGYykw3NDTIbdu2yYaGBq/nOHLkiATkp59+amzbu3evBGRtba2UUsqUlBS5fv16p+NSUlLk2rVr2x1z7tw5KaWUkyZNkk899ZTTMQsXLpT5+flSSimXLl0q58+f77Tfbrc7rT/44INy3rx5TtsWL14sFyxYIKWUcteuXTI6OlpeuHDB2H/8+HEJGCV2Afn8888b+19++WWZkJAgpZSyra1NxsTEyO3btztdY8KECfLf//63bGhokFFRUfJPf/qTPHXqVPub14O43ptgIUjKTDcA47p5jkPAX1223Qk0dHRsd/v9mjVrJGCUhg6lEtG+wHxv9O+fl5dn3JO+TG8qM21V3ZsGzJFaMag2IFxK+Z4QYiXwR+AKi+exynDH3xMu208ASR0c5+6Y4W7aBi1W438/++wzhBCMHj3a2DZu3DgADh8+zJAhQzweO2LECONzdHQ0AGfPniU6OprKykq+8Y1v8K1vfcto09zcTFpamrGekpLi9TscOXKkXZuUlBRjWqSyspLz588zdOhQpzb9+/fn0KFDhknWVc6zZ7WSIzU1NZw7d47rr78e81R0U1MTn332GTabjaKiItauXctPfvITxo0bxw9/+ENV8jn4eB24Gvi0G+fYDqS5bBuPpkD4DHeJkszJ1/p6Gnd9OqWmpkYVyuplWFUUBKBX2ahGe1iXA4eBVD/I1SO4mCAN07snGhsbO2xjlebmZo9VBSdMmEBzczMTJkzwWnkwISEBKSUff/yx8VDW5YuPj+f8+fMIIWhqanI6j5TSaVtjYyOgmcjPnz9PcnIyeXl53HTTTe2uef78eVpbW2lra2t3TvP60KFD2bp1q9O2yspKWltbOX/+PMOGDSM2NpYjR47gzudMP+7ChQtOn/V90dHRREdH89prrzFlyhS3x8+YMYNnn32W1tZWXnrpJRYvXsyUKVMYNWqUx3vqD1zvTbDQ3Nzss99zN3gN+J0Q4gpgD9p0poG0lqPlYeBtIcRP0XyU0oH7gZ/4UlB3ioA5+ZpK466hEtL1QjyZGqSzGa8Ebb4P4GlgIzAHeBL40Mo5Oji/mnroIvPnz5e33HKLrK+vl7W1tfL666+XOTk5xv6MjAz5i1/8wukY1+mIyspKCchjx45JKaX885//LCdPnizff/992dbWJu12u9yyZYv8+OOPpZTa1MM3vvENp3O6mtfLy8tlRESEfO6552RLS4ssLi6WMTExxtRDa2urnDlzprz//vuNaZITJ07IZ5991jgHIHfs2GGsb968Wfbv399Y/+EPfyjnzZsnKyoqpJRSnjlzRhYVFcnjx4/L48ePy+eee06ePn1aSinl66+/LsPCwuTRo0c7c3t9gpp68Nr327wsrZ04zw3AB0Aj8DGaoiA6Oq4z/b4vTi10hL/M673hXvemqQerUQ+/QrMqAPwMSEbzSJ7v6JC+phI4DpidEgcA1wJvezluh/kYB9d1cIzPCITX75NPPsmAAQMYP348l112GYmJiU6x2T/72c/4xz/+QVxcHDfffLOlc957771897vfZenSpcTFxZGSksKvf/1rWlpaLMs1fvx4nn32WX76058yaNAgfv/733PHHXfQv39/AMLCwigsLKSpqYmrrrqK2NhYZs6cybZt2yxf45e//CXXX389N954I7GxsaSlpfHXv/4VgLa2Nh555BFSUlIYOHAg3/3ud3nqqaecpjIUQcFAIEJ2M0eLlPIVKeWVUsoBUsrxUsr/cwx+PqM3RioEK7r1RuWZCBI8aRAdLcAQLGjsXo6PQQthnII2rfFzx+dkx/4fAqeBW4DL0ZwpjwIDTed4A/hf0/pMtDK1PwImAD9GM2XO6EgeX1gUzI5N3T1XqGHlrfnWW2+V3/72t3tAmuBCWRQ8jgH9HP3zMn9fy9PS05bE3oayKHgm1H433vp8l2NXpKk8bBeZinOc9GrH8jhwF7AGiAIeAQajRVjMl1KeNR0zDvjMJNPbQojFaHXsf4HmIPVlKeXObspqCTVH6UxhYSHXXnstAwcO5KWXXuLFF19k69agSr2vCCBSylYhxCEgMtCy+BNVLbLzKD+H4MKSoiCE6I+W0GQe7kvBTu/shaWUW7g4neFuvwRWORZPbUa72fYc1utUdAm94+fk5FBYWGgoBmowcGbz5s3k5uZy4cIFUlJS+Nvf/kZGRkagxVIEFw8CvxZC3CGl7JWZevp6NIQi9LFqUXgMuBF4EfiILlZ46y3oHX/Lli1GuVhADQYuPPzwwzz88MOBFkMR3PwALZHbESHEYbS8CgZSSl+HXvc4ytKoCHWsKgo5wE0yyEq2BgpzBbi5c+c6DQBqMFAoOoVfrX/BgDKjK0Idq4rCSaBXmgW7grnjmweAzg4GUkq3OQQUCn8jfRsQ0GWklKsDLYNCofCO1fDInwAPCSEG+1OYUKUrYZERERFBmYRH0Tdobm4Oqjz8QojPOSo+3ieEmBtoeYIFVWhJEQxYVRQ2ATbgpBDiMyHEAfPiR/lCgq7E/A4dOpQjR45gt9uD5u1O0Tdoa2vjxIkTDBo0KNCiIIRIEkK8i5bK+Ydooc1vCCF2CiFGBla6wNMb8wko5Sf0sPpK8U/gMuD3uKnw1tfJyclhy5Yt5OTkWD4mNlYruHn06FGam5s7aB38NDc3ExEREWgxgpJgvDfR0dHBEp3zf2jVaFOllJUAQoixaFlf/w+4NYCyBZze6AipokBCD6uKwnXA53oqH0GoUVhYSFFREQcOHOCFF14gPj7eUqhkbGysoTCEOr4swd3bUPfGK9cBc3UlAUBKeUAIcT9aQrWgp6amhnXr1gGwbNkynypgvdERsjcqP70dq4pCFXDBn4KEEuY8CuvXr8dut5OamkpZWRnLly9n7ty5SmNWKKzjzkIZMlbLdevWsXq15pMZHR2t+nwH9Eblp7djVVH4HrBGCPEtKWWFPwUKBdzlUVixYgX79u1j7dq1xMfHA0pjVigs8AbwRyHEEinlZwBCiGS0ac6gtijoLwx2u1ZYNysry1KfV5kaFaGGVUXh30B/oFwIcQGtnoKBlLJ32M8tYs6jMG3aNKC9yVFpzAqFJe4HCoEDQoijjm0j0UpOLwmYVG5wzcja0NDA6tWrycvLY82aNZYf/J2do1eKhSLQWFUUlvlVihBDN53V1NQQHR2tOrBC0UWklJ8JIa4CstAKuQHsl1IWB1Ast7haEnUFQZ+CXLdunSUfhc7O0SvnP0Wg6VBREEJEANnAT6SUn/pfpNBBvRkoFN3HUdfldccStLjLyJqQkEB+fn6nfBTczdF7GxuU858i0HSoKEgpm4UQ89FKNitMqDcDhaL7CCFmAJ/HfcG5+wMilBtcM7Lq+QBycnJoaNBKVHgbC7wpA97Ghs44/5WXl7N8+XLWrl1LWlqa5e+mUHjD6tTD88AtwG/9KEvI0VnvXfVmoFA4I4T4AVpJ+QrgKM7RDkEd+WB+uK9atapT7V3HDV+NDcuXLzccrF955ZVunUuh0OlMeOTPhBDXArtpX+Ftra8FC3bMbwdgrcS0CgtSKNrxHeB+KeW6QAvSWTr7cPfW3ldjw9q1a53+KhS+wKqicBdQB1zhWMxIoE/8KltaWsjPzyc3N9fp7QBUiWmFoovEAkUdtgpSqqurWbJkCevWrevQ1J+QkGCMHf7wU6qpqaGwsJDHH39c+UApfIolRUFKOcbfgoQCNTU1hkLg7u1ATSkoFJ1mPbAQ+FOgBekI1wyMBQUF5OfnA5rJ34qpX3/BaGho8HnElPKBUviLTpePE0LEoDkqN3TYuJeRkJDgFC/dnRLTCoUCgM+A1UKIWcCHgFPhk2Ca1iwoKDCiG3Tuu+8+ysvLLZv69ZeJhoYGnz/UlQ+Uwl9YVhSEEPehVXdLcqwfBn4jpQz6NwFfER4erhQChcK3fA04B8x0LGaCalozNzfXiG4AWL16NWvWrDGsDJ1hyZIlhkXBVygfKIW/sKQoCCF+ghYe+Vtgm2PztcCvhRCxUspf+0m+kKArjo0KhSK0pjUTEhJYtWqVMQWRl5dn+UGvjxF6NkdQVkhF6BDWcRMA7gXukVKullK+4VhWAd90LH0ac834devWsXLlyi69ZSgUiuBHn4LoTKlus/+APn2pUIQKVqcehgK73Gx/FxjmO3FCh/LycpYtW0Z6ejpf/epXAc00qRQEhaJ30xVfAPMxwWxp7C3ZY+12O6WlpaSnp2Oz2QItTshjVVH4GLgN+IXL9tuAcp9KFCIsX76c4uJiiouLSUxMNMyIS5YsYdeuXSxZ0nE9m97SKRWKvkJX+2yo+A/0lsiJ0tJSiou1ciGzZs0KsDShj1VFYRXwLyFEJrDdsW0WMAf4Hz/IhRDiIJDiZleRlPIGD8e4y+T2TSnlX3wpG2gJTZqamkhPT3d6s1i/fj1FRUVMmzatw2xteh37hoYGS5ndFApFYPFneGMw0FsiJ9LT053+KrqHJR8FKeXzwAzgOHCjYzkOTJdSvuAn2aYBI0zLVWhe0P/q4Livuxz3uD+ES0tL4/XXX2fNmjUA5OfnU1NT06Vzbd++vd2xeh75rp5ToVD4Bj3RWk1NDbm5uUaf1/2SrBAq/Vm3fIS68mOz2Zg1a5aadvARlsMjpZT/Be7woyyu16s2rwshvgqcoWNFoV5KedxvgrnBbK5btmyZ5bCnZcuWsWvXLoqKiigoKHAy9fUWE6BC4Q0hRCswQkp50mV7PHBSStkvMJJdxJxobcWKFe1KzFs5funSpUYNBtWfFaGG1agHAIQQI4UQU4QQV5kXfwlnuq4Avgo8KaU830HzPwghaoQQu4QQ9wohOvUdO8L1zaCmpoaGhgYjVKozGnlCQgKPP/64Wy/onJwcsrOzycnJ8aX4CkWwITxs7w809aQgnjAnWjNvs9rPCwoKKCoqIjs7O+RN+n0Vu93O9u3bsdvtgRYlIFjNo5AOPAlMoH3HloC/tf7rgDHAYx20+zmwGS2By+eB3wEJwC/dNRZC3APcAzBy5Ej279/v9eSNjY3s2bOH4cOH88EHHxAWFkZbWxvjxo1j1KhRVFdXU11d7fUc7rjxxhvbHXv8+HEWL17M0aNHaWtr6/Q5e5rGxsYO719fRd2b9gghljs+SuBeIcQ50+5+aHlaynpcMDd0N9FaqEQ8BIJQceju686RVqceHkVLtfp12peC7Qm+DuySUn7grZGU8kHT6vtCiH7AT/GgKEgpH0X7bkydOlVOnDjRqxD79+9n8uTJ7N69mz179pCfn8+cOXOYO3cu06dPp7Cw0Gc/+MTERHbv3s31118f1B1IZ//+/XR0//oq6t645duOvwItO2OraV8TcBAtf0vQYvUhFyoRD2b075aTk+PTcc2VUJli7evOkVYVhcuAdCnlx/4Uxh1CiKHATcB9XTh8JxArhBgmpTzhW8k0tm7dSnR0NIBPM66F4uCiUFhFz8gohNgM3CKlrAuwSB3i+vDszVkW9Qf4li1b/OpbESpRFrpzZF/FqqKwBxiOlk+hp7kLuIBWZa6zTAEagXpfCaN3oMzMTObMmQNghEOqjGsKReeQUs7TPwd7wTm97//973+nrKyMvLy8Xtvn9e+Uk5PD3Llz/fYd1QtRaGBVUfgJsEYI8TM0pcG1wtspXwsGhhPj14BnpJTnXPYtA5ZJKSc41r+ApszsAM4D89ASRD0qpbzgK5lyc3OdtOy8vDxuuOGGoJ9jUyiClVApOGfu+9nZ2SxbtqzbfT5Y5+jND3D1IFdYVRSKHX834eyfIPCvM+Nc4FLch2UmAGmm9WbgW2jV5sKAA2jOjY/4UqCEhATWrl3LuXPnEEKwZMkS0tLSOj5QoVC0I5QKzulRSuYCcPn5+d16yIfKHL2ib2NVUZjXcRPfI6XcjIfwKUdRqlWm9deA13pCrvXr11NSUgJAYWGhEVcdjG8GCkWQoxecM08tviGE+AR4CAgaRQGc37Tz8/MtP+Q9jQ+hMkev6NtYUhSklFv9LUioUFNTw/btWhbrrKwso4OrdMwKRZcIiYJz7h70nXnIexof1By9IhTwqCgIIcZIKSutnMThSzBKSvmZzyQLUgoKCiguLiY7O5vHH39cWQ8Uiu4REgXnXGs8+DtsUKEIJrxZFHYIIV4B/iql3OGugRBiMLAYuB/NF6DX11h2TZ6il5tOS0sjLy+PZcuWeT1eTVEoFE6soocLznUFvd9XV1ezevVqHn30USoqKiwVh9KzuFoZHzzRU3kNFAp3eEtvPAE4BbziSIm8UQhRIIT4sxDiGSHEh8BJNEfD70ope72S4A693PQjjzxCdHS0186r53zvTDEZhaI3E6CCc92moqKCCRMmYLfbO+zPBQUFrF692hgfampqWLVqFatWrbJcJEq3aCxfvrzHxg9fF7IKlcJYivZ4tChIKeuBFUKInwM3ALPRyj5HATVoVRk3Sin39oCcQUFLSwtf/OIXKSkp4ZVXXuG5557zWG7aHSrnu0LRnp4uONcV3OVP2bp1K1/+8pc7zKXg6sugKw4A0dHRTj4KHTk9+juvgRlfR2SoCI/QpUNnRkcRpuccS5+mpqbGiHbYunUra9asITExkfXr11syA6qc7wqFe4QQI9EcG52snFLK9wIjkTPdyZ/i6rCYm5tLQ0OD8dmMp4dpIPIa+DoiQ0V4hDBSSrVIydVXXy074sMPP5Rz5syRaLkjZEZGhgRkdna2rK6u7vD43sxHH30UaBGCllC8N8Bu2QP9DkgH9qHVemhzWVr9fX0r/V7//5WVlcnMzEw5Z84cWVZWZuk+VldXyzVr1lgeHzrbvjv44lqd/W335PcLNKHW7731eZ+WYO7thIeH89xzz5GamgpoFR6zsrIoKirq0pyhmrNTKIyCc9cCY9GqxOrL2ADK1Q49f8rWrVspLCwEvPfhrvgkdaZ8dXfRrRc96S8ViGsquo/VhEsKNB+FgoICnnjiCe68804qKiq49dZbmT9/fpfMaWrOTqHwfcE5IcSP0ZI1PSKl7FqYgQtdyZ+i+yRNmDCBnJwcy9fpqaioQEwFqOmH0ERZFDpBTU0NK1eu5K233uL2228HtDrlOTk5rFu3rlNezKB1FldHKGVlUPQx9IJzPkEIcQ1wD/Chr84JzvlT1q/Xkkjm5+djt9s9HpObm0t2djZlZWWGBcLKdXrijTtQYdo9aTFR+BBPcxJ9bbEyV/nee+/J7OxsuWPHDpmXlyezsrIMHwUcfgtr1qzp8DzeWLNmjU/O09OE2nxcTxKK94ae81H4HPAOkIWWiXGIeenkuQYBn6KlnN8CrOvoGKs+CmVlZTI7O1uWlZXJ6upqo89nZWXJvLw8j3PuPeWj0NnjfDXOhOJvu6cItXvjrc9bmnoQQswBGqWUOx3rd6FVddwHfF+6VHbsrZw6dYqioiL27t1LVVUVd999N9nZ2TzwwANMmzYN6NikVlNTw7p1WsoJd9XnlGlO0cfwZcG5R4HnpJSbhRB5PpKPlpYWli1bRklJCZMmTSIxMdGYUiguLmb+/Plu35DNSZKsvr13NaVzZ6cx1Tij6AxWfRR+j6MAkxAiDfh/wN/QcivkA9/0g2xBR1tbGwBVVVUAbN68mcrKSqZNm2ZMO3Q0IOhzmtA+hho8DxQqo6Oil+KTgnNCiK8DqfghH4M5LHrDhg28/PLLAE5ZEt31T/3hrYdVWsni2FU6++BXNSYUncGqopCKNpcI8EXgdSnlt4QQM4D/0EcUhbCwiy4dY8eOZdiwYVRWXiyH0ZFWb3aImjNnDg0NDdTU1FgaNJTjo6I3In1QcM7x8vIQMFtK2Wyh/T1ofgyMHDmS/fv3e20/aNAgXnvtNWpra2ltbeXAgQNcddVVHD58mIULF1JdXc3x48cZPnw4e/bsYfhwzeVi4cKFTJ06lbi4OH7wgx/Q1tbG0aNHndr4khtvvJHq6mqqq6t9fm5PNDY2dnj/+iq96d5YVRTauGgC/DywwfH5OBDva6GClSFDhpCVlUV6ejqgOTNlZWWxZMkS8vPzDc9mT1q92SFq2rRpTpaFjt4ylKlQ0VvwQ8G5DCAB2Kc1B7TxKlMIcS8QLaW8oO+QUj6KNk3B1KlT5cSJE73KsH//fj788ENDUbfZbIYT45o1a1ixYgWJiYm8+uqr/POf/2TdunWkpaUBzpZAgPfee4/rr7/er1bBnrQ+7t+/n47uX1+lN90bq4rCLuABIcTraPHO9zi2jwaO+UGuoKS+vt6Yk9Qzq3366ad84xvfYOtW7cXI29u++WFfW1vLrl27sNvthsLg7VhlKlT0InxdcO4FYLfLtgLgEzRLQ1N3Bc7NzeXQoUM888wz1NbWkpWVxaxZs4w+nZCQwL59+yguLmb58uW88sormhAulsCe6MPK+qjwNVYVhe8CTwM3Ab+SUn7q2P4/gNuO3huJi4sjOzubnJwc4uPjefbZZykrK6OyspKsrKwOpxISEhLIzc2loKCAhoYGioqKmDZtWoe54rtCeXk5y5cvZ+3atcbbjUIRJEwAfopWcK4N+C9wFGgEBqPlVpgIvItWcG6jt5NJrS5NvXmbEKIBOCV9UItGz5+yatUqvv3tb7Ns2TLS09NZsmSJk2Py2rVrAYy/oHIV6CgfqxDHUziElQUYAER05xzBslgJk3rzzTedUjaXlZXJrKwsuWLFCpmXl2cp3EgPS8rLy+t0GFRnQqD08K3s7GzL5+8OoRYK1JOE4r2hB8Ij0QrM3YrmLL0BeA14Evg+cHk3z70FH4VHFhcXS0CuWLHCKRTal2HRPYWvUyhb/W2Hath3dwi1fu+tz3cqM6MQYiowDnhZStmANg/Y0kUdJeSIi4tjwoQJRsrmFStW8PrrrwOaxqz7GrhiDolcsmQJ0LXCUFZNijU1NUyaNImmpiantxuFIpiQfiw4J6Wc66tz6dFO27ZtY8eOHcycORObzWaERdvtdqqrq1m1ahVLliwxIiGC8c1ZH0P8GYHhjmC0ciisYzWPwjDgRWA6mgZ9KXAAWItmLvyOvwQMJk6dOkVZWZlTClcdbz4E5rKyQLtUr1ax2tkKCgrIz89nzZo1atpBofAR+/btMz4XFxcza9Ysli1bxtKlS42qktu3b6e4uNjrgziQZnh97GhoaOhRPwblYxXaWLUoPAycQItwqDJt/zfwR18LFeykp6c7eTJ7S6AEWufctGkTxcXF7fZ1BqudTWnvCoXv0MOiz5w5A8CxYxf9t/V6DqmpqVRUVJCWlkZkZKRXJ+XOOhv6UrHQxxBvFlB/o/wVQg+risLngc9LKetM4UegpUtN9rlUQcrQoUNZsWIFGzZsoKKigi1btjiFOYJ7a0FCQgLr1693Ui78idLeFQrfMWTIEObMmUNTUxNCCN5++21SU1NZuHAhr732Gnl5ecaUQ0NDA4888ohXJ+Xc3Fyqq6vZtGkTOTk5HVr99CRtmzZtYv369T55uAZyjFBRGaGHVUUhCvchRoloUw99hueee47KykoGDRpEUVER1dXVDB48mLq6Oo/HdCWVq0KhCA5OnTrF1q1bycvTskK//fbbVFRUGBVk8/LyDL8E6DgviqdQSlf0cUPP2VBcXGz4RoUyyuIZelitHlkC3GVal0KIfsAPgTd8LRSAEGKVEEK6LMc7OGayEGKrEOK8EOKIEOLnwsUE0h1qamqMTIwXLmj5W3bt2kVdXR0TJkxg2bJlbqs/6hr08uXLVS12hSKEWbZsGVlZWQBUVFQwYcIE7Ha70a+tVkdcu3Yt2dnZXp2N9XHDZrORl5dHXl6eR2fpUKo4qypIWsdut7N9+3avVUp7AqsWhZXAViHENKA/8DtgElq1tll+kg2gHJhrWm/11FAIEQu8jqbUTEOL1S4AGtDk7Tb6D/yxxx6jvr6eQYMGceutt/LJJ58wY8YMwL1ZTe/cOTk5zJ07V2nSCoWDUCg4p2dk1Qfr9evXs27dOsNx8Qtf+IKRX8UqaWlpHi0JOuZxw1skhTLl+w+73U5paSnp6enYbLYev35paanh2zZrlj8ftd6xZFGQUn4ETAbeRqvyNgDNkTFdXky+5A9apJTHTYu3JOa3AzZgqZRyr5TyOeA3wHJfWhUSExONtJynT59m1KhRREZGkp+fz5IlS8jJySEvL89IvgQXEy156+yh9lagUPiI3wPDwang3IdoaZnzAyfWRfSMrPn5+YbVYNWqVaxfv541a9Zgs9koKiqisLDQ7fHu+raV/q6/mBQWFnq1RObm5volaZvi4oO6tLQ0INdPT093KhsQMDwlWAj0glat0o6Wsa0SeAYY66X9P4FXXLZNQwvnHNPR9TqTcCk5OVmmpKTI++67z0i0hCkJy4oVK4ykSjodJRzxR0ISXydX8UaoJRfpSULx3tADCZe0y3BW79fAT9BytADMAA77+/pW+v17771nJFYz9yW9f5WVlTn9de1v7vp2Z/p7T/bjztLV33YwfyczDQ0Nctu2bbKhocHSdjOh1u+99XnLCZeEEDZgCjAUF0uElPL5TugmVtmJ5hdR5rjmz4C3hRCTpJS1btoPBw67bDth2teuCE1nq8jFxMTwxBNP6AMcsbGx2Gw2Nm3aRFtbG6dPn+bChQvExsYyefJkp3PqleQSEhLcXqej/V3BXUU7f9GbKqX5GnVvvBL0Bed0iwJoEQjLli0DcMqfsGLFCvLz891OAbhz3uuMQ19vjGIKlekSm83m1uQfLFMCPYYnDcK8AFlANVqndl1arZyjuwsQA5wElnvYvwn4u8u2ZLQ3/YyOzm/lzeLZZ5+VgIyKipJz5sxxSt+qvyFkZ2d7fLPoaZRFITgIxXtDz1kUitGsgXeiRVaNc2yfAxzw9/Wt9PsPP/xQZmVlGf19woQJ8r777pOATE1NlWVlZVJKrb/l5eXJvLw8r33OSr/0dd/t6HyduZ65bW+3KHiir1kUrEY9/AF4Ba3ca5jL0q+jg32B1Jya9qFlhXTHcWCYy7Zhpn3dJiYmBoDs7Gzmzp3LfffdR2ZmJtXV1YZvwqRJk1i/fr0RClleXh4w3wNv3sXKJ0IRJHwXzVK5jiAuOHf69GkA+vfvT1lZGevXrwe0yAfdNyEhIYHo6GhWr17tNbJJf5u22sYXfbWja1qRqSttPRHqkQ+6pSEQDo4BwZMGYV7QIgfGWWnrrwXNgfIY8HMP+78JnAEGmLb9BDgCiI7Ob+XN4vnnn5eATEpKkoDMzMw03jLmzJnj9Nahf9YLx7ibi7T6BuIPfO0TEWras5U3Al8RavdGyp6zKHha6KGCc53xTdIXm81mfDZbFKTsurXAdZt5XfeDMvs8deZ6Vtr1tEWhL9AT98aX45i3Pm+1024Csq209dUC/BbN/DgGzbHpZYcikOLY/7/AG6b2g9AsB88AlwO3ONp/38r1rAwY//73vyUgw8LCJCAzMjKcBhB94ADkfffdJ7Ozs+WOHTs8dkD9Ye3LB7ZVAlVFLljYtm2bXLVqldy2bZvfrxVq90bKnlcUgKnAl4Fox3o0EO7v61qdenDt6xkZGTI+Pt54KcjLy7M05eiu3+3YsUMmJiZ6HAe8KQrmKc9AmPGt/LZDfZqhq/REv/flOOatz1t1ZvwL8FshxEhgD9Bs3imlfM/ieTrDKGA9kIDmH/EOcI2U8pBj/wi0Spa6DKeFENcBjwC7gTq0/Ak+K5/Y3Kx9bb2a3OzZs5k/fz6vvPIKu3fvZurUqcybN88wRz3yyCPMnTvXo7NObm4uDQ0NxueepDc6SHUGPdwo4GFHfZxQKTh31VVXcfz4cebNm8cll1zC9u3bqa2tJTExkeLiYoqLi428CuDZQc9TnpXq6moSExPdjgPLli3zWJchNzeXLVu2OFW09SW+qMsQKo6LvsRut3Pu3Dnsdnunpic6m7ehx8YxTxqEecG9E2OPOjP6e7HyZrFt2zbjjSI8PNx4g5g2bZoE5KhRowzNv69p0aH41txThOK9oeecGZ9GUxQG4xwqmQXs9/f1rfT71157zcliuGLFCnnffffJrKws+eqrrxpWxBUrVnicPtDxZFGYMGGC3LFjR4eyuMOfY01HU5Q9ZVHw9Xf099Tjtm3bZHFxcaff9HvS0umKtz5v1aIwptsaSS+gsVErayGEICUlhdzcXHbs2EFsbCwALS0tgKYV9vU3doXCIkFfcE53ZAwPD6eiooL8fC0P1Jo1a9izZ4+RyvmWW27hrbfeMo5z9ybtbly45ppruhU+66uxxp31wBd1GXwhn6+tEv4Ob0xPT2ffvn1MmjSp08eZ/wYLlhQFedHcr0Czwnz66ad8+qnmoK2Xnx0zZgzHjx/3aDLyZXlVVapV0UsI+oJzerTTwoULaWxsJDk5mYqKCqqrq/nqV79qmP4ffPBBp7wK7h6y3e23/uz37qpUBssLj68LSfn7gWyz2YiJiel0VISnvA2BpjMJl64H7gPGAguklJ8JIb4GVEop/VIYKpjQ55qGDx/O8ePH6devH62trURFRXH+/HlSU1NZu3YtDz74IEuWLHF7Dl9oxfpA0dDQ4LHevUIRQugF537iWPd7wbnOoheAKy0t5ciRI2RmZlJSUkJJSQl2u51JkybR1NTEt7/9bQCj5oP5Iav32+rqavLz82loaHBbkr4jRaAn5vuDsUqlrxUWXz6QA10PoiewpCgIIW5Hc2j8K5qpMMKxqx9awaig6ND+pLS0lMbGRsaOHcvx48dpbW1l1KhRHD58mMGDB/PEE0/w2muvUVRUxKRJkwzHJHNnNxd5yc/P7/CtwN2goQ8UeXl5Kr+7ojcQqIJzlmlt1WrRHTt2DLjo1AzwzDPPUFurJYqNjIykqKiIadOmtSs1rfdbvfKkJzpSBPxZolnPOOmv8/uKYLOm9oUsjZ2pHvl1KeUzDiuCzjvAL3wvVvCRnp7OG2+8wfvvv29s0xOg1NXV8dprrxnV5Xbu3ElJSQng3Nl1rdhTqldXvFWiDJZOolB0BynlR0KIyWh5UC5wseDcI1LKYwEVzoG7aKfZs2fz7LPPUlVVRVJSEqmpqTzwwAPMnTuXhoYGVq5cSUNDg6EwuKsE6Y6cnBw2bdpEdXU1NTU1Rh83Pxw7erPu6oNUL3blj3P7kp6wqnTGSpCenk5zczNNTU2djnIIFawqCpfiPkvaOSDWd+IELzabjcjISKe64I2NjcbUw6ZNmzh+XEsAOWPGDObNm2dUkHTtUFbfCty1C5Y5Q4XCV0gpjwN5gZbDE8nJF30qw8PD+epXv0phYSHDhg2jqqqKI0eOcOTIEd566y1WrFhBTU0N0dHRVFdXs3r1amOaQe+3eht3VsXCwkIj3DIxMdE4pjMPx648SK0qAMEQ6uhPq4pOZ6wENpuNiIgIiouLiYyM7JVWBauKwlFgPODq1JiJ5p3c69E1xbi4OOrr60lMTKS6uprz588THx/Pjh2aHpWammqkN125ciXR0dHtOpSVh72/NPeunjcY3iQUvZMAFJzrFOaXg1GjRjF//nyqqqro378/AElJSXzta18jNzfXqZ+sW7fO4zk9PXA95VbJyclhy5Ythv+DGde+2ZUHqSd59HPrlhD9+oGcmuiJl6XOOjsGa7SCz/AUN2le0KYe9qPNGZ5Fy5i4FC0R0n1WzhHsS0fx1Hpc7MyZMyUg+/fvLwE5cOBAp4xtK1askFJ2P+63OymWvV27q+f1RTx1XyUU7w09l0choAXnrORR2LBhQ7sMrObl7rvvdioGp/cT1zLU3kpUu/ZV1/Tu3vqfL9KxexozzJkf3V1D/233tbwxVgi1fu+tz3emQ/8KsJs68XngQavHB/vS0YDR0NAgX375Zac874CMi4sz/mZkZMgVK1ZYrh7prXN1p+N5Gzi6et6Ojgu1TtGThOK96UFFYR/wD2BkT1zPdbHS7zdt2mT0+/79+xu1XnCkcjbXdXFXv8W1P5r7kqe+qqdt1s/rbUzx50O6I4VG/237unZMoPCUiKkrCZpCrd/7RFHQzoMNLSf7dCCmM8cG+2LlzWLTpk1OyoGekc1VeXDVvjvS1n3duQKRCS3UOkVPEor3pgcVhYAWnOusJRGQ06ZNk2PGjDGysL766qsyMTFRvvrqq7K6utpj/9cftroSYFYWXPuZ3kYfY6yOEebz9cRbvjeLQihaGTxlRuxKxsRQ6/c+UxR682JFUXj33XcNJWHAgAHytttuc6ogOWLECJmZmSlfffVV4y1ASs8KQTB3pM4qMeZO0dFbSGcI5ntklVAbMKT0Pmj4ciEABefMixWLQmFhYbuXAf0hrqdf1l8Q9Ad8fHy8fPXVV51+u3qfysvLa5fqOS8vT65YscKwRHSmD3myUPTEW76337a76/dU1dauXqe3WhSsyN9tRQHYDLzpZnkDeAX4A3CVlXMF69LVuUr9zaIrFgUppSwrK3NSKoKF7lgUOprX7Ay9waQZTAOGVXpQUbgF+Aj4GlqV2KvMi7+vb6Xf79ixQw4YMMCtf8LgwYONvzt27HCaMtAViI7GAfMxHf3W3Z3DnV9ET1sUrMraU7UMAlkzQSeY+r2V++ELReFPQD1QBvzTsexHq9BYALyHVlHy81bOF4yLlQHjP//5j1enJkCOHTu2nUXBG/rDNDs7u8O2wYyyKHgmmAYMq/SgohDQgnOd9VHo16+fBGRsbGw7K0NWVla7glHuxgHX3/SKFSuczuHtIe9OcfaX35EVOvvbtvpm3l3ZfGm5aGhokJs3b5Zvvvlmn7UoWA2PbAT+IaX8rnmjEOJ3gJRSXiWE+APwS3pxlka9aE1UVBSNjY36QGfEWVdVVXHgwAH++Mc/eszQ5hrK9MADD3DgwAEeeOCBwHypDqipqTHCvJYtW2YpNNIcvtTdMCaVN6LXE9QF50pLSwkLC2PKlCm8/fbbtLa2Eh0dzdixY3n//feJjY1l8ODBJCUlGfkP1qxZw7p168jPz3caB/QQQz39+pYtW3j88ceNBD1ZWVlGjQV3SdlqampoaGggLy/PbXhibW1tp0KY/Z0TwV1Itc1mIz09vcNkRt2VzRcpmvWkS01NTUYCvebmZqqrq1mwYEFIhYl3935YVRSWAte42f7/0BIx/QB4DC1ne69F/1GfP3/e2DZo0CCqqqpYsWIF27Zt48SJE3z729/2mKHNtQO89dZblJWV8dZbb3HNNe5ucWApKCgwakq4ywmhUHQHGeQF59LT09m8eTPvv/8+0dHRDBs2jAMHDhgVJePj46msrORLX/oS4eHhCCG49tpryc/PN3IO6ImXNm3aRHFxMXl5eWRnZ1NUVERBQQHLli1r90LhLheC3hfXrFnj9JDSxxS9OJWugHT0IHO9RndypZiPtdlslJaWsn37dn74wx8Czg97K8mMrOaCcHfdztRc8JaBUZdzzpw5zJkzByklhw8f5sCBAwDcfvvtlq7RG7CqKAi0/OufuGy/zLEPtApwbT6SKyjRc76DZl2QUnL69GmioqLYvHkzu3fvBuCPf/wjr7zyitsMbXo+ddeOEKy51T0lgFEofEUwF5yz2WyEhYVht9uJjo7mwIEDDB48mMrKSgYMGEBlZaVhUdTfOr///e/z9ttvGw9s3SLX3NxMXl6eMQboD7ja2lojbfPKlSs9Vm30NFaY00MDhgLSkVLveo3uvMWbj505cybFxcVkZGS4rUdjJTmRVUuiu+seOHCAUaNGMWPGjA4VBm9Ki1lOm82G3W6npaUFKSULFiwA+kZBKMCyj8LDQC2wApjrWFYANcBaR5uvA29ZOV8wLlZ8FJ566imv/gkDBgyQo0aNkjt27HCaY9OdlfLy8pzO19l5uLKyMmMeNNjm7INpPs5MT3lZeyNY74036DkfhdvRkrg9jJabZaxj+zeAjf6+vpV+//zzz7fr667+Cffdd5+xLSUlxSkSYseOHUaYox7ZYM6zoPsp4cWR0epY0Z25/a5cw114ZE/2OdfrPvnkk3LVqlWG4563KIY333xTbty40bLvgTuHQG9OgqHW7731eauduR/wI7RUzrqj0VG0UrD9HG2SgVFWzheMi5UB4+mnnzY6dEREhLzyyitlQkKCBKQQwtinhz/pHd+Tc59VxyR9m57YxdOA4npsTzoCBmunUN7PXaMHFYUPgMWOz2dNisKVwAl/X99XTswZGRnGy8KIESOM8EldWdDbmcOp9T5s5QXAn9E/nR0nzLIE22/b1fGwo7wIq1atkk8++aQlRaG6ulo++eSTTvfJm1IUbPemI7z1eUtTD1LKVuDXwK+FELGObWdc2lRZOVcoo91LjebmZqqqqqirq2u3Dy6a7HWzvWvVSH2/2TGpvLycm2++mbKyMuMYuGheu++++zh48CALFixwOw3gajoMhgIugaY7Odj7jFkxsAR9wTndidlMXFwcl156KSdOnKCqqoqIiAhAKxSnl6NOSkpi0aJFADQ0NLB161ZaWlqYNm0atbW1XHvttdTU1FBYWGg4MXqioynK7vgXdHacMMtSXV3dqWv5G5vNxty5c411T/0/PT0du93O/v37qaiooLS0lFmzZhl9Pi0tjfLycqe+X15eTkVFBaNHjw4pR0ZfYNVHwcBVQegrfPbZZyQkJJCUlMSRI0cADCVBJyMjg/nz5xtzkLt27aKoqMhwAjR3MHeOScuWLaOsrIzU1FSnAUH/3NDQQEVFBffcc4/bH6onv4e+7FvQHW/fvlBnPggI+oJz7pTE+vp6du3axX333ccbb7zBHXfcwZ49e4wxYcyYMWzdupUTJ05QVlbGihUrOHHiBG+//bZxjgcffJC5c+cajojuHBCtlpe28rA3n0s/xlwC29s44aqI6Nc4ceIE27dvD1pl2l3/r6mpYePGjSQmJlJXV8eYMWOMEtF6nz948CAVFRU0NzcTERFBenq6k9Kht21ubmbr1q1A7x4jOq0o9FVefPFFJk6cyM0338wjjzzits3YsWOJjo4GYN26dRQVFREfH8/kyZONkrJ6B3PXOdPT0ykuLmbRokVOA4beMXXnSE8d2tUBSIUWdo+0tDQOHjxIWlpaoEXpzTwK/J/DeRHgEiHEtcAaYFXApDLhzolZLzv/wgsvcOTIEe677z6am5sBrYLsE088wfLly9mxYwczZ87EZrNRVlZGVlYWaWlp7N27l0mTJpGTk8OmTZsoKipi3bp1LFu2zOmB3JECYK7uCN4f9uZzAU7ndT23q2LgSQ673R4QZbqrFhS73c4zzzxDbW0tbW1tZGVlGQ/7o0ePGk6KycnJtLW1UVlZSVWVZiyfNWuWEdqph0xmZmaSlZXVe6tGOlCKgkVuuukmPvroI1544QWPbYqKinjqqaeM6QbQYpu/973vtZtOcMfKlStJTEx06uyeNPnO0FGnUiWk3dOXTY09hZRyjRBiEPA6MAAtC+wF4LdSSvcaeQ9TX19vfLbZbDQ0NGC327Hb7dTX1yOEMJSE5ORkKioqePDBBzlx4gQA+/fvZ9asWUbEgzlPQmJiovGCsHnzZux2O/n5+Tz66KO8/PLLHt/29T6r52SAjqcN3J3Lk2Lhqhh4ksNms3X6QWm329m5cydCCC6//PJ2Jn69jbdpP28KVEchj7W1tcTHx3PNNdfw7rvvkpmZSVVVFRUVFYwcOZIZM2awYcMGIwwyNTXV+H66xSEjI4PU1FQmT57cJ8aGsI6bKAAuueQSampqjGkHV8LDw5k3bx4AmzZtwm63c99995GVlcXDDz9Mdna2Uy15/YdeUFBgbNMVAXcx0gUFBZSXl3PDDTdQXl7e7vo1NTXk5+dTU1PTbp+7a5mPW7p0qcf9oYrdbmf79u3Y7fYunyM9Pb1PvC0EGinlT4EEtGJz1wCJUsqgyUBm9j8aOnQoY8aM8bj/y1/+spEjYdiwYcTHx1NXV0d+fj7R0dEkJCRQXl7Opk2bjAew/jArKSlhw4YNAFRUVBhTmK6Y+yzgFILobRwwjy/uxhozubm5Tuc1v6SYzx8WFsasWbMsTTvoffLdd9+lpKSErVu3snHjRoqLiyktLXVqqz+QXbeb5cvLy6OhoaHdd/V2rN6n7777bt59910qKiooKSkxQlyFEJSWllJRUUFycjJjxoxhwYIFxvfTj4+IiKCiosLtWNwr8eTlCBwA4h2ffw7YPLX1xwL8GNgFnEGrV/8ScHkHx4zGvVfywo6uZyWV64YNG9wWh9GXIUOGOK3rZWfdFYOxWuPB7JHsLd2zp4IzrufwdJwua1fpiodvV6IyrB4TDNEOOqHm/Sxlz0U9BHrpqN9XV1fLV155RcbHxxv9eurUqTI8PNypr8fGxsqZM2fKsrIyWV1dbUQ3TJ06VU6dOlWOGTNG7tixQ0p5MRxywoQJRrrmvLw8I6pJrx+zYsUKpwgDPTpCj7Awp3vWQy7NlSk7+l5diYhyjb4w/7Y7CovU++Sbb74p33zzTbl582ZZXV3dqSJMVkp0ux7r7Vx6FIO5jf558+bNHscQKyGgodbvvfV5bw/d88Aljs+twFBPbf2xABuBXOByYDKwATgODPFyjK4oLACGm5bIjq7X0YCxadMmWVxcLK+77jqvYVLR0dFyypQpRrEYPexJrw5nfph7erB7wptyoXccqwOF63HdDaHsSqfoSsiX1WPcdfxA5VIItQFDyp5TFAhwwbmO+v3jjz8ui4uL5R133OGxzw8bNswpn8KECRPklClT2rVLTU01akHooZPm/u9aRbKsrMx4+JeVlRnH6Iuel8VcVMrqeNLVcEvX8eKjjz6y9GCV0voD3NtxngpgebuO60uDXr/j8ccf9xrq2N1xI9T6vbc+781HoRT4uxBiG1r2xR8IIc65ayil/IWX83QJKeUC87oQ4k7gNDALzbrgjVop5XFfynPs2DGSkpIYNmyY13YNDQ28//77gJbe1Zz/3Yw5ksFqaFJaWhqvvPKK231WHR49HRcIuhKVYfUYs7fz9u3bVfRC8LIfuA3tJeBdx7ZpaAr+C8C1wLeEEAtlALI0jhgxAsDwN3CH7sMwcOBA/t//+3+0tLQQGRlp7E9KSiIyMpKKigry8/MBuO222xg/fjx2u93JxyA6OtqIhiosLHT6XFFRAbSPrtLJysoyfCD0KQh3fkcd1Yzw5rPkbrzQTf0dOfa5RiBYjSoytzP3f09j17vvvsvWrVtpbm5m7ty5pKen09TURHNzs+G/oEefbNy40UjF7CqPL+pF9Ba8KQq5aEWebkbTVr8AtLhpJwGfKwpuGIjmU1HXUUPgeSHEALSU0w9LKZ/r7sXnzZvHRx99xObNm722mzNnDhUVFRw5coQ5c+bw4YcfMmfOHBoaGli4cCG7du1iyZIl7R7sOTk5Hju2Ozx15lCKdOiKrF05pju5FBR+J6gLzs2aNYs33niD7du3t9sXERFBc3MzEyZM4MKFC9TX13P27FkAmpqajHa33XYbNpuN1atXM2DAABobGykqKqK+vp5JkyaRl5dHdXU1q1atYsmSJYB7h0PdSdpcnK2mpga73U5mZqbT79vs7KeHY5sjGNzVjHB3rLkglSflwTXVsVWs9kvX83fU//X7pP/VI1SKi4uNUEe73c7hw4eJi4tj8+bNzJgxw234Y2edLHsrHhUFKWU58D8AQog2YI6U8mRPCeaGPwDv4z45i845tAJV29GUmhzgWSHEUinlk66NhRD3APcAjBw5kv3793s88ZkzZ4iLi+ORRx7hzBnPqST69+/PhQsXAM3R5+abbza2nTx5ksWLF3P06FHa2tpoaWmhpqaGhQsXcvToUYYPH86ePXsYPny4sS8hIYHw8Pb/puPHjzu17yodXccqjY2NXu9foBkyZAiHDgWm/lCw35sAs5QgLjhns9k4ffq0W6fY5uZmYmNjiYyMZP78+Tz5pDbE6ApESkoKX/rSl1i5ciW1tbXs2rWLuLg4nn76aQYMGOB0Lt3SoOdccfdgXrVqVTsZCgoKjGNLSkpITEx0m7PFXQTD4sWL3eZA8FSQypPls6tv3laPc9fO24Ncj1bYt28fM2bMICEhoZ2ycd111zlZGiMjI5k1a1Y7K2RTUxORkZHGcX01t4rVzIwBjY4QQqwFZgOzpZYl0i1Syhrgd6ZNu4UQCcBKoJ2iIKV8FC2Om6lTp8qJEyd6lOEvf/kLl156Ka+88gqPPfaYx3aJiYmEhYU5mSqnTZtGVFSUEUozffp0p3Kzunfx7t272+3Lzs52SsRijpvevXs3119/fbfCc/QwrTVr1nTLErF//3683T9/ECphnYG4NyFEUBec++STTxgyZAjx8fHU1ta223/mzBl27drFrl27DIV9ypQp7Nq1i5MnT3LkyBGjKFRRURErVqxgwoQJRrg0YFgr5syZY1gWAK9hj+ZxQK9KmZWV5TZKwZwltry8nMLCQnJzcykvL3f70OtMQaqewqwYAGzYsMGYinGdzjh16hQRERGcP3/emFpwp2ykp6fT3NyMlLKdVUPPoWK32ykpKXGaxtCP7UtYfoUUQlyBpt1fhjbd8BGQL6Xc6yfZ9Os+DCwG5kkpD3ThFDvRplG6xalTpwDtzdQb5pSm4eHhTJs2jR07NCNISUkJeXl5LF++nKKiIvLy8gwlISEhgZycHCOFc15eHllZWUYiFn3w8HVa5kAPAN1BpajuFTwO/E0IcSlalBNoPgo/BP7hWJ8D+HWc8cQzzzzDzJkzyc7O5oknnvDa9vjx4yQnJzN+/Hh2797N+fPnefrppwHt95mdnQ1AWVkZmZmZREZGGgmLsrKymDVrlqEcZGVlufUhcJc/Yf369e0UZlclOjo6mpUrVxpvymbfKCsPPW9Tfr40x3s6l/lNHrTwUXN+Ax19fejQoWzcuJHMzEyP17LZbEyfPt1tGKWeQ0UPhdV8/ZytG31pGsKSoiCEyAGeB94CXnVsng2UCiFukVJ25FzYJRxzk19GUxLKOmrvgSnAse7K0q9fP8A5S5s39LKk8+fP58SJExw4cMCYDyspKSE7O5slS5ZQWFhoHLN8+XLKysqIjo5m4cKFABQXFxtJWHJzc8nJyWHLli1OORl0uvKG3R2fBteUsD1NKCs5CoMfACeA76E5MILm2JgP/NaxvpGL4047hBA/Bm4B0tCSNb0D/NgXLzGTJ08GMOo3dERVVRVPPfWULhdSSlJTU7HZbBQVFXHu3DkjO+MjjzxiJGqaNWsWS5YsYfPmzRw+fJji4mInh0gdXTl2fclw7cN6u4aGBsMHCrQXGf2B6ytnPfNDXM9caC7N7Pow9bbNU0pkXQHQs1pmZma6LSOtf6fNmzdTW1vL/v37qaqqcjtFsXPnTg4fPmxMVXi6np4Qytv37u3TEFYtCr8EfiWlzDNvFEL8wrHP54qCEOIR4E40Z8o6IYQ+iJyTUp5ztPlfYLqU8vOO9aVAM1rERhuaA+Z9aG8n3ULv0HrxF2+EhYXxxhtv8NprrwFw/fXX88gjj1BfX09JSQlZWVlMmzaN9evXs3r1aiPP+9q1a9m1axfV1dU8+OCDPP744wCGcrFp0yZmzZpFUVERc+fO9Tg4QM+8YZuvd+ONN/r9eq6EkuOmwj3SNwXn5gJ/QrNICDTn6mIhxGVSylPdkW/SpEkcPHjQeJhYJTY2lqysLEpLS1mwYAHXXHMNiYmJlJSUABcd7caNG8cNN9xg+BLo+xMTEykqKqKgoMDJZ8GcqtlbXQhdeT506BCrV6+mrKyM48eP88ADD7TL/tpdzOZ414enu4ept23uIifMikVpaSlbt24lKyvL7Vu83vb8+fOANu2nR6WYlZjS0lLjXruzTJiVKE8vXX0pxbtVRWE84M7u9gTa/L8/+Jbjr6un82ou5oAfAYxz2f8zIAUt98PHwN3uHBk7i25JsKIotLW18dprrxnmvsGDBzNlyhQATp8+TVpaGqtXryYvL8/I4qYPAqmpqQwcOJAHHnjAMBnqP+ji4mJmzZrllDHNjPkN25t1oSuWB3fHBHMVOUXo4aogdOK47oRSe+XFF18kPT2dnJwcfv/731s+7syZMzz//PMAPPLII/zrX/+iurqa/v37ExUVZThHbtiwgdtuu41169axZMkSGhoajOmB7OxspwJOHdV8WLp0KUVFRcBFvwR9/cUXXzR8LPQQa1+Zzs0PVdc5fHdz+h1tc5d22WyxcD3WVZEoLi42pgzq6+sNRcD1PE1NTQghmD59epe+f19K8W5VUTgJXA1UuGy/Gs1s6HOklMJCm7tc1h9Hm/P0G1anHj777DNKSkqIjY2lrq7OqdJkQUEBd999NwBr165l7ty55ObmsnTpUsOfQbco6B3ebrdjs9mcQqO84W1g6Yrlwd0x5jf67ioKoeKYqAgJOhNK7RX9zb9///5dOj42NhYhBNXV1YSFhXHhwgUuXLhAY2Mj/fr1o66uzigyFx0dbeRGmDVrllNfN085uvYVs5KgKxd6CCTAhAkTePjhh/njH//IAw88YExj6s6Mrp79ZjqrTLhOZ7iuezqfzWZrN22hY86DoN8b87nsdjtvv/02zc3NTJ8+3Wg7fPhwIiMjGTduHBs2bDCsFWlpaZSWlrqduugMfcmx0aqi8Bjw/4QQqYBeJ3UW2vxivj8ECzb0mvRhYR0HgIwZM4bnn3+e+vp6wwIRExPDuHHjKC8vx263Gxr+rl27jKiGtWvXGj9y3cqQm5trDCDmB6i7B6tr7LT5rxlv+zw9sP3tD6AcExU+xGsodWfCojMzM7HZbMybN49JkyZ1SoiwsDDa2pyDNfr164eU0tjer18/+vXrZ4RZ/ve//2XcuHEMHDiQY8eOGQr40aNHWbx4MVVVVZSXlxuh0QkJCVRWVrJ48WLuueceYmJiOHbsGAsXLuSqq64CNAfs+vp6fv3rX1NTU2McO3ToULKzs5FScvbsWfbt20dMTIyTvOfOnaOxsbHdvra2Nux2O/369TPun77NZrN5HCfPnj1LY2Mje/fuZeDAgZauBdr4eebMGad9evuoqChmz56NzWbj0KFDRtv+/fszePBgDh06xPDhwzl06BDx8fFUVlZ6vE5n8RZ23avCoj2lbDQvaPN+3wMOo839tzk+fwcQVs4R7EtHqVxXrVoli4uL5cqVK72mcAbk3LlzPe6bOXOmBOSUKVNkSkqKkdPdnI60o3zm5roP5u16iucdO3Z0OS1zV1O7djddqa9SSQcjoZbKVcqeS+Hs6wVYCxwFxlppb7Xf//SnP+2w3+tLdHS003pycrIcNWqUBOSoUaPklVdeKUeNGiUzMjKMug16fQc99TumFM1SXkzTrKdx1muz6Nv1VPH6ceb+pLfRU0G7poH3lqq4oaHBqMtg3q+nay4pKZFvvvmmcY5Vq1bJJ5980mPa4zfffFOuWrVKbty4sd01O5Jj8+bN8s033zTqQ7j+NadefvLJJ420zZs2bZKrVq2SmzZt6vA6viTU+r23Pm81j4IEHgYeFkIMdGw7a+XY3oZrohR36D4FutczaFaGW2+9lVtuuYXc3FwjzTNo85R65jbdsdEcA23+C9rbt9nMqFNYWGjMSRYVFRnn6owpvzOWA19GPSjHxL6DEOIAME1KWSuE+DlaSemul/m8eN7uhlK3w/zmbxV9umLo0KGMGzeOq666CpvNZsT+Hz58GNAcGS+//HJ27NhhmNXr6uqIi4ujvr7eqSrismXL2LVrl9HvH3jgAZYuXdqukiVoJnmzv4IZfYyYM2cOs2bNMsz83kL+jh49SkVFBREREUY7Xd6mpia2bdtmTF0cPHiQiooKSktL3UYCzJgxwykstKmpyai66y0Kw2azERERQXFxMZ999hmVlZVGbgN9v1nuRYsWOeVdCA8PRwhhWDx6e5SCr+l0Kr6+qiBYRQjBNddcw/vvv8/111/Pf/7zH8LDw40f8GuvvUZZWZkxV7Zx40YjccicOXMoKipizZo1Rm0I1weoOU+763SE/sDWPaP1HAx67QcrCkNCQkK7lK+e6GrUQ0/6I/SlWOcQYgRgA2qBPOAvQLcUBR+FUnvEqm+SmZMnT3Ly5EnD7+j222+ntbWV2tpazpw5w9atW5k7dy5ZWVmGk11mZiafffYZ9fX17Nmzh/z8fHJycli/fj2TJk1i2rRpLFu2zFAEsrKyyM7OZsuWLUyfPt2p75tfJFzrv1x55ZUUFxdz8OBBFi1aZPQN3eFP366XXHaNDNCzuEZGRhrpo80P6LS0NLdZH/WH9JYtW4CLU7qeMPdf/fp2u53KykrjJcxut7dLwKRfRz9eCMHWrVudlB2Fdbqes7cPo8cHu0NKaRQcOXr0KOHh4bS0tPD888/z/PPPs2LFCiNqoba2lvLychYtWsSyZcuMDG0bNmxg5cqVHvOwe8rTblYqHn/8cSMxi6ec754e2OvWrWP16tVs2rSJ9evXO7XNyckxMrt1NeqhJ/0R+lKscwjh04JzVkKpu4sV3yRP6JbFwsJCzp49S1JSEuPHjyciIgK73U5aWpoRGjhv3jxWr17NhAkTmDx5MitXrmTDhg2GsvGb3/yGnTt3Mm/ePFJSUvjc5z7HjBkzeOKJJ2hubiYmJgabzUZeXp5bWfQxwm63U11dbbz9p6ens3PnTs6fP8/gwYOdtkP7aATdMhAXF2f4briLPgD3/W769OlG3QV3uMtz4CqLvq4rM2PGjDEKP7kqPh0VrFJ4RykKnSQsLIwpU6YYyoA39M7dr18/443EXNRk6dKlFBcXc/jwYVauXMm6deuMzIx6/LSOuzhqd5gf/q7VJF0f0B09sIuLiw059LZbtmwxTJorVqzoUtSD2fLRmUJYXcHfnsnKYtElfF1wzkoodZfojoIAzn1fLxh15MgRoqKiqKioYMeOHUb2QCEES5YsMfqrnvr5+PHj2Gw2srKymDRpErt37wZg2LBh7Nu3jwsXLnDPPffw0ktaJGhGRgYffPCB0bfN2RhXrVpljBGLFy9m9OjRhpKgT5mClltAjw5w99vW39jNznruwg9dH9yux3vCnOdAVwC2b9/uFN2gy6bnM0hMTGxnNehqwSqFM0pR6CRtbW1O/gXe0N8kYmJiOH36NKmpqSxZsoSamhpWrVrF3r17GTVqFGVlZaxZs4bExEReeOEFCgsLycnJYeXKlZSWlrJu3ToKCwstvYW7Pvxdc757+wsY86IrVqzAZrO1a5OTk2OEc3YVXSa9zkRH36k7+Hs+UlksOo/0ccE5aSGUuqvoPgodmcjdMXz4cFpbW90q0YMHDzZ+8+++q1XX3rVrF2vWrGHVqlVGeuFf/vKXzJgxg5dffpnBgwdz6tQprrzySg4ePIjNZqNfv35UVFSQlJREVlYWL774IiNGjODtt99mxYoVToWPdAoKCli1ahXNzc1897vfxW638+GHHwIwatQoxo4dy4wZM7z+tnUFOS4uztjmWn1x37591NbWejT319TUsHHjRhYsWNDOx8Cc50DP1piSkgJoVlt3KZ31e2B+KVD+CL5BKQpdwNO0gyv6HNrp06dJTk6moqKCe+65xyh5CjBo0CDg4gNHz774t7/9zagKt2zZMtatW+c2dbPrlIA3i4O5tLVeO8LVz8E8tWF+eJsVDl891EMxBbOrBaEvxVL7AxnggnP+5OzZs4Zjoyu7du3ixIkTjB49mpKSEjIzM4mPjyc5OZl//vOfzJo1i127tNIXn332GYMHDyY8PJwDBw4wYMAABg8ezLFjxwwnSymlkbb4hRdeMArKmetIfPWrX2X79u0sXryY5uZmmpubKS0t5eDBg0b2wrFjxxrOheY0xq7+Btu2bWPHjh0sWLCALVu2IKVkxowZxkN5w4YN1NbWEh8f77FvmP2zRo8e7aSU6CGpdrud5557DtBCWS+99FK3iZr0z8pq4B+s1nr4ElAvpdzkWP85WhzyPuAuKWW3aymEEt58FOBimVkzemlq3ZwWGxvLmTNniIyMJDU1le9///uAZu4vLi4mNTXVODY9Pd3wVp47dy45OTksX76cBx54gAcffNCIcCgqKjIUjdra2nY+BeZ8C3oyFnAuX9uTD29fRjr0lIOk61uWemPpPoEqOGeF7kw9mJWEpKQkFi1axKeffkp5eTnZ2dkcPnyYIUOGMG3aNGJiYoy0w8OHD3caP+rr6xkwYACNjY2AFp+v155obW3FZrMxefJkI57/iiuu4De/+Q3jx49nzpw53H333SQkJLBp0yZ27NhBRkYG99xzD0VFRdjtdjIzM2ltbWXEiBHMmDHD6TvoOV0qKyuBi5aFI0eOAHD+/HmnKQs9+iEzM5O6ujpuuukmj6mWBw8ezODBg8nMzCQqKsptOuTS0lIqKytJTU01ciXomPud6oP+xapFYRXwXQAhxFXAT4CfAwvRyjrf5gfZghIrPgquSkL//v0NjV1HrzBXXV1NdXU1v/vd74zjxo4da3gaL1q0iK9+9ausX7+evLw8cnJyuPHGG6moqOCdd97h1KlTZGVlGcmadEVDD6XSFQi9OIzuhKiXpnXF32GK/nqg95SDpLIg+JZAFZzrSRITE1m8eDEDBw4kISGBq6++mvDw8Ha/fz1R0eHDhwkLC2PEiBHU1dUZLxlmRo4cSWxsLBUVFcyaNYtnnnmG2tpaBg8eTFRUFNOnT2fr1q1GQSrQnKv1v9HR0VRWVlJZWUlERAS33nqrkeXQta4CtK+HkJSURFVVFW1tbYwdO5ZRo0YhhHAaU2pra6mqquKSSy5pJ39paalhMamq0kp5uEuH7M7HQPkF9TxWFYUUoNzxeRHwgpRyjRBiE1pltz5DW1sb5eXlHTc0ceHCBUBzMjpy5AhVVVUIIZgzZw5JSUm8++67RjU5gC9+8Yvs27ePoqIiEhMTKSwsNKYDCgsLDXOdXvp61qxZpKWlMWvWLMPUqKeGzsnJYdq0abz++uu8/fbbHDp0iISEBNLT0400sT0ZruivB3pPWUKUBcHn9HjBuZ6murraKQ+DHlrojra2NuLi4oyHJ1x0iIyMjKSpqYnY2FgWLVpEUVERLS0tbNu2jfPnzxMeHk5dXR07duxg6tSp2Gw2p5wGSUlJHDp0iKSkJNLS0vjvf/9LXV2d05x/aWmpURMiPT2d5uZmpJRMnjzZ6eE8e/ZsIiIiiI6O5otf/KKTlbWpqcmouqn7K+ghk3olRrMPgqepBHDf35RfUM9jVVFoRMufDvB54O+Oz6dN2/sEYWFhpKWlWYp60ImPjyc6OpqJEyfS3NxMVVWVoU1nZGRQUVFBZmamEdKk53vXpw50a4L+ENRrPwBG/YeamhrDjJienk58fLzhj7Br1y5DXvO8oB5i2VWnwq4kXLJauKqzBGvCJrvdzrlz59x6fiuAwBScs4xrCmZ3xMfHc8stt5CYmMhHH33EW2+9xeTJkxk4cCApKSns3LnTUqI2nejoaJqbm42pCz1qoqmpCdCmMQsKCkhOTga0xE3Hjh0zHvCjRo0yUsXHxcUZD199qkzP31JXV0dqaqox3aAnS0pJSeHTTz8lLS3NSGikO0Wa8y7MmDGDvXv3snPnTiZPnszevXupqqqisrKSyMhI4yFuPtac60D3hdCx+tBXVr2ex6qi8BbwO0fc81TgVsf28cBn/hAsWOmKRaG2tpba2lr+/ve/G9siIiKIiopiz549AKxfv96YS6ytrWX9+vXGdte8CWafAp38/HzD+bGkpITExEQjrLGoqIjMzEwiIyN58MEHee2117Db7TQ0NFBTU9Pubdxq5cmuJFwyP9B7Iuoh0JSWltLY2OgxU52i5wvO+ZpFixaRlJQEwJQpU4xaDTrXX399h74Oel0IIQRnzpxx8m8YOHAgNpvNmD7UXwoOHjwIaNliBw0axIABAzhy5Ahnz541QjFjY2ONhEQLFixg1qxZbN68mYqKCgYPHmxEHNjtdkaOHElSUpIxJVFUVMRXvvIVALdZF/Xf9rZt24wIB2g/TWEOlRw5ciRpaWlODpCdVaCVVa/nsaooLAP+jKYg3CulPOrYfj19bOqhKxYFHXNMte51rHP+/Hlj6sEc0nT33XczYcIErr32WqdzuYt20P0OsrKyPIZAFhQUsGzZMuMhHx0d7ZQPQW9jpfJkd8tMh2LUQ2dJT09n3759nS4o5K952CCc3w3qgnNWnBmHDRvmtO5a7KijcwghDMuFlNJ4yOtIKTlxQtOZGhoajGiqiIgIJkyYwN69e2lsbCQuLo7k5GRj2iI2NpakpCReeuklw5/gzjvvNEI96+rqeO+995g/f76RtyArK4tRo0ZRVVXFiBEjjMRHzc3NxMXFkZKSQn19PZs3b2by5MkcOHDAuGZKSgopKSntHv42m43IyEjj/OXl5Ybvg9nyoAherNZ6OIyWFMV1+3d9LVCw09bW5rFamDeEEE5pYIcNG0ZtbS0tLVqemYEDB9KvXz8+//nP89577xnt9CqTDz74IK+88oqhIDQ0NBi1IfQESOvXr/dqytczLjY0NLBkyZJ24ZZWkjqZH+7dLTPdmXTRoYrNZjOy5XUGf83DBuH87i+Bc8D3gQcd246ipXb+v0AJpWNl6qG76A9+T5w7dzG5pJSSfv36ERkZydmzZ/nwww8NGevr652qIfbr148dO3YYisuFCxeoqakxcrucO3eOvXv3ctVVV9Hc3ExGRgbNzc2GEpmcnMxf//pX6uqcq3Xr45/dbmfo0KGGYpGSktJuOkHHHGq5Z88eMjIyvGZmVAQXnY79EULECSGGmBd/CBashIWFMWfOnE4do5eW1UlOTqalpYWWlhbCw8O58soriYiIoL6+nsLCQiorKxkzZgxZWVnU1tYyYcIE1q5dC2gP+5UrV2K321mzZg1r1641UkK7mvXLy8tZunQpK1eupKCgwEkmPdxy/fr15OfnG0rCypUrKSwsNJI11dTUGPuh/dSBuXBNV9Cv6SqfVVzl6yx2u53t27dbzo3RU+dKT0/3S8pZf523qzgK1z0spRwFDAIGSSlHSSn/IDt6ggYJ5ge5P3At29y/f38jlNKsyAwfPtxY1x0bBwwY4JQR8oknnqCkpMRwsD579ixPPfUUW7duZf/+/WzdupWXXnqJ5uZmNm/ebCgJKSkpZGRkOCVYOnDgAGfOnCE8PJysrKx2oZWu32HWrFmUl5dTUlJCdHQ08+bNc1Kgfdl/FL7Fah6FFLTCLXOBSPMutLhn66XVQpy2tjY2buzcbEtkZKTRsYUQNDU1UVtbixCClpYWKioqaGhocPIczsjI4J133iEjI4OCggIjvtjcidy9jdfU1BgFY3Rrg7fiMK61IMx/wfM0hKftHTkouu7v7vRDd6MofPmG7ctz+WseNpjnd0O14JyeNM1fmPu8XgHRHcePHzd8JXRLpZ57QUcPtXTN02D+W1VVRVVVFVOmTGHgwIGEhYVx9dVX8+GHH5Kdnc22bdtoa2tj0KBB9OvXD7vdbtnXwJsjYhBauxQOrPooFABxwFfRzIIhoen7i854MAOGkgCa6fD48ePG5wEDBtDQ0MCECRO46qqrePrpp0lKSuLVV1+lrq6OAwcO8Le//c2oJql3xp07d7JkyRKjY+m54XX/hgkTJvDAAw8wbdo0tzLploHy8nJ27dpFTk6O28gBT46OnqYn3D24PTlAuqaY7gq5ubk0NDQYjpmdnb7wpQe1Hk7W1NTk8yiHIPQtUDjoScNHR9dqaGhg3LhxVFZWtps2GThwIElJSRw8eJDGxkZiYmIICwsjOjraSOCUkpJCa2srhw8fpq6uzrBGvPrqq5w/f566ujruvvtu46E+e/Zsdu3axaBBgyw93L0pqiqaIXixqihMB64JhkxpgSYsLIybb77ZcDzsLHr9B50BAwYwe/ZsiouLjXl+PeuZ3nbnzp0AhrdzamqqkQ1Ntxbo/gegeR2XlZXx2muvER0d7bU4zPLly42Mj64PbHfWgY7e4DuySvjagTEhIcH4jrpjZmfw5Ru2zWYjIiKC4uJinztpqbctRUcIIaivr+fMmTO0tbXRr18/40UEtGmGiooKWlpaiI+PN6IUAKZNm0ZNTQ3Z2dns2bOHw4cPI6Vk+PDhHD9+3CkE891330VKyZQpU+jXrx9Tp07t8OHuLpdCZwtFKQKHVUWhEujvT0FChba2NuLj4/n5z3+OlJLTp08THR3NkSNHGDVqFK+//jqXXnopGzdupLa2FpvNxrRp0xBCsGfPHtLS0nj//fcN82F9fT0ff/wxM2fO5O233zZio6uqqgyFYvLkyaxatcpIwQoY+duXLFnCunXr2Lx5MwCTJk0iKyuLxx57zJi3z8vLw263uy0O4zo14brfVSnozINeHxz0fPCerBbe8DSVYd4eTNET/norUm9bwUtXCkZ151qerAq6hVLPt9DW1tau1kRLSwuDBw9m8eLFPPXUU4ZiceDAAWpra9m4cSOZmZns27ePqqoqYzyKiopi/PjxnDhxgoaGBnbv3s2YMWOIiYkhLi6uQyuXrui65lJQhAZWFYXvAP8rhPiWlNI13rlP4RrqNGSI5ss5duxY4GLM9JgxY/jggw9ISkpixIgRAFx99dUMHDiQz33uc5w8eZK33nqLz3/+8xw+fJgRI0Zw2WWXERsbS0tLC1VVVVx66aVGB1y9ejU2m405c+YwZMgQfvOb3/Dxxx/zzDPP8N577yGE4IorriAmJobBgwdz88038+mnn/LII48Y0xagKRjm4jAAixcvpry8HJvN5tTh3T2AO3rQm5WLmTNnUlxcTEREhEerRUdY9ZHwdx4Gq6b/vuhboOg5vE09hIWFER4ebvgleFMoAG655RY2bNhAdHQ0hw8fJj4+3niI19bWMnbsWMPp8cSJE07ZIgFGjBhBbGyspdBfc9SDXtpaETpYVRReRLMolAshLuBSO15KGetrwUIVXZGIiIhg6tSpTvv0MKXw8HBGjhzJ//zP/xAWFuZ2Xj0xMZF7770XgOeee468vDwaGhqM8Kfnn3/eGBCuvvpqACM3/ODBgwG48sormTdvHldeeSUTJ040HBn37NljVKxzLUVrfhh1xX/ArFzoD9Tk5GQiIiK69MbvyVrgbyuCq2KgTP/+QRWc8x1tbW2WIjDq6+t56aWXDGtCVFQUmZmZRnbFpqYmkpKSkFIaU5zDhg1j9OjRCCG4/PLLjemDQ4cOWfKZMSu6vTEMurfTmYRLCh/jLRFLa2srTz31FEII6urqEEIQHx9PS0sLra2tNDY2Gsfr85HHjh0z6ke88MILTJo0icTERIqLi53m73XnSv1vWlqa28ptZjp6o25razNK0ZqVC31wcFU4rL6he1JW/J2y2VUxUKZ/v7GKECk49/Of/zzQIvgEPXRSd1Q8evQol156KQkJCURERLB161YjhPazzz6jsrKSqKgoZsyYQWlpqbJu9UGsJlx63N+CeEII8S1gBTAC7S3ju1LKt7y0nwOsBSahRWiskVL+pSdk9TV6uFJkZCQtLS2GGVDH7NVszgf/wgsvYLfb2bhxIzfddBODBw+mvr7e8MTXnZj0v+Xl5VRUVDBy5EijTKzrw9s8x6jneoeLD3zdiQ+svXF7Op8rgfL2d1UM1ODoN0Ki4Fx3yk0HEt1/Qp9u0EOy3WV/hPbVGm+99VZ27tyJEIJ3333XyKio+kLfwvKvXwjRXwhxtxDit0KIfCHEXUIIvzo4CiG+DPwBeAhIR0vx+qoQItlD+zFAkaNdOvC/wB+FEF/0p5z+YuDAgURHR9PU1OQxQ1xMTIyRQjYhIYH9+/djt9sJDw9nxIgR/Pvf/6auro7du3dTWloKaH4UqampzJ07l+3bt5OWlkZWVpZRJlZvZyYtLc2YwzTv37ZtG8XFxbS1tZGZmUlzczN2u73D5Cnp6emkpqa2O58r5qp2rue02+1s3ryZLVu2eLyO1SQuru10xUCFIvod14JzusdtUBScC1UFQUdKiZSS/v37G+uuxMfHG9UeXX/3evrlrVu3IqUMqmRdip7DasKly4DXgFhgj2Pz14HVQoiFUsr9fpJvOfAPKeVjjvVvCyEWAt8Efuym/b3AUSnltx3r+4UQM9Dyxv/HTzL6jcbGRqfEKIDh7Hj+/HmklDQ2NhIVFQVoGeJ0v4WWlhb27r0YzRoXF0daWpoxPXD77be3qwgHOKVVrampYePGjSxYsIDy8nJqa2vbFXzRc0K0tLQQGRlpOC8CXi0MNpuNRYsWGdYCT5jfcFytEHp+el1ud9ex6lvgqZ3KX+B3VMG5HuCqq67i+PHjVFZWAtp4MHHiRE6cOMGBAwcoLy/36DvgamVQ9D2s+ij8ASgF7pRSngEQQsQCTwK/Bxb4WjAhRCRaBbnfuuzaBMz0cFiGY7+ZjcBSIUSElLLZzTFBi6uSEBYWZmRW02lpaeHUqVMAjB49mpMnTxIREcGJEyfo378/Fy5cYMCAAaSmprJz5052796N3W7nuuuuc6oIt2HDBhYtWkR6erpRBKasrIy6ujpaW1u55JJLmDNnDtOnT3caLLKzs9m4cSODBg1i3LhxgPM8vjclwIo539zGtYKdp5r2drudd999l+bmZqSUzJkzx6McetGb8+fPM2bMmHZ+GsqJ0e+ognM+JCoqivj4eA4fPgxcVAhmz54NaLlUjh07xrx586iqquL66683HBM9oabdFFYVhVnANF1JAJBSnhFC/BR4xy+SQQJaamjXUrMngCwPxwznounS3D7ccT4nD2ohxD1oHtaMHDmS/fs9G0Zmz55NTEyM0eGCCT22WghBQkIC/fr149JLLwWcK1aGhYUxe/ZsIiMj2bdvH3a7nSuuuIKUlBQuXLjA3r17uXDhguEcOWnSJMLDwxkwYADnzp1j4MCB7Nu3zyn3fFNTE6NGjaKpqcmoQ6+/teif9RK6uo+Evq5XwhNCEBUVRWNjIwMGDHD665rnfsqUKYwfP54BAwawb98+hg4dSlhYmFOhrnPnztHa2mocFx0dTWVlpdP1QfPxqKuro62tjf79+5OUlMSBAweorq425I2NjSU7OxubzWb8Ply/S0c0NjZ6/W31ZVTBOWuYf7OgpY0OCwujvr6ekSNHGknaIiMjWbJkCf/5z384cOAAQ4YMYfbs2YZyf91111FTU8Mzzzxj+CgpJUDREVYVhUa0FM6uDHLsC0mklI8CjwJMnTpVTpw40WPbf/3rX8yePZtt27b1lHjdQq8Op9eP0B+8oIVRhoWFGUmihg8fzqlTpxgyZAi7d+8GtDeRK664gmnTpmG323n11VeN/A56sieAP/7xj5w6dYq5c+fS2NholLkGjM+zZs0ypjlc13X0THG634L+13wt8zSA/qZv3q9jtiiEh4czadKkdu3tdjvPPfcclZWVpKSkIKV0KpUrhDC8v13P7/pdOmL//v14+20pNIQQcbj4TUkpTwVGmuBCVxD0l4Irr7zSmOqbOHEiCxYs4MUXX+Smm27CZrPxxS9+kQ0bNhjWN/PvVE8GFx8fr/wNFJawqii8BDwmhPg6Fy0IGcD/Awr9IRhQA7QCw1y2DwOOezjmuIf2LY7z9QkiIyONaQvdOc9cHEbP6w5w+PBhw0ypWx6ioqLIzs7m5MmTgBYVceDAAUBLLGWuZTB27FhOnTpFWFgYycnJpKamkpyczP79+51M+a5RBPq0QUtLC8ePH+fAgQOkpqayYMECRo4cSUtLC0lJSU4DmXkawF24olmRmDt3rrFubqf7aTQ3NxuWj8TERE6dOkVycjKHDh3i0KFDZGRkGI5b5vSzepz51KlT+fTTT0lLSyMhIUH5MnQRVXDOGv369SM9PZ2IiAh27NjhNN2m/+aWLbsYxe7NB2jBggXGX/VbVVihM5kZH0dzPGp1bAtDUxK+63uxQErZJIT4L3Ad8G/Truvw7Ji4Ay3Eysx1wO5Q80/oDnoK1379+tHW1ubW03no0KFcuHCBlpYWI81rUlISERERVFRUGG8dBw8eZMGCBcZ8P0BJSQlHjx5l0aJFzJs3j9raWpqamti8ebPx8NUzvO3du9dwkDS/1dhsNqN2vetDNjIykpKSErKyspwGMteB0fVt3tWf4PXXX+f999+npqaGm266iVmzZrF582ZKSkrIyMggMzMTIQRVVVVGae+xY8dy4MABp1oNZqdP/XsNHjyYuro6ioqK+MpXvqJ8GbqOKjjngfDwcHJycti6dSu1tbXExcWRnp5OdHS0xz5gxtP+hIQEbr/9dn+KruhlWM2jUA/cJIS4FJjg2Ly/B9I5rwWeEEK8C2xHi2oYifYGghDinw75vuJo/xdgmRDi92jWjlnAXcASP8sZlOgWgvDwcFJSUvj000+NfRcuXOD06dPGenJysmGSr6urMwamiooKEhMTmT9/PqA91I8ePerkAHnJJZfQ1tbGiBEjGDduHMnJybS2tjJixAiklB7zJZgVBPOA5motsNvtbNu2jRMnTnD99dcDGNEb+tRKaWmpYdFIS0vDbrcbUR/vv/8+V111FZdccokRVx4ZGcn06dMN60JKSorhi6AX4dKtMU1NTcyZM4fLL7+cpKQkmpqaqKqqoq6uzkjPbSVplcItfbrgXHh4OC0tLcTExBAeHs6oUaOor68nLCyML3zhCyQkJDBu3DgnRVopooqexqpFAQAp5SfAJ36Sxd31nhVCxAM/Q0u4tBfIllLqnmvJLu0rhRDZwMNoIZRHgfullCEXGulLoqOjOXjwoNO206dPM2jQIENZ0PMfmOcvBwwYQH19PVVVVfzzn/8kPj6eiIgI4uLiDCVi3bp1XH/99Zw+fZozZ87Q1NTEp59+SmVlJePGjSMtLY09e/ZQUVHBu+++y9y5cwE8OlS5M+GXlpYaxbCefvpp4uLiDMvFrFmzjLd53a9h9OjRNDc309JyMdP4U089xYwZM5g8ebJhGdm5cydvv/02oFkI9uzZw4wZM4y5X8CwNuh+DHp2uiNHjhAfH8+ECRMMZUO/tkpR2yl6fcE5IQSLFi1i48aNTkWaUlNTyczMpKSkhAULFnj83SjlQBFoPCoKQoj/A34spWxwfPaIlPJ+n0t28dx/Av7kYd9cN9u2Alf5S55QxGw5cEUPoTx27JhRTW7AgAHU1tYyYsQIhgwZghCCyspK4+Fs5vz587z66qtcffXV7Nq1C4CMjAzjzX7v3r3U1dUBzsledIUkLi7Oaa5fz/7W3NxsKBXp6enU19ezb98+6urqqKurIz4+3rAcNDc3G7nq9YIzulUgMjKSpqYmLly4QElJCXv27GHgwIFUVVUxZ84cZs6cyf79+6mrq6OkpMTITAkYkRyA4bug79enIUpKSqioqCAzM1Mlo+kavb7g3IABA5g8eTLjxo1j586dtLS0EBERYYQaq2kARbDjzaIwGYgwffaEmlMMQYQQRolsPTW0njJad3zUnR5HjhzJyZMnaWpqIj4+3vCBOHv2LAMGDDCKXeklZ48ePcqhQ4eoq6szqmqOGTOGGTNmGNfXHar0h/HGjRu5/fbbDWXCrFTYbDbi4uI4f/48Y8eORUpJZWUl5eVa5l89OiEhIcEo4DR58mRjikT/DrqSUVdXx5gxY5g+fTqgmX/tdju1tbWkpaUZb3B2u52WlhY++ugj6uvrGTNmjGHpWLRokZFvYuTIkcyYMUM5hnWNXltwLjIy0rAmgLNfjkIRSnhUFKSU89x9VoQu7mrZDx482DCHmvcPGjTIcHQ8dOiQoRycPn3aMOmbH9rjx4/n3LlzRvU6vaZEbGys8aZtfpDqDlU1NTW8/PLLNDU1UVNTY5j+Xd/MXf0W9CiEPXv2GAmV7Ha7ERKmO2GOHDnS+F4lJSVERUVx/vx5kpOTsdlsbN++nZKSEsaMGUNlZSV79+41LBk2m43w8HDq6+tJSkoCMKI9zE6XqampPvsf9UF6VcG55ORkkpKSDB8YpTwqegOd8lEwI4RIBQ5LKUM2j4Iv0b3lgxVXJUFKyYABA5BSkpiYSHV1tdP+06dPExcXB2g1J/QiMsnJydTW1jJw4EAj74I+MOrtxowZw7x58zqcewXtYXzmzBmnCAJPKZ/N2/VcDOboiO3bt1NRUWHUpBg9erRTZEVkZCRpaWlGJjq73W44Kurhkq6KlO78eP78eY4cOWJYPqB9pshAziOHanhmIAvO+ZL+/fszefJk5s2bF1L3X6GwgtVaDw8B5VLKx4U2cm5CK+ByWghxvZTSX9kZQ4Lk5GS++MUvkp+f3yPXGzp0qJHjwCquD8CwsDAaGxuNjG46I0aMYMyYMRw7dozKykpSU1NJTEzk6FEts+6pU6c4ffo0p0+fJjJSC3sPDw+nqamJs2fPkpqaakQ3uM69mvMR6A/r0tJSw4dh+PDhTu08PfTM54H21gbz+XXMioauuJgVDf3tz9WSMX36dCIiIkhOTjYUH/M5rdSr6AlCOTzTUVzuduAytKnMfcB6KeUFrwcGCebfvELRG7FqUbgd+LLj8/XAFOAax/b/Bfrs1MSoUaP48pe/7PdBQg+jApg4cSITJ040Sr7GxsZy5swZ4uLiDD+D6Oho4uLinBSBfv36MXnyZA4ePGi0c2X06NFcd911Tg9rgCNHjlBVVUVqaqpRS2Lu3LmMHTvWCCMcM2aM1wHTXAlSz73Q3NxsmGv19NgdPfTMDo/Tp09v96C26iXeUV4G13O5czoLFo90dwmoQoEAFpzzCWPHjlVKgqLXY1VRGAYcdnzOBv4lpXxXCHEK2O0XyUKAiIgIlixZ0u1BYvDgwcycOZPNmzezcOFCjh8/bvgFnD17lltuuYWkpCTeffddpJRMnz6dDRs2ANrbzOjRoykuLmbq1KkkJycbqVzLysqcFIXW1lYSEhKIjY01qi4OGzaM2tpaQwnRrQSuD8Avf/nLRl16gAMHDjB27FjD6W/fvn1kZGR4vRdmU/2YMWM4fPiwMV0zfvx449iOHnpmh0ezUgHeK1a6EiwPeV8Qwt+lxwvO+QrdkqiUBEVvx6qiUAukoCkL84EfmY4XfpArJLjzzjudBgk9W58nwsLC+OY3vwlAUVER8fHxREdHG2bvqVOnAhi14V3RnezAfRpW11SueoKlQYMGMX78eKKiooyHr163QT8uPDzccMByh+68V1xc3C4U0GazERMT0+GAaTbV2+123n77baO2QmcsAu4cHq1WrFQEHYEoONdtesqSqFAEA1YVhf8ATwshPgaGcLH86xSgV8Y+d8TgwYO55JJLnLalpKR4VRRGjhxpzI9/5Stf8djOCq5pWN09WKdPn86RI0eoqKigrq6OuXPnOlWRe+qpp6ioqODw4cOWChz5oi697gfw3HPPAdo962zImDvHRnefFSFByBWci4mJ8YklUaEIFTqukauxHM1E+BFwnZRSTy82Aq2WfJ9Dzx1gRk9sJIQgPj4e0PwHpkyZwpgxY7jpppt6VEb9DV7PWKgXSNJZsGABY8aMISMjw9JbuP6A7u4AWVpaajhKmnMrKPokesG5WUKIfo5lNv4tONctli5dqpQERZ+iQ4uCECIC+BXwiCl1MgBSyof9JViw84UvfKHdtuzsbCMt8aRJkwzzeCAHFW+e+QkJCd22bHQFX1gmIHhCAoNFjhClxwvOdYehQ4eqFN2KPkeHioKUslkI8S08pFHuq7gbLBISErj77ruD7qERbI5uvpInWEICd+7cSUlJCU1NTSrzXicJYMG5LjFu3LhAi6BQ9DhWfRQ2Ap8D/u5HWXoFwfZQ7s0ES0igHgmi/1V0np4uONdV9BBehaIvYVVReAN4SAhxBfBfoMG8U0r5vK8FC2ZGjRoVaBEUBI9SpidlCrTCEioES8G5rhAsVkKFoiexqiisc/x112kl0M834oQGeqEjhQKCR2EJIUKy4NyQIUMCLYJCERAsKQpSSqvREX0C5amvUHSdUC04p6cMVyj6GkoB6ALK/KhQ+A8hRKoQYkCg5XBF+Sco+iqWFAWh8S0hxD4hhF0IMdax/UdCiC/5V0SFQtFbEUI8JIRY6vgshBCvAx8Dx4QQ1wRWOmfUC4Kir2LVovAd4GfAozinbD5CL6snr1AoepTbgXLHZ3PBuX+iFZxTKBQBxqqicC/wdSnlH4AW0/b3gEk+l0qhUPQV3BacA/4IdCqMxGH1rBRCNAoh/iuEuNbHsioUfRKrikIKsNfN9mYgynfiKBSKPoZecA60gnNvOD53quCcEOLLaGnmH0JTMN4GXhVCJPtOVIWib2JVUTgAXOVmezZa/QeFQqHoCnrBudfpXsG55cA/pJSPSSn3Sym/DRwDvulLYRWKvojVPAq/BdYJIWxoWn6GEOJOYCVwt7+EUygUvZ7lwEE0q8LKrhScE0JEAlejjVNmNgEzfSOmQtF3sZpHoUAIEY5m1rMBTwBHgfullM/6UT6FQtFL8WHBuQS0pG8nXLafALLcXPce4B7QSr/v37/f44lnz55NTEwMs2fP9tqur9LY2Kjuiwd6072xalFASvkYWjnYBCBMSnnSX0IJIYYAq4Hr0N40aoCXgZ9JKWu9HHcXUOBmV5SUMihr2ysUfZVAFZyTUj6KFsHF1KlT5cSJEz22/de//sXs2bPZtm0beXl5PSViyLB//3683b++TG+6N51KuCSEGIcWujRdz6XgJ0YCSWhTG5OBO4BMYL2FY+1oZktjUUqCQhG06AXnukMNWonqYS7bhwHHu3luhaLPY8miIISIB/4G5ABtFzeLl4G7vb3ldwUp5V7gFtOmCiHECuBlIUSslPKM98OlGhwUitCg2wXnpJRNQoj/olkg/23adR2as6RCoegGVqce/gqkAtcCOx3bZqA5Gz2G80PdX8QCF9AsBt6IEkIcQpuzfB94QEpZ6mfZFApF1/BVwbm1wBNCiHeB7Wi5X0YCf+m2hApFH8eqorAA+LyUcodp23YhxDeAYt+L5YwQIg54EHhMStnipWk5WhTGB8BAtIyS24UQVzrq3bueVzk1+Yje5Ljja9S98YyvCs5JKZ91WD5/hjbluBfIdnWSVCgUnceqolCNi0nQgR0tYYolhBC/BH7aQbN5UsotpmNigJfQ0kWv9HagQ5ExlBkhxNtoVoVv4+aNpTNOTZs3byYtLY3y8nI+//nPd/AV+h69yXHH16h70zNIKf9EDztGKhR9AauKwi+A3wsh7pRSHgEQQiQBv3Pss8rvgSc7aFOlf3AoCUWO1Rs765QopWwVQuwGLu3Mce6oq6tz+qtQKLqPEEKgJUW6DxgDXC6lPCCE+BFwQEr5r0DKFxUV5fRXoeiLWFUUvguMBg4KIY44tiUBjcBQIYTxti6lvMLTSaSUNWgeyh0ihBgIvIqW4GmhlPKcRVnN5xDAFWhTEd1i+vTpTn8VCoVP+A6apfA3wK9N2/WCcwFVFLQh5OJfhaIvYlVReM6vUrjgUBI2oTkw3gxECyGiHbtPSSmbHO3eAN6VUv7YsZ4HvAN84jj2fjRFodtpXEtLS7n66qspLS3luuuu6+7pFAqFhl5w7hXH1KROUBSca21tdfqrUPRFrGZmXO1vQVy4Gi1fA2i16c3MA7Y4Po8DPjPti0PzORgOnAZKgUxHNbpukZaW5vRXoVD4hKAuOKf7ligfE0VfxmoehUQAKWW1Y30y8GVgn5TSShKkTuFwZuzQ1ielHO2y/j3ge76WB+C6665j3759ZGRk+OP0CkVfRS845xqdEBQF51S/VyisTz38C62+w98dKZxL0Go9fFsIMVJK+Tt/CRgs2Gw2YmJisNlsgRZFoehNBHXBOdXvFQrrisIVaHP/ALcCFVLKaUKIm4B8tOiHXo3dbufcuXPY7XY1aCgUPiLYC86pfq9QWK/1EAXoUQdZQKHj83vAJb4WKhgpLS3lzJkzlJaqJI8KhS+RUj4mpUwBhgLDpZSjpJR/C7RcoPq9QgHWFYVPgFuEEJcA89EiEkArulLvB7mCjvT0dGJjY0lPTw+0KApFr6MHC851CtXvFQrrisJqtDjng8A7Ukq93sMCtMiCXo+aq1QofI8QIl4I8QLay8gLjuUTIcSLjpTMAUX1e4XCoqLgqOCWDEwFFpp2FQPL/SCXQqHoG5gLzg1wLJloWRofC6BcCoXCgVVnRqSUJ4ATLtt2emiuUCgUVghowTmFQtExHhUFIcT/AT+WUjY4PntESumuRKxCoVB0hE8KzikUCv/hzaIwGYgwffaE9J04CoWij+GrgnMKhcJPeFQUpJTz3H1WKBQKH/JdfFBwTqFQ+A/LPgoKhULhB3q04JxCoeg83nwU/m71JFLKgKdaVSgUoUcACs4pFIpOIqR072IghHjJZVMm0AbscaxfjhZeWSKlzPGbhD2EEKKa9oVpXEkAanpAnFBE3RvPhOK9SZFSJvr7Ij1dcM7N9VW/7x7q3ngm1O6Nxz7vzUfhC/pnIcSPgfNArpSywbEtGvgbFxWHkMbKoCiE2C2lnNoT8oQa6t54Rt0brwS04Jzq991D3RvP9KZ7YzUz4/3AKl1JAHB8fhD4tj8EUygUfQJ3BecmAV8BvhEwqRQKhYFVRSEGGOlm+wi0im8KhULRFfp8wTmFItixqij8BygQQiwWQox2LIvRph6e9594QcejgRYgiFH3xjPq3ngmFArOqf+fZ9S98UyvuTcenRmdGgkRhZYA5W4uJmFqQVMUfiCltPtNQoVC0WsRQtwCrEfzl3pDSjnfsf2nwCwpZXYg5VMoFBYVBaOx5sA4zrH6qdlnQaFQKLqCEGIY2tTmB1LKNse2GcBpKWVZQIVTKBSdUxQUCoVCoVD0Laz6KPR5hBDfEkJUCiEahRD/FUJcG2iZAo0Q4sdCiF1CiDNCiGohxEtCiMsDLVew4bhPUgixLtCyBANCiP9zWCf1zx6XIJBV9XsXVL+3Rm/q90pRsIAQ4svAH4CHgHTgbeBVIURyQAULPHOBPwEzgc+h+a0UCyGGBFKoYEIIcQ1wD/BhoGUJIlwLznlaAvrwUf3eI3NR/d4rva3fq6kHCwghdgIfSim/btr2CfCclPLHgZMsuBBCxACngZullK6ZPfscQohBaGF+XwPygL1SymWBlUphFdXvraH6vTO9sd8ri0IHCCEigau5GLalswlNo1ZcZCDab6ou0IIECY+iPVQ2B1oQRedQ/b5TqH7vTK/r96p6ZMckAP2AEy7bT6AliFFc5A/A+8COAMsRcIQQXwdSgTsCLUuwESIF51S/t47q9w56a79XioLCJwgh1gKzgdlSytZAyxNIhBBpaPPas6WUzYGWJwhxra/gseBcTwql6Dyq31+kN/d7pSh0TA3QipYpzsww4HjPixN8CCEeBhYD86SUBwItTxCQgfZGuk8IoW/rB2QKIe4FoqWUFwIlXKAJkYJzqt93gOr37ei1/V45M1rA4dT0gZTyHtO2j4H/9HWnJiHEH9DKAs+TUu4PtDzBgBAiDhjlsrkALV3xQ2gllFXHA4QQx4DPSyk/ctk+CS1T4/DASKb6vTdUv29Pb+73yqJgjbXAE0KId4HtwL1omeT+ElCpAowQ4hHgTuBmoE4IoQ/q56SU5zwe2MuRUtbjUqdACNEAnJJS7g2ETEGMXnDuI5ftwVBwTvV7N6h+757e3O+VomABKeWzQoh44GdoA9heIFtKeSiwkgWcbzn+vuGyfTWwqmdFUYQoesG5FVwsN30N8BsCXHBO9XuPqH7fx1BTDwqFImCognMKRfCjFAWFQhFwVME5hSJ4UYqCQqFQKBQKj6jMjAqFQqFQKDyiFAWFQqFQKBQeUYqCQqFQKBQKjyhFQaFQKBQKhUeUoqAIaoQQUghxa6Dl6AghxE1CiE+EEC1CiH8EWh6FIpRR/T64UFEPiqDGkfWtLthzpAshaoC/An9Ey1B3OsAiKRQhi+r3wYXKzKjoMkKISCllkz+vIaUM+gI8jhzv8cBGKeWRAIujUPgV1e81+lS/l1KqxWUBtgB/RssYdwqoBr4D9AceQcvnXQXc6XJcEvAMUOdYXgEuNe0fB7yIVn2uAXgPuNHlHAfRUsb+P+AMcBhYYUHmLwD/BRqBSuBXQKRj352AHZhgav+/wGfAYNN1VwFPAuccMv7A5RoSuA8ttW4D8NuOru3YfwvwIVqVwFPAVmCYY98ljntyyiFjGbDY5Zq3mtYnA8Wmc/0DGGTa/w/gZcf/64jj/1AA2ExtMtHSBZ8DTgPvApd7ubeDgccd5zrvuP4kx765DhnNy1wP5xkGFDrOcQjIRUsLvMrUZhDwKHASOOu4V1NN++9yyP15x7ENwGZgjNXfQ0f/k766oPq96veq37u/F4HunMG4oA0YZxwd6FLg+44fwquOH2Iq8CBwARjhOMYGfOz4wV4BTEAzSR3Sf6zAlWiFZSY7zvFToAnnjnwQqAWWOdp823HtDC/yLnDIm4s2KM0DynF0aEebp9EGqEjHj7wZrfKb+bpnHDKNB77hkO0Wl857EvgaMBYY09G1geGO83wfGA1c7jheHzBeAl533JsxwEJgobsBA4gGjgIvOO7hHMc9/4/LgHEaeAyYCMxHG+B/7Ngfjtbxf+uQdwJwGzDRy/19EW0gy3RctxBtsI1y3M/LHHLe4vi+kR7O8xrwAVo52iloufLP4hgwAAFsQ3vQTOfi7+wMF39ndzn+d8WONlcApWhvNZZ+Dx39T/rqgur3qt+rfu/+XgS6cwbjgjZg7DCtC7S3i0LTtgjHTdd/zHejlRMVpjb90Dr/l7xc6x3gZy4dd71Lm0/MbdycowR4wGXbzWgaqO6HMshx7kcdP/bfuLQ/CLzusu2vwDbTugT+2JlrA1c5jkvxIPuHQJ6X72YeML6ONhgMNO2f62iT6lj/h+P79TO1eQwodnwe4mg/x+Jv4VJH+0zTtkEOOb7mWE/AyxuFo02ao801pm2XAK1cHDA+57hvUS7Hvg+sdHy+y3GeNNP+29EeXvr/ulv/k766oPq96vcXj1X93rQoHwXPfKh/kFJKIcRJYI9pW7MQog4Y6th0NZpmfFYIYT6PDUcOe0c++zzgRrRqdBHAAPO1XK/t4KjpOu64GpguhPihaVsYmuY7HDgmpTwthFiKNhi+j2bmdGWHm/VbXLbt7uS1P0DTgvcKITY5Pj8npax2tP0D8BchxEI0TXuDlPK/Hr7nROBDKeVZ07a3gTY07b7Cse0jKWWrqc1RYAaAlPKUwzt5oxDiDcc1n5NSVnm5Zhume+O4l3sc17TKBMd5jPsnpfxMCHHU1OZqtN9LtctvaAAX6yAAXJBSlrt8v0g0U+kpuv8/6cuofq/6vX5N1e8dKEXBM80u69LDNj3ENIz/3975hFhVR3H88x1wEzkghETUQgls0SaCmv4KERIKpqtyU6kQtAoihlo0hikiSSQEjYJEg7ZIC2YKYkDaSGktDBc5oxNS4MaQsBYTRHVcnN+jy+Pe53u+x9S8vh8Y3h3und/5zf298+Oc8/v+7k1HfLamrV/K5wGyxPYqmS0sAlPkYN/IdqetrCPkK16P15yrfgkeJyPZ1cAomfX0SvvLejrajoi/JG0gXx28AdgJ7JO0PiLORcQRSbPARuBJ4GtJ+yLizR77FZXjjvcvIrZLepcci83AXklbImK2D5uDYAS4AjxWc+63yvGfDf2ofhdvekxutvNDgv2+Hvt9vc1B8J/3ewcKg+MssA24GhHXGq55FJiKiE8AJLUixosDsH1PRPzQdIGkB4E3gK3ABFmWa88axmp+n+vXdmQN7DRwWtJu4HvgGTLCJSIuk6XRwyUafpn699rPATskraxkFw+TDnKjfrb36Vyxv1/SF8DzQN2EMVfaf4gs7SFplFyz/KAHk/OlnfuBb0o7dwJ3VK45Swqf/o6IS738P230PSama+z3DdjvgSHxewcKg+MYmTFMS5og1dF3AU8DkxGxQE4MWyVNk9HvLrK81C+7gc8l/QR8TEae9wIPRMS4pFtJVfNkRHwmaR74TtLOiDhSaWdM0uvACXIN8DlyHawf22NkxjBLRs33kfflPICkg6RY7CKZ7TzVOlfDMTJinir3eBWpEv+0k3NUkbSGFGzNkOrotaQw6P266yNioYzXIUkvkgKpvWSk/1E3Nks7F0oGNSnpJVKV/DaZXbYyg5PAV+R3aJycZG4n78nJiDjVpbm+xsT0hP3eft/IsPi9n8w4ICJikSzxXSJLP/Pk1ppVpNoW4BVSPXyKdJIz5bhf27PAJlLl+m35eY2ctCDXA/8Axsv1C2T0flDS3ZWm3uEfNe0eYCIiTvRp+1fgEXLr0gK59eytiDhazo+QDys5T6qgr5BRfp2tRVLZO1rsTJPR8Y5OfWxjkVR3HycnqQ/JiWh/h7/ZXuzNlM9bSIX27z3YhRQkXSbXi2eK3Z/JyaMV7W8EviQzvwukw68j1yO7YgBjYrrEfm+/74IXWOZ+7yczGgAk/Qi8FxEH/u2+/F+QdBs5EWxrlaWNWUrs90vPcvR7Lz0Ys0RIegJYSaroV5OlzKvkPmtjzBAyDH7vQMGYpWMFWdpdS5ZCz5D7tNsV5caY4WHZ+72XHowxxhjTiMWMxhhjjGnEgYIxxhhjGnGgYIwxxphGHCgYY4wxphEHCsYYY4xpxIGCMcYYYxq5Dl3t+qww9jfVAAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 576x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "filter_result = sc.pp.filter_genes_dispersion(\n",
    "    adata.X, min_mean=0.0125, max_mean=3, min_disp=0.5)\n",
    "sc.pl.filter_genes_dispersion(filter_result)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "adata = adata[:, filter_result.gene_subset]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/ag16115/anaconda3/lib/python3.8/site-packages/scanpy/preprocessing/_simple.py:373: UserWarning: Received a view of an AnnData. Making a copy.\n",
      "  view_to_actual(adata)\n"
     ]
    }
   ],
   "source": [
    "sc.pp.log1p(adata)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "regressing out n_counts\n",
      "    sparse input is densified and may lead to high memory use\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/ag16115/anaconda3/lib/python3.8/site-packages/statsmodels/tsa/base/tsa_model.py:7: FutureWarning: pandas.Int64Index is deprecated and will be removed from pandas in a future version. Use pandas.Index with the appropriate dtype instead.\n",
      "  from pandas import (to_datetime, Int64Index, DatetimeIndex, Period,\n",
      "/home/ag16115/anaconda3/lib/python3.8/site-packages/statsmodels/tsa/base/tsa_model.py:7: FutureWarning: pandas.Float64Index is deprecated and will be removed from pandas in a future version. Use pandas.Index with the appropriate dtype instead.\n",
      "  from pandas import (to_datetime, Int64Index, DatetimeIndex, Period,\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "    finished (0:00:37)\n"
     ]
    }
   ],
   "source": [
    "sc.pp.regress_out(adata, 'n_counts')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "sc.pp.scale(adata, max_value=10)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# PCA"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [],
   "source": [
    "adata.obs['clusters'] = labels\n",
    "sc._utils.sanitize_anndata(adata)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [],
   "source": [
    "list_labels =  np.unique(labels).tolist()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {},
   "outputs": [],
   "source": [
    "ordered_cats = list_labels"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {},
   "outputs": [],
   "source": [
    "colors = {\n",
    " 'Endoderm': 'purple',\n",
    " 'Epidermal' : 'dodgerblue',\n",
    " 'Forebrain / Optic' : 'lightsalmon',\n",
    " 'Germline' : 'darkolivegreen',\n",
    " 'Hindbrain / Spinal Cord' : 'firebrick',\n",
    " 'Mesoderm' : 'teal',\n",
    " 'Midbrain' : 'pink',\n",
    " 'NaN' : '#8A8A8A',\n",
    " 'Neural Crest': 'darkorange' ,\n",
    " 'Other' : 'yellow',\n",
    " 'Pluripotent': 'mediumorchid'\n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "adata.uns['clusters_colors'] = [colors[clus] for clus in list(adata.obs['clusters'])]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "metadata": {},
   "outputs": [],
   "source": [
    "Y = np.array(adata.to_df())\n",
    "n,p = Y.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "5079 5498\n"
     ]
    }
   ],
   "source": [
    "print(n,p)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "## dimension selection\n",
    "# rmin = 1\n",
    "# rmax = 50\n",
    "# ws = wasserstein_dim_select(Y,rmin = rmin, rmax = rmax)\n",
    "# dim = np.argmin(ws) + rmin\n",
    "# print(f'Dimension selected: {dim}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "metadata": {},
   "outputs": [],
   "source": [
    "dim = 29"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "metadata": {},
   "outputs": [],
   "source": [
    "zeta = p**-.5 * pc_scores(Y, dim)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "metadata": {},
   "outputs": [],
   "source": [
    "adata.obsm['PC_scores'] = zeta"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "# ## plot after applying PCA\n",
    "# fig,ax = plt.subplots(1,1,figsize = (10,10))\n",
    "# for t in ordered_cats:\n",
    "#     idx = np.where(adata.obs['clusters'] == t)[0]\n",
    "#     ax.scatter(zeta[idx,0],zeta[idx,1], marker = 'o',edgecolor='black', linewidth=0,s = 80,label = t,c= colors[t]);\n",
    "# ax.set_title(f'PCA',fontsize=25)\n",
    "# ax.axes.xaxis.set_visible(False)\n",
    "# ax.axes.yaxis.set_visible(False)\n",
    "\n",
    "# ax.legend(loc='upper center', bbox_to_anchor=(1, -0.02),\n",
    "#           fancybox=True, shadow=True, ncol=8)\n",
    "\n",
    "# # fig.savefig(f\"planaria.pdf\", bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# method comparison"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "method_comparison_df = pd.DataFrame()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### inner products"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%%time\n",
    "ip_clust = AgglomerativeClustering(affinity = ip_affinity, linkage = 'average',distance_threshold=0, n_clusters=None)\n",
    "ip_clust.fit(zeta);"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%%time\n",
    "ances = {}; desc = {}\n",
    "ip_ranking = np.array([get_ranking(ip_clust,t) for t in range(n)])\n",
    "\n",
    "ip_kt_z = [kendalltau(ip_ranking[i], true_ranking[i]).correlation for i in range(ip_ranking.shape[0])]\n",
    "np.mean(ip_kt_z)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%%time\n",
    "ip_clust = AgglomerativeClustering(affinity = ip_affinity, linkage = 'average',distance_threshold=0, n_clusters=None)\n",
    "ip_clust.fit(Y);"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%%time\n",
    "ances = {}; desc = {}\n",
    "ip_ranking = np.array([get_ranking(ip_clust,t) for t in range(n)])\n",
    "\n",
    "ip_kt_y = [kendalltau(ip_ranking[i], true_ranking[i]).correlation for i in range(ip_ranking.shape[0])]\n",
    "np.mean(ip_kt_y)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### ward"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%%time\n",
    "ward = AgglomerativeClustering(linkage=\"ward\", distance_threshold=0, n_clusters = None)\n",
    "ward.fit(zeta)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%%time\n",
    "ances = {}; desc = {}\n",
    "w_ranking = np.array([get_ranking(ward,t) for t in range(n)])\n",
    "\n",
    "w_kt_z = [kendalltau(w_ranking[i], true_ranking[i]).correlation for i in range(w_ranking.shape[0])]\n",
    "np.mean(w_kt_z)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%%time\n",
    "ward = AgglomerativeClustering(linkage=\"ward\", distance_threshold=0, n_clusters = None)\n",
    "ward.fit(Y)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%%time\n",
    "ances = {}; desc = {}\n",
    "w_ranking = np.array([get_ranking(ward,t) for t in range(n)])\n",
    "\n",
    "w_kt_y = [kendalltau(w_ranking[i], true_ranking[i]).correlation for i in range(w_ranking.shape[0])]\n",
    "np.mean(w_kt_y)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### upgma"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%%time\n",
    "average = AgglomerativeClustering(linkage=\"average\", distance_threshold=0, n_clusters = None)\n",
    "average.fit(zeta)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%%time\n",
    "ances = {}; desc  = {}\n",
    "a_ranking = np.array([get_ranking(average,t) for t in range(n)])\n",
    "\n",
    "a_kt_z = [kendalltau(a_ranking[i], true_ranking[i]).correlation for i in range(a_ranking.shape[0])]\n",
    "np.mean(a_kt_z)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "sem(a_kt_z)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%%time\n",
    "average = AgglomerativeClustering(linkage=\"average\", distance_threshold=0, n_clusters = None)\n",
    "average.fit(Y)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%%time\n",
    "ances = {}; desc  = {}\n",
    "a_ranking = np.array([get_ranking(average,t) for t in range(n)])\n",
    "\n",
    "a_kt_y = [kendalltau(a_ranking[i], true_ranking[i]).correlation for i in range(a_ranking.shape[0])]\n",
    "np.mean(a_kt_y)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### cosine"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%%time\n",
    "cosine = AgglomerativeClustering(affinity = 'cosine', linkage = 'average', distance_threshold=0, n_clusters = None)\n",
    "cosine.fit(zeta)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%%time\n",
    "ances = {}; desc  = {}\n",
    "cs_ranking = np.array([get_ranking(cosine,t) for t in range(n)])\n",
    "\n",
    "cs_kt_z = [kendalltau(cs_ranking[i], true_ranking[i]).correlation for i in range(cs_ranking.shape[0])]\n",
    "np.mean(cs_kt_z)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%%time\n",
    "cosine = AgglomerativeClustering(affinity = 'cosine',linkage = 'average', distance_threshold=0, n_clusters = None)\n",
    "cosine.fit(Y)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%%time\n",
    "ances = {}; desc  = {}\n",
    "cs_ranking = np.array([get_ranking(cosine,t) for t in range(n)])\n",
    "\n",
    "cs_kt_y = [kendalltau(cs_ranking[i], true_ranking[i]).correlation for i in range(cs_ranking.shape[0])]\n",
    "np.mean(cs_kt_y)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "method_comparison_df['ip_sem'] = [sem(ip_kt_z),sem(ip_kt_y)]\n",
    "\n",
    "method_comparison_df['ward_sem'] = [sem(w_kt_z),sem(w_kt_y)]\n",
    "\n",
    "method_comparison_df['upgma\t_sem'] = [sem(a_kt_z),sem(a_kt_y)]\n",
    "\n",
    "method_comparison_df['cosine_sem'] = [sem(cs_kt_z),sem(cs_kt_y)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "method_comparison_df['ip_z'] = [np.mean(ip_kt_z),sem(ip_kt_z)]\n",
    "method_comparison_df['ip_y'] = [np.mean(ip_kt_y),sem(ip_kt_y)]\n",
    "\n",
    "method_comparison_df['w_z'] = [np.mean(w_kt_z),sem(w_kt_z)]\n",
    "method_comparison_df['w_y'] = [np.mean(w_kt_y),sem(w_kt_y)]\n",
    "\n",
    "method_comparison_df['a_z'] = [np.mean(a_kt_z),sem(a_kt_z)]\n",
    "method_comparison_df['a_y'] = [np.mean(a_kt_y),sem(a_kt_y)]\n",
    "\n",
    "method_comparison_df['cs_z'] = [np.mean(cs_kt_z),sem(cs_kt_z)]\n",
    "method_comparison_df['cs_y'] = [np.mean(cs_kt_y),sem(cs_kt_y)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# method_comparison_df.to_csv('method_comparison.csv', index=False)\n",
    "# method_comparison_df"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### hdbscan"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import hdbscan"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import sys\n",
    "sys.setrecursionlimit(10000)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "hdbscan_ = hdbscan.HDBSCAN()\n",
    "hdbscan_.fit(zeta)\n",
    "\n",
    "hdbscan_.single_linkage_tree_;\n",
    "hdbscan_.children_ = np.array(hdbscan_.single_linkage_tree_.to_pandas()[['left_child','right_child']], dtype=int)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%%time\n",
    "ances = {}; desc = {}\n",
    "hdbscan_ranking = np.array([get_ranking(hdbscan_,t) for t in range(n)]) \n",
    "\n",
    "hdbscan_kt_z = [kendalltau(hdbscan_ranking[i], true_ranking[i]).correlation for i in range(hdbscan_ranking.shape[0])]\n",
    "np.mean(hdbscan_kt_z)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "hdbscan_ = hdbscan.HDBSCAN()\n",
    "hdbscan_.fit(Y)\n",
    "\n",
    "hdbscan_.single_linkage_tree_;\n",
    "hdbscan_.children_ = np.array(hdbscan_.single_linkage_tree_.to_pandas()[['left_child','right_child']], dtype=int)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%%time\n",
    "ances = {}; desc = {}\n",
    "hdbscan_ranking = np.array([get_ranking(hdbscan_,t) for t in range(n)]) # range(n)\n",
    "\n",
    "hdbscan_kt_y = [kendalltau(hdbscan_ranking[i], true_ranking[i]).correlation for i in range(hdbscan_ranking.shape[0])]\n",
    "np.mean(hdbscan_kt_y)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "method_comparison_df['hdbscan_z'] = [np.mean(hdbscan_kt_z),sem(hdbscan_kt_z)]\n",
    "method_comparison_df['hdbscan_y'] = [np.mean(hdbscan_kt_y),sem(hdbscan_kt_y)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "method_comparison_df.to_csv('method_comparison_df.csv', index=False)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Appendix method comparison"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "linkage = ['complete','single']\n",
    "metric = ['euclidean', 'cosine']\n",
    "combs = list(itertools.product(linkage, metric))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "method_comparison_df_v2 = pd.DataFrame(columns = ['linkage','metric',\n",
    "                                                  'zeta_mean','zeta_se','Y_mean','Y_se'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "for i in range(len(combs)):\n",
    "    print(combs[i])\n",
    "    \n",
    "    on_zeta = AgglomerativeClustering(affinity = combs[i][1], linkage = combs[i][0],distance_threshold=0, n_clusters=None)\n",
    "    on_zeta.fit(zeta);\n",
    "    ances = {}; desc = {}\n",
    "    on_zeta_ranking = np.array([get_ranking(on_zeta,t) for t in range(n)])\n",
    "    kt_z = [kendalltau(on_zeta_ranking[i], true_ranking[i]).correlation for i in range(on_zeta_ranking.shape[0])]\n",
    "    \n",
    "    on_Y = AgglomerativeClustering(affinity = combs[i][1], linkage = combs[i][0],distance_threshold=0, n_clusters=None)\n",
    "    on_Y.fit(Y);\n",
    "    ances = {}; desc = {}\n",
    "    on_Y_ranking = np.array([get_ranking(on_Y,t) for t in range(n)])\n",
    "    kt_Y = [kendalltau(on_Y_ranking[i], true_ranking[i]).correlation for i in range(on_Y_ranking.shape[0])]\n",
    "    \n",
    "    \n",
    "    new_row = {'linkage': combs[i][0],'metric':combs[i][1],\n",
    "               'zeta_mean': np.mean(kt_z),'zeta_se': sem(kt_z),\n",
    "               'Y_mean': np.mean(kt_Y),'Y_se': sem(kt_Y)}\n",
    "    method_comparison_df_v2 = pd.concat([method_comparison_df_v2,pd.DataFrame(new_row, index=[0])]).reset_index(drop=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "method_comparison_df_v2.to_csv('method_comparison_v2.csv', index=False)"
   ]
  }
 ],
 "metadata": {
  "anaconda-cloud": {},
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.8.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
