{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "b9e2d729",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "MPS available: False\n",
      "CUDA version: 11.7\n",
      "MPS not available — running on CPU instead.\n",
      "CPU cores: 48\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "import sys\n",
    "import json\n",
    "import logging\n",
    "import argparse\n",
    "import torch\n",
    "from scipy.sparse import issparse\n",
    "import numpy as np\n",
    "import anndata as ad\n",
    "import pandas as pd\n",
    "\n",
    "print(\"MPS available:\", torch.backends.mps.is_available())\n",
    "print(\"CUDA version:\", torch.version.cuda)\n",
    "\n",
    "# Show MPS device info safely\n",
    "if torch.backends.mps.is_available():\n",
    "    print(\"MPS backend is active on macOS (Metal).\")\n",
    "    print(\"Device: Apple GPU via Metal Performance Shaders (MPS).\")\n",
    "else:\n",
    "    print(\"MPS not available — running on CPU instead.\")\n",
    "\n",
    "print(\"CPU cores:\", os.cpu_count())\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "fe4f5248",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "AnnData object with n_obs × n_vars = 762039 × 1000\n",
      "    obs: 'size_factor', 'cell_type', 'replicate', 'dose', 'drug_code', 'pathway_level_1', 'pathway_level_2', 'product_name', 'target', 'pathway', 'drug', 'drug-dose', 'drug_code-dose', 'n_genes'\n",
      "    var: 'gene_short_name', 'n_cells', 'highly_variable', 'means', 'dispersions', 'dispersions_norm'\n",
      "    uns: 'hvg', 'pca', 'rank_genes_groups'\n",
      "    obsm: 'X_pca'\n",
      "    varm: 'PCs', 'marker_genes-drug-rank', 'marker_genes-drug-score'\n",
      "Shape: (762039, 1000)\n",
      "Observation columns: ['size_factor', 'cell_type', 'replicate', 'dose', 'drug_code', 'pathway_level_1', 'pathway_level_2', 'product_name', 'target', 'pathway']\n",
      "Feature columns: ['ENSG00000243620.1', 'ENSG00000271503.5', 'ENSG00000259124.1', 'ENSG00000121101.15', 'ENSG00000160963.13', 'ENSG00000135346.8', 'ENSG00000143839.14', 'ENSG00000100867.14', 'ENSG00000140986.7', 'ENSG00000230666.5']\n",
      "Unique conditions: ['tak_901', 'ag_490', 'abexinostat', 'alisertib', 'busulfan', ..., 'ac480', 'cediranib', 'thiotepa', 'cerdulatinib', 'zileuton']\n",
      "Length: 189\n",
      "Categories (189, object): ['2_methoxyestradiol', '_jq1', 'a_366', 'abt_737', ..., 'xav_939', 'ym155', 'zm_447439', 'zileuton']\n",
      "Number of cells per condition:\n",
      "control              17565\n",
      "ellagic_acid          6255\n",
      "divalproex_sodium     6198\n",
      "ruxolitinib           6139\n",
      "mc1568                6120\n",
      "                     ...  \n",
      "alvespimycin_hcl      2085\n",
      "patupilone            1798\n",
      "flavopiridol_hcl      1703\n",
      "epothilone_a          1401\n",
      "ym155                  989\n",
      "Name: drug, Length: 189, dtype: int64\n"
     ]
    }
   ],
   "source": [
    "import scanpy as sc\n",
    "# Load your dataset\n",
    "adata_sc = sc.read_h5ad(\"./scrna-sciplex3/hvg.h5ad\")\n",
    "\n",
    "# Basic overview\n",
    "print(adata_sc)\n",
    "print(\"Shape:\", adata_sc.shape)\n",
    "\n",
    "# View column names (metadata about each cell)\n",
    "print(\"Observation columns:\", adata_sc.obs.columns.tolist()[:10])\n",
    "print(\"Feature columns:\", adata_sc.var_names[:10].tolist())\n",
    "\n",
    "# How many drugs (conditions)?\n",
    "print(\"Unique conditions:\", adata_sc.obs['drug'].unique())\n",
    "print(\"Number of cells per condition:\")\n",
    "print(adata_sc.obs['drug'].value_counts())\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "4b97042f",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import sys\n",
    "import json\n",
    "import logging\n",
    "import argparse\n",
    "import geomloss\n",
    "import random\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from torch.utils.data import TensorDataset, DataLoader\n",
    "from tqdm import tqdm\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "from pathlib import Path\n",
    "from typing import Dict, Tuple, List, Optional\n",
    "from umap import UMAP\n",
    "import matplotlib.pyplot as plt\n",
    "from scipy.optimize import linear_sum_assignment\n",
    "from sklearn.preprocessing import StandardScaler, MinMaxScaler\n",
    "from sklearn.metrics.pairwise import rbf_kernel\n",
    "from typing import Dict, Tuple, List\n",
    "from scipy.stats import ks_2samp\n",
    "from scipy.spatial.distance import cdist\n",
    "from sklearn.metrics import r2_score\n",
    "\n",
    "import gc\n",
    "gc.collect()\n",
    "\n",
    "def median_heuristic_gamma(X: np.ndarray, Y: np.ndarray) -> float:\n",
    "    \"\"\"\n",
    "    Median heuristic for RBF bandwidth: gamma = 1 / median(||x - y||^2).\n",
    "    Uses the median of pairwise distances in the pooled set.\n",
    "    \"\"\"\n",
    "    Z = np.vstack([X, Y])\n",
    "    # Sample if too large for efficiency\n",
    "    max_samples = 5000\n",
    "    if Z.shape[0] > max_samples:\n",
    "        idx = np.random.choice(Z.shape[0], size=max_samples, replace=False)\n",
    "        Z = Z[idx]\n",
    "    D2 = cdist(Z, Z, metric='sqeuclidean')\n",
    "    # Use upper triangular without diagonal\n",
    "    triu = D2[np.triu_indices_from(D2, k=1)]\n",
    "    med = np.median(triu[triu > 0]) if np.any(triu > 0) else 1.0\n",
    "    return 1.0 / max(med, 1e-12)\n",
    "\n",
    "def mmd_distance(X: np.ndarray, Y: np.ndarray, gamma: float) -> float:\n",
    "    \"\"\"\n",
    "    Unbiased MMD^2 estimator using Gaussian (RBF) kernel, sklearn backend.\n",
    "\n",
    "    Args:\n",
    "        X: (n_samples, n_features) first sample\n",
    "        Y: (m_samples, n_features) second sample\n",
    "        gamma: RBF kernel bandwidth; if None, uses median heuristic\n",
    "\n",
    "    Returns:\n",
    "        Unbiased MMD^2 value\n",
    "    \"\"\"\n",
    "    n = X.shape[0]\n",
    "    m = Y.shape[0]\n",
    "\n",
    "    # Kernel matrices\n",
    "    Kxx = rbf_kernel(X, X, gamma=gamma)\n",
    "    Kyy = rbf_kernel(Y, Y, gamma=gamma)\n",
    "    Kxy = rbf_kernel(X, Y, gamma=gamma)\n",
    "\n",
    "    # Unbiased: exclude diagonal entries\n",
    "    np.fill_diagonal(Kxx, 0.0)\n",
    "    np.fill_diagonal(Kyy, 0.0)\n",
    "\n",
    "    term_xx = Kxx.sum() / (n * (n - 1)) if n > 1 else 0.0\n",
    "    term_yy = Kyy.sum() / (m * (m - 1)) if m > 1 else 0.0\n",
    "    term_xy = 2.0 * Kxy.mean()\n",
    "\n",
    "    mmd2 = term_xx + term_yy - term_xy\n",
    "    mmd2 = max(mmd2, 0.0)  # Numerical stability\n",
    "    return float(mmd2)\n",
    "\n",
    "def r2_feature_means(y_true: np.ndarray, y_pred: np.ndarray) -> float:\n",
    "    \"\"\"\n",
    "    R^2 computed across features between mean vectors of y_true and y_pred.\n",
    "    \"\"\"\n",
    "    mu_true = y_true.mean(axis=0)\n",
    "    mu_pred = y_pred.mean(axis=0)\n",
    "    ss_res = float(np.sum((mu_true - mu_pred) ** 2))\n",
    "    ss_tot = float(np.sum((mu_true - mu_true.mean()) ** 2))\n",
    "    if ss_tot <= 1e-12:\n",
    "        return 1.0 if ss_res <= 1e-12 else 0.0\n",
    "    return 1.0 - ss_res / ss_tot\n",
    "\n",
    "def wasserstein_pointcloud(\n",
    "    X,\n",
    "    Y,\n",
    "    p: int = 2,\n",
    "    a=None,\n",
    "    b=None,\n",
    "    method: str = \"emd\",          # \"emd\" (exact) or \"sinkhorn\" (approx)\n",
    "    reg: float = 1e-1,            # Sinkhorn regularization (only used if method=\"sinkhorn\")\n",
    "    return_plan: bool = False,\n",
    "):\n",
    "    \"\"\"\n",
    "    Compute Wasserstein distance W_p between two empirical distributions supported on point sets X and Y.\n",
    "\n",
    "    Parameters\n",
    "    ----------\n",
    "    X : (n, d) array-like\n",
    "        Source points.\n",
    "    Y : (m, d) array-like\n",
    "        Target points.\n",
    "    p : int\n",
    "        Order of the Wasserstein distance (commonly 1 or 2).\n",
    "    a : (n,) array-like or None\n",
    "        Weights for X; if None, uniform weights.\n",
    "    b : (m,) array-like or None\n",
    "        Weights for Y; if None, uniform weights.\n",
    "    method : str\n",
    "        \"emd\" for exact optimal transport (requires POT),\n",
    "        \"sinkhorn\" for entropic approximation (requires POT).\n",
    "    reg : float\n",
    "        Entropic regularization strength for Sinkhorn.\n",
    "    return_plan : bool\n",
    "        If True, also return the optimal transport plan.\n",
    "\n",
    "    Returns\n",
    "    -------\n",
    "    Wp : float\n",
    "        Wasserstein distance of order p.\n",
    "    plan : (n, m) ndarray, optional\n",
    "        Optimal transport plan (only if return_plan=True).\n",
    "    \"\"\"\n",
    "    X = np.asarray(X, dtype=np.float64)\n",
    "    Y = np.asarray(Y, dtype=np.float64)\n",
    "    if X.ndim != 2 or Y.ndim != 2:\n",
    "        raise ValueError(\"X and Y must be 2D arrays with shape (n, d) and (m, d).\")\n",
    "    if X.shape[1] != Y.shape[1]:\n",
    "        raise ValueError(f\"Dimension mismatch: X has d={X.shape[1]}, Y has d={Y.shape[1]}.\")\n",
    "\n",
    "    n, d = X.shape\n",
    "    m, _ = Y.shape\n",
    "\n",
    "    if a is None:\n",
    "        a = np.full(n, 1.0 / n, dtype=np.float64)\n",
    "    else:\n",
    "        a = np.asarray(a, dtype=np.float64)\n",
    "        a = a / a.sum()\n",
    "\n",
    "    if b is None:\n",
    "        b = np.full(m, 1.0 / m, dtype=np.float64)\n",
    "    else:\n",
    "        b = np.asarray(b, dtype=np.float64)\n",
    "        b = b / b.sum()\n",
    "\n",
    "    # Cost matrix: C_ij = ||x_i - y_j||^p\n",
    "    # Compute squared Euclidean via (x-y)^2 = x^2 + y^2 - 2xy for speed\n",
    "    X2 = np.sum(X * X, axis=1, keepdims=True)          # (n, 1)\n",
    "    Y2 = np.sum(Y * Y, axis=1, keepdims=True).T        # (1, m)\n",
    "    sq = np.maximum(X2 + Y2 - 2.0 * (X @ Y.T), 0.0)     # (n, m)\n",
    "    if p == 2:\n",
    "        C = sq\n",
    "    else:\n",
    "        C = sq ** (p / 2.0)\n",
    "\n",
    "    try:\n",
    "        import ot  # POT: Python Optimal Transport\n",
    "    except ImportError as e:\n",
    "        raise ImportError(\n",
    "            \"This function requires the POT library. Install with: pip install pot\"\n",
    "        ) from e\n",
    "\n",
    "    method = method.lower()\n",
    "    if method == \"emd\":\n",
    "        # exact OT: minimizes <P, C>\n",
    "        P = ot.emd(a, b, C)\n",
    "        cost = float(np.sum(P * C))\n",
    "    elif method == \"sinkhorn\":\n",
    "        # entropic OT approximation\n",
    "        P = ot.sinkhorn(a, b, C, reg=reg)\n",
    "        cost = float(np.sum(P * C))\n",
    "    else:\n",
    "        raise ValueError('method must be either \"emd\" or \"sinkhorn\".')\n",
    "\n",
    "    Wp = cost ** (1.0 / p)\n",
    "\n",
    "    if return_plan:\n",
    "        return Wp, P\n",
    "    return Wp\n",
    "\n",
    "def summarize_metrics(y_true: np.ndarray, y_pred: np.ndarray, median_gamma: float) -> dict:\n",
    "    \"\"\"\n",
    "    Compute a standard set of metrics: MMD^2 (RBF), R^2 of feature means, median KS across features, and Wasserstein distance.\n",
    "    \"\"\"\n",
    "    # Drop any samples that contain NaNs in either true or pred\n",
    "    mask = (~np.isnan(y_true).any(axis=1)) & (~np.isnan(y_pred).any(axis=1))\n",
    "    if mask.sum() < len(y_true):\n",
    "        print(f\"[summarize_metrics] Dropping {len(y_true) - mask.sum()} samples with NaNs.\")\n",
    "    \n",
    "    y_true = y_true[mask]\n",
    "    y_pred = y_pred[mask]\n",
    "\n",
    "    out = {}\n",
    "\n",
    "    out['mmd2_gamma_median'] = mmd_distance(y_true, y_pred, gamma=median_gamma)\n",
    "    out['mmd2_gamma_0.5'] = mmd_distance(y_true, y_pred, gamma=0.5)\n",
    "    out['mmd2_gamma_1.0'] = mmd_distance(y_true, y_pred, gamma=1.0)\n",
    "    out['wasserstein_distance'] = wasserstein_pointcloud(y_true, y_pred, p=2, method=\"emd\")\n",
    "    out['R2_feature_means'] = r2_feature_means(y_true, y_pred)\n",
    "    return out\n",
    "\n",
    "def split_train_test(X: np.ndarray, Y: np.ndarray, train_fraction: float, seed: int = 42) -> Tuple[np.ndarray, np.ndarray, np.ndarray, np.ndarray]:\n",
    "    if X.shape[0] != Y.shape[0]:\n",
    "        min_len = min(len(X), len(Y))\n",
    "        X = X[:min_len]\n",
    "        Y = Y[:min_len]\n",
    "\n",
    "    n = X.shape[0]\n",
    "    n_train = max(1, int(n * train_fraction))\n",
    "    rng = np.random.default_rng(seed)\n",
    "    idx = rng.permutation(n)\n",
    "    tr_idx, te_idx = idx[:n_train], idx[n_train:]\n",
    "    return X[tr_idx], X[te_idx], Y[tr_idx], Y[te_idx]\n",
    "\n",
    "def topk_markers(adata, drug: str, k: int = 50, rank_key: str = \"marker_genes-drug-rank\"):\n",
    "    R = adata.varm[rank_key]\n",
    "\n",
    "    # --- get the rank vector for this drug ---\n",
    "    if hasattr(R, \"columns\") and hasattr(R, \"iloc\"):  # pandas DataFrame\n",
    "        if drug in R.columns:\n",
    "            r = R[drug].to_numpy()\n",
    "        else:\n",
    "            # fallback: interpret columns as ordered groups; try to map via rank_genes_groups names\n",
    "            names = adata.uns[\"rank_genes_groups\"][\"names\"]\n",
    "            groups = list(names.dtype.names) if (hasattr(names, \"dtype\") and names.dtype.names is not None) else list(names.columns)\n",
    "            r = R.iloc[:, groups.index(drug)].to_numpy()\n",
    "    else:  # numpy array (or array-like)\n",
    "        names = adata.uns[\"rank_genes_groups\"][\"names\"]\n",
    "        groups = list(names.dtype.names) if (hasattr(names, \"dtype\") and names.dtype.names is not None) else list(names.columns)\n",
    "        r = np.asarray(R)[:, groups.index(drug)]\n",
    "\n",
    "    # smaller rank => stronger marker\n",
    "    idx = np.argsort(r)[:k]\n",
    "    gene_ids = adata.var_names[idx].to_list()\n",
    "    gene_short = (adata.var.iloc[idx][\"gene_short_name\"].to_list()\n",
    "                  if \"gene_short_name\" in adata.var.columns else None)\n",
    "    return gene_ids, gene_short, idx\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 57,
   "id": "eb96ecaa",
   "metadata": {},
   "outputs": [],
   "source": [
    "def AOT(\n",
    "    X_tr_pre, Y_tr_post, X_te_pre, Y_te_post,\n",
    "    top_feature_subset: Optional[List[int]] = None,\n",
    "    n_epochs=3000, batch_size=2048, lr=1e-3, device=\"auto\", seed=12345, verbose=True\n",
    "):\n",
    "    import torch, numpy as np, random, ot\n",
    "    from torch import nn\n",
    "    from torch.utils.data import DataLoader, TensorDataset\n",
    "    from sklearn.preprocessing import StandardScaler\n",
    "\n",
    "\n",
    "    # ---------------- Device check ---------------- #\n",
    "    if device == \"auto\":\n",
    "        device = \"cuda\" if torch.cuda.is_available() else (\"mps\" if torch.backends.mps.is_available() else \"cpu\")\n",
    "    \n",
    "    # Helper: only pin memory for CPU→CUDA pipeline\n",
    "    def should_pin(device, tensor):\n",
    "        return (device.startswith(\"mps\") and tensor.device.type == \"cpu\")\n",
    "\n",
    "    # ---------------- Helper Functions ---------------- #\n",
    "    def cost_matrix(X, Y):\n",
    "        X2 = (X**2).sum(1, keepdims=True)\n",
    "        Y2 = (Y**2).sum(1, keepdims=True).T\n",
    "        return X2 + Y2 - 2 * X @ Y.T\n",
    "\n",
    "    def exact_ot_plan(X, Y):\n",
    "        a, b = torch.ones(len(X)), torch.ones(len(Y))\n",
    "        C = cost_matrix(X, Y)\n",
    "        Pi = ot.emd(a, b, C, numItermax=1e7)\n",
    "        return Pi, C\n",
    "\n",
    "    class Mapper(nn.Module):\n",
    "        def __init__(self, d):\n",
    "            super().__init__()\n",
    "            self.first = nn.Linear(d, d, bias=False)\n",
    "            nn.init.eye_(self.first.weight)\n",
    "        def forward(self, x):\n",
    "            return self.first(x)\n",
    "        def loss(self, Xmap, Y, P):\n",
    "            return torch.norm(Y - P.T @ Xmap)**2\n",
    "\n",
    "    # ---------------- Data Setup ---------------- #\n",
    "    scaler = StandardScaler()\n",
    "\n",
    "    XY = np.vstack([X_tr_pre, Y_tr_post])\n",
    "    XYs = torch.tensor(scaler.fit_transform(XY), dtype=torch.float32, device=device)\n",
    "\n",
    "    n = len(X_tr_pre)\n",
    "    Xtr_s = XYs[:n]\n",
    "    Ytr_s = XYs[n:]\n",
    "\n",
    "    Xte_s = torch.tensor(scaler.transform(X_te_pre), dtype=torch.float32, device=device)\n",
    "    Yte = torch.tensor(Y_te_post, dtype=torch.float32, device=device)\n",
    "\n",
    "    # Build dataloader (pin only if CPU→CUDA)\n",
    "    pin = should_pin(device, Xtr_s)\n",
    "    train_loader = DataLoader(\n",
    "        TensorDataset(Xtr_s, Ytr_s),\n",
    "        batch_size=batch_size,\n",
    "        shuffle=True,\n",
    "        pin_memory=pin\n",
    "    )\n",
    "\n",
    "    # ---------------- Seed ---------------- #\n",
    "    torch.manual_seed(seed); random.seed(seed); np.random.seed(seed)\n",
    "\n",
    "    mapper = Mapper(X_tr_pre.shape[1]).to(device)\n",
    "    opt = torch.optim.Adam(mapper.parameters(), lr=lr, weight_decay=1e-4)\n",
    "\n",
    "    # ---------------- Training Loop ---------------- #\n",
    "    for epoch in range(n_epochs+1):\n",
    "        mapper.train()\n",
    "        for bx, by in train_loader:\n",
    "\n",
    "            Xmap = mapper(bx)\n",
    "            with torch.no_grad():\n",
    "                P,_ = exact_ot_plan(Xmap, by)\n",
    "                if isinstance(P, np.ndarray):\n",
    "                    P = torch.from_numpy(P).to(device=device, dtype=torch.float32)\n",
    "                else:\n",
    "                    P = P.clone().detach().to(device=device, dtype=torch.float32)\n",
    "                    \n",
    "            opt.zero_grad(set_to_none=True)\n",
    "            loss = mapper.loss(Xmap, by, P)\n",
    "            loss.backward()\n",
    "            opt.step()\n",
    "\n",
    "        # Logging\n",
    "        if verbose and epoch % 10 == 0:\n",
    "            mapper.eval()\n",
    "            with torch.no_grad():\n",
    "                train_map = mapper(Xtr_s)\n",
    "                train_map_unscaled = scaler.inverse_transform(train_map.cpu().numpy())\n",
    "                Y_train_unscaled = scaler.inverse_transform(Ytr_s.cpu().numpy())\n",
    "                train_mmd = mmd_distance(train_map_unscaled[:, ], Y_train_unscaled[:, ], gamma=median_gamma)\n",
    "\n",
    "                te_s = mapper(Xte_s)\n",
    "                te = scaler.inverse_transform(te_s.cpu())\n",
    "                test_mmd = mmd_distance(te[:, ], Yte.cpu().numpy()[:, ], gamma=median_gamma)\n",
    "                test_wasserstein = wasserstein_pointcloud(te[:, ], Yte.cpu().numpy()[:, ])\n",
    "\n",
    "            print(f\"Epoch {epoch} | Train Loss {loss.item():.4f} | Train MMD {train_mmd:.4f} | Test MMD {test_mmd:.4f} | Test Wasserstein {test_wasserstein:.4f}\")\n",
    "\n",
    "\n",
    "    # ---------------- Final Eval ---------------- #\n",
    "    mapper.eval()\n",
    "    with torch.no_grad():\n",
    "        te_s = mapper(Xte_s)\n",
    "        te = scaler.inverse_transform(te_s.cpu().numpy())\n",
    "        test_mmd = mmd_distance(te[:, ], Yte.cpu().numpy()[:, ], gamma=median_gamma)\n",
    "        test_wasserstein = wasserstein_pointcloud(te[:, ], Yte.cpu().numpy()[:, ])\n",
    "\n",
    "    recovered_perm = torch.argmax(P, axis=1).cpu().numpy()\n",
    "    return {\n",
    "        \"y_pred\": te,\n",
    "        \"test_mmd\": test_mmd,\n",
    "        \"test_wasserstein\": test_wasserstein,\n",
    "        \"model\": mapper,\n",
    "        \"scaler\": scaler,\n",
    "        \"device\": device,\n",
    "        \"recovered_perm\": recovered_perm\n",
    "    }\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7fab6d49",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 49,
   "id": "6d38280c",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "X_pre cells: (17565, 1000)\n",
      "X_post cells: (3277, 1000)\n",
      "(2621, 1000)\n",
      "(656, 1000)\n",
      "(2621, 1000)\n",
      "(656, 1000)\n",
      "Median heuristic gamma: 0.0022858211277792303\n"
     ]
    }
   ],
   "source": [
    "drug = \"trametinib\"\n",
    "X_pre = adata_sc[adata_sc.obs[\"drug\"] == \"control\"].copy().to_df()\n",
    "X_post  = adata_sc[adata_sc.obs[\"drug\"] == drug].copy().to_df()\n",
    "\n",
    "print(\"X_pre cells:\", X_pre.shape)\n",
    "print(\"X_post cells:\", X_post.shape)\n",
    "\n",
    "top_genes_ids, top_genes_short, top_genes_idx = topk_markers(adata_sc, drug, k=100)\n",
    "\n",
    "X_tr_pre, X_te_pre, Y_tr_post, Y_te_post = split_train_test(X_pre.values, X_post.values, 0.8)\n",
    "\n",
    "print(X_tr_pre.shape)\n",
    "print(X_te_pre.shape)\n",
    "print(Y_tr_post.shape)\n",
    "print(Y_te_post.shape)\n",
    "\n",
    "\n",
    "# Compute median heuristic gamma on training data\n",
    "median_gamma = median_heuristic_gamma(X_tr_pre, Y_tr_post)\n",
    "print(\"Median heuristic gamma:\", median_gamma)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 51,
   "id": "b2a6a182",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "**************** Run: 0 ****************\n",
      "Epoch 0 | Train Loss 3746983.5000 | Train MMD 0.0092 | Test MMD 0.0117 | Test Wasserstein 7.5287\n",
      "Epoch 10 | Train Loss 3616252.5000 | Train MMD 0.0077 | Test MMD 0.0109 | Test Wasserstein 7.3324\n",
      "Epoch 20 | Train Loss 3508505.2500 | Train MMD 0.0071 | Test MMD 0.0100 | Test Wasserstein 7.2164\n",
      "Epoch 30 | Train Loss 3412116.5000 | Train MMD 0.0068 | Test MMD 0.0091 | Test Wasserstein 7.1416\n",
      "Epoch 40 | Train Loss 3323678.0000 | Train MMD 0.0065 | Test MMD 0.0083 | Test Wasserstein 7.0993\n",
      "Epoch 50 | Train Loss 3241594.0000 | Train MMD 0.0062 | Test MMD 0.0077 | Test Wasserstein 7.0778\n",
      "Epoch 60 | Train Loss 3164914.7500 | Train MMD 0.0058 | Test MMD 0.0072 | Test Wasserstein 7.0666\n",
      "Epoch 70 | Train Loss 3092988.2500 | Train MMD 0.0056 | Test MMD 0.0067 | Test Wasserstein 7.0626\n",
      "Epoch 80 | Train Loss 3025315.7500 | Train MMD 0.0054 | Test MMD 0.0063 | Test Wasserstein 7.0638\n",
      "Epoch 90 | Train Loss 2961495.7500 | Train MMD 0.0052 | Test MMD 0.0060 | Test Wasserstein 7.0687\n",
      "Epoch 100 | Train Loss 2901190.0000 | Train MMD 0.0050 | Test MMD 0.0057 | Test Wasserstein 7.0767\n",
      "Epoch 110 | Train Loss 2844111.5000 | Train MMD 0.0049 | Test MMD 0.0054 | Test Wasserstein 7.0875\n",
      "Epoch 120 | Train Loss 2790010.2500 | Train MMD 0.0048 | Test MMD 0.0052 | Test Wasserstein 7.1006\n",
      "Epoch 130 | Train Loss 2738665.0000 | Train MMD 0.0047 | Test MMD 0.0050 | Test Wasserstein 7.1150\n",
      "Epoch 140 | Train Loss 2689879.0000 | Train MMD 0.0046 | Test MMD 0.0048 | Test Wasserstein 7.1307\n",
      "Epoch 150 | Train Loss 2643474.0000 | Train MMD 0.0045 | Test MMD 0.0046 | Test Wasserstein 7.1477\n",
      "Epoch 160 | Train Loss 2599290.7500 | Train MMD 0.0044 | Test MMD 0.0045 | Test Wasserstein 7.1656\n",
      "Epoch 170 | Train Loss 2557182.7500 | Train MMD 0.0043 | Test MMD 0.0043 | Test Wasserstein 7.1846\n",
      "Epoch 180 | Train Loss 2517017.5000 | Train MMD 0.0042 | Test MMD 0.0042 | Test Wasserstein 7.2048\n",
      "Epoch 190 | Train Loss 2478673.2500 | Train MMD 0.0041 | Test MMD 0.0041 | Test Wasserstein 7.2259\n",
      "Epoch 200 | Train Loss 2442038.0000 | Train MMD 0.0041 | Test MMD 0.0040 | Test Wasserstein 7.2475\n",
      "Metrics: {'mmd2_gamma_median': 0.004011356739597405, 'mmd2_gamma_0.5': 0.00016825159990863005, 'mmd2_gamma_1.0': 0.00013004045970516418, 'wasserstein_distance': 7.247520558194178, 'R2_feature_means': 0.7109692250151094}\n",
      "**************** Run: 1 ****************\n",
      "Epoch 0 | Train Loss 3746983.5000 | Train MMD 0.0092 | Test MMD 0.0117 | Test Wasserstein 7.5287\n",
      "Epoch 10 | Train Loss 3616252.5000 | Train MMD 0.0077 | Test MMD 0.0109 | Test Wasserstein 7.3324\n",
      "Epoch 20 | Train Loss 3508505.2500 | Train MMD 0.0071 | Test MMD 0.0100 | Test Wasserstein 7.2164\n",
      "Epoch 30 | Train Loss 3412116.0000 | Train MMD 0.0068 | Test MMD 0.0091 | Test Wasserstein 7.1416\n",
      "Epoch 40 | Train Loss 3323678.0000 | Train MMD 0.0065 | Test MMD 0.0083 | Test Wasserstein 7.0993\n",
      "Epoch 50 | Train Loss 3241594.0000 | Train MMD 0.0062 | Test MMD 0.0077 | Test Wasserstein 7.0778\n",
      "Epoch 60 | Train Loss 3164914.7500 | Train MMD 0.0058 | Test MMD 0.0072 | Test Wasserstein 7.0666\n",
      "Epoch 70 | Train Loss 3092987.7500 | Train MMD 0.0056 | Test MMD 0.0067 | Test Wasserstein 7.0626\n",
      "Epoch 80 | Train Loss 3025315.7500 | Train MMD 0.0054 | Test MMD 0.0063 | Test Wasserstein 7.0638\n",
      "Epoch 90 | Train Loss 2961495.7500 | Train MMD 0.0052 | Test MMD 0.0060 | Test Wasserstein 7.0687\n",
      "Epoch 100 | Train Loss 2901190.0000 | Train MMD 0.0050 | Test MMD 0.0057 | Test Wasserstein 7.0767\n",
      "Epoch 110 | Train Loss 2844111.7500 | Train MMD 0.0049 | Test MMD 0.0054 | Test Wasserstein 7.0875\n",
      "Epoch 120 | Train Loss 2790010.2500 | Train MMD 0.0048 | Test MMD 0.0052 | Test Wasserstein 7.1006\n",
      "Epoch 130 | Train Loss 2738665.0000 | Train MMD 0.0047 | Test MMD 0.0050 | Test Wasserstein 7.1150\n",
      "Epoch 140 | Train Loss 2689878.7500 | Train MMD 0.0046 | Test MMD 0.0048 | Test Wasserstein 7.1307\n",
      "Epoch 150 | Train Loss 2643474.2500 | Train MMD 0.0045 | Test MMD 0.0046 | Test Wasserstein 7.1477\n",
      "Epoch 160 | Train Loss 2599291.0000 | Train MMD 0.0044 | Test MMD 0.0045 | Test Wasserstein 7.1656\n",
      "Epoch 170 | Train Loss 2557182.7500 | Train MMD 0.0043 | Test MMD 0.0043 | Test Wasserstein 7.1846\n",
      "Epoch 180 | Train Loss 2517017.5000 | Train MMD 0.0042 | Test MMD 0.0042 | Test Wasserstein 7.2048\n",
      "Epoch 190 | Train Loss 2478673.5000 | Train MMD 0.0041 | Test MMD 0.0041 | Test Wasserstein 7.2259\n",
      "Epoch 200 | Train Loss 2442038.0000 | Train MMD 0.0041 | Test MMD 0.0040 | Test Wasserstein 7.2475\n",
      "Metrics: {'mmd2_gamma_median': 0.004011356739597405, 'mmd2_gamma_0.5': 0.00016825158535671483, 'mmd2_gamma_1.0': 0.00013004045828668434, 'wasserstein_distance': 7.247520554679105, 'R2_feature_means': 0.7109691882704823}\n",
      "**************** Run: 2 ****************\n",
      "Epoch 0 | Train Loss 3746983.5000 | Train MMD 0.0092 | Test MMD 0.0117 | Test Wasserstein 7.5287\n",
      "Epoch 10 | Train Loss 3616252.5000 | Train MMD 0.0077 | Test MMD 0.0109 | Test Wasserstein 7.3324\n",
      "Epoch 20 | Train Loss 3508505.7500 | Train MMD 0.0071 | Test MMD 0.0100 | Test Wasserstein 7.2164\n",
      "Epoch 30 | Train Loss 3412116.0000 | Train MMD 0.0068 | Test MMD 0.0091 | Test Wasserstein 7.1416\n",
      "Epoch 40 | Train Loss 3323678.0000 | Train MMD 0.0065 | Test MMD 0.0083 | Test Wasserstein 7.0993\n",
      "Epoch 50 | Train Loss 3241594.0000 | Train MMD 0.0062 | Test MMD 0.0077 | Test Wasserstein 7.0778\n",
      "Epoch 60 | Train Loss 3164914.7500 | Train MMD 0.0058 | Test MMD 0.0072 | Test Wasserstein 7.0666\n",
      "Epoch 70 | Train Loss 3092988.2500 | Train MMD 0.0056 | Test MMD 0.0067 | Test Wasserstein 7.0626\n",
      "Epoch 80 | Train Loss 3025315.7500 | Train MMD 0.0054 | Test MMD 0.0063 | Test Wasserstein 7.0638\n",
      "Epoch 90 | Train Loss 2961495.7500 | Train MMD 0.0052 | Test MMD 0.0060 | Test Wasserstein 7.0687\n",
      "Epoch 100 | Train Loss 2901190.0000 | Train MMD 0.0050 | Test MMD 0.0057 | Test Wasserstein 7.0767\n",
      "Epoch 110 | Train Loss 2844111.7500 | Train MMD 0.0049 | Test MMD 0.0054 | Test Wasserstein 7.0875\n",
      "Epoch 120 | Train Loss 2790010.2500 | Train MMD 0.0048 | Test MMD 0.0052 | Test Wasserstein 7.1006\n",
      "Epoch 130 | Train Loss 2738665.0000 | Train MMD 0.0047 | Test MMD 0.0050 | Test Wasserstein 7.1150\n",
      "Epoch 140 | Train Loss 2689878.7500 | Train MMD 0.0046 | Test MMD 0.0048 | Test Wasserstein 7.1307\n",
      "Epoch 150 | Train Loss 2643474.0000 | Train MMD 0.0045 | Test MMD 0.0046 | Test Wasserstein 7.1477\n",
      "Epoch 160 | Train Loss 2599290.7500 | Train MMD 0.0044 | Test MMD 0.0045 | Test Wasserstein 7.1656\n",
      "Epoch 170 | Train Loss 2557182.7500 | Train MMD 0.0043 | Test MMD 0.0043 | Test Wasserstein 7.1846\n",
      "Epoch 180 | Train Loss 2517017.5000 | Train MMD 0.0042 | Test MMD 0.0042 | Test Wasserstein 7.2048\n",
      "Epoch 190 | Train Loss 2478673.5000 | Train MMD 0.0041 | Test MMD 0.0041 | Test Wasserstein 7.2259\n",
      "Epoch 200 | Train Loss 2442038.0000 | Train MMD 0.0041 | Test MMD 0.0040 | Test Wasserstein 7.2475\n",
      "Metrics: {'mmd2_gamma_median': 0.004011356739597405, 'mmd2_gamma_0.5': 0.00016825159990863005, 'mmd2_gamma_1.0': 0.00013004046081491367, 'wasserstein_distance': 7.247520572700156, 'R2_feature_means': 0.7109691564878572}\n",
      "**************** Run: 3 ****************\n",
      "Epoch 0 | Train Loss 3746983.5000 | Train MMD 0.0092 | Test MMD 0.0117 | Test Wasserstein 7.5287\n",
      "Epoch 10 | Train Loss 3616252.5000 | Train MMD 0.0077 | Test MMD 0.0109 | Test Wasserstein 7.3324\n",
      "Epoch 20 | Train Loss 3508505.2500 | Train MMD 0.0071 | Test MMD 0.0100 | Test Wasserstein 7.2164\n",
      "Epoch 30 | Train Loss 3412116.5000 | Train MMD 0.0068 | Test MMD 0.0091 | Test Wasserstein 7.1416\n",
      "Epoch 40 | Train Loss 3323678.0000 | Train MMD 0.0065 | Test MMD 0.0083 | Test Wasserstein 7.0993\n",
      "Epoch 50 | Train Loss 3241594.0000 | Train MMD 0.0062 | Test MMD 0.0077 | Test Wasserstein 7.0778\n",
      "Epoch 60 | Train Loss 3164914.5000 | Train MMD 0.0058 | Test MMD 0.0072 | Test Wasserstein 7.0666\n",
      "Epoch 70 | Train Loss 3092987.7500 | Train MMD 0.0056 | Test MMD 0.0067 | Test Wasserstein 7.0626\n",
      "Epoch 80 | Train Loss 3025315.7500 | Train MMD 0.0054 | Test MMD 0.0063 | Test Wasserstein 7.0638\n",
      "Epoch 90 | Train Loss 2961495.7500 | Train MMD 0.0052 | Test MMD 0.0060 | Test Wasserstein 7.0687\n",
      "Epoch 100 | Train Loss 2901190.0000 | Train MMD 0.0050 | Test MMD 0.0057 | Test Wasserstein 7.0767\n",
      "Epoch 110 | Train Loss 2844111.5000 | Train MMD 0.0049 | Test MMD 0.0054 | Test Wasserstein 7.0875\n",
      "Epoch 120 | Train Loss 2790010.2500 | Train MMD 0.0048 | Test MMD 0.0052 | Test Wasserstein 7.1006\n",
      "Epoch 130 | Train Loss 2738665.0000 | Train MMD 0.0047 | Test MMD 0.0050 | Test Wasserstein 7.1150\n",
      "Epoch 140 | Train Loss 2689879.0000 | Train MMD 0.0046 | Test MMD 0.0048 | Test Wasserstein 7.1307\n",
      "Epoch 150 | Train Loss 2643474.0000 | Train MMD 0.0045 | Test MMD 0.0046 | Test Wasserstein 7.1477\n",
      "Epoch 160 | Train Loss 2599290.7500 | Train MMD 0.0044 | Test MMD 0.0045 | Test Wasserstein 7.1656\n",
      "Epoch 170 | Train Loss 2557182.7500 | Train MMD 0.0043 | Test MMD 0.0043 | Test Wasserstein 7.1846\n",
      "Epoch 180 | Train Loss 2517018.0000 | Train MMD 0.0042 | Test MMD 0.0042 | Test Wasserstein 7.2048\n",
      "Epoch 190 | Train Loss 2478673.5000 | Train MMD 0.0041 | Test MMD 0.0041 | Test Wasserstein 7.2259\n",
      "Epoch 200 | Train Loss 2442038.0000 | Train MMD 0.0041 | Test MMD 0.0040 | Test Wasserstein 7.2475\n",
      "Metrics: {'mmd2_gamma_median': 0.004011356739597405, 'mmd2_gamma_0.5': 0.00016825158535671483, 'mmd2_gamma_1.0': 0.00013004045717693487, 'wasserstein_distance': 7.247520563052435, 'R2_feature_means': 0.7109691381155381}\n",
      "**************** Run: 4 ****************\n",
      "Epoch 0 | Train Loss 3746983.5000 | Train MMD 0.0092 | Test MMD 0.0117 | Test Wasserstein 7.5287\n",
      "Epoch 10 | Train Loss 3616252.5000 | Train MMD 0.0077 | Test MMD 0.0109 | Test Wasserstein 7.3324\n",
      "Epoch 20 | Train Loss 3508505.2500 | Train MMD 0.0071 | Test MMD 0.0100 | Test Wasserstein 7.2164\n",
      "Epoch 30 | Train Loss 3412116.5000 | Train MMD 0.0068 | Test MMD 0.0091 | Test Wasserstein 7.1416\n",
      "Epoch 40 | Train Loss 3323678.0000 | Train MMD 0.0065 | Test MMD 0.0083 | Test Wasserstein 7.0993\n",
      "Epoch 50 | Train Loss 3241594.0000 | Train MMD 0.0062 | Test MMD 0.0077 | Test Wasserstein 7.0778\n",
      "Epoch 60 | Train Loss 3164914.7500 | Train MMD 0.0058 | Test MMD 0.0072 | Test Wasserstein 7.0666\n",
      "Epoch 70 | Train Loss 3092988.2500 | Train MMD 0.0056 | Test MMD 0.0067 | Test Wasserstein 7.0626\n",
      "Epoch 80 | Train Loss 3025315.7500 | Train MMD 0.0054 | Test MMD 0.0063 | Test Wasserstein 7.0638\n",
      "Epoch 90 | Train Loss 2961495.7500 | Train MMD 0.0052 | Test MMD 0.0060 | Test Wasserstein 7.0687\n",
      "Epoch 100 | Train Loss 2901190.0000 | Train MMD 0.0050 | Test MMD 0.0057 | Test Wasserstein 7.0767\n",
      "Epoch 110 | Train Loss 2844111.7500 | Train MMD 0.0049 | Test MMD 0.0054 | Test Wasserstein 7.0875\n",
      "Epoch 120 | Train Loss 2790010.2500 | Train MMD 0.0048 | Test MMD 0.0052 | Test Wasserstein 7.1006\n",
      "Epoch 130 | Train Loss 2738665.0000 | Train MMD 0.0047 | Test MMD 0.0050 | Test Wasserstein 7.1150\n",
      "Epoch 140 | Train Loss 2689879.0000 | Train MMD 0.0046 | Test MMD 0.0048 | Test Wasserstein 7.1307\n",
      "Epoch 150 | Train Loss 2643474.0000 | Train MMD 0.0045 | Test MMD 0.0046 | Test Wasserstein 7.1477\n",
      "Epoch 160 | Train Loss 2599290.7500 | Train MMD 0.0044 | Test MMD 0.0045 | Test Wasserstein 7.1656\n",
      "Epoch 170 | Train Loss 2557182.7500 | Train MMD 0.0043 | Test MMD 0.0043 | Test Wasserstein 7.1846\n",
      "Epoch 180 | Train Loss 2517017.5000 | Train MMD 0.0042 | Test MMD 0.0042 | Test Wasserstein 7.2048\n",
      "Epoch 190 | Train Loss 2478673.5000 | Train MMD 0.0041 | Test MMD 0.0041 | Test Wasserstein 7.2259\n",
      "Epoch 200 | Train Loss 2442038.0000 | Train MMD 0.0041 | Test MMD 0.0040 | Test Wasserstein 7.2475\n",
      "Metrics: {'mmd2_gamma_median': 0.004011356739597405, 'mmd2_gamma_0.5': 0.00016825159990863005, 'mmd2_gamma_1.0': 0.00013004045717693487, 'wasserstein_distance': 7.247520557034477, 'R2_feature_means': 0.7109691564878572}\n",
      "**************** Run: 5 ****************\n",
      "Epoch 0 | Train Loss 3746983.5000 | Train MMD 0.0092 | Test MMD 0.0117 | Test Wasserstein 7.5287\n",
      "Epoch 10 | Train Loss 3616252.5000 | Train MMD 0.0077 | Test MMD 0.0109 | Test Wasserstein 7.3324\n",
      "Epoch 20 | Train Loss 3508505.7500 | Train MMD 0.0071 | Test MMD 0.0100 | Test Wasserstein 7.2164\n",
      "Epoch 30 | Train Loss 3412116.0000 | Train MMD 0.0068 | Test MMD 0.0091 | Test Wasserstein 7.1416\n",
      "Epoch 40 | Train Loss 3323678.0000 | Train MMD 0.0065 | Test MMD 0.0083 | Test Wasserstein 7.0993\n",
      "Epoch 50 | Train Loss 3241594.0000 | Train MMD 0.0062 | Test MMD 0.0077 | Test Wasserstein 7.0778\n",
      "Epoch 60 | Train Loss 3164914.5000 | Train MMD 0.0058 | Test MMD 0.0072 | Test Wasserstein 7.0666\n",
      "Epoch 70 | Train Loss 3092988.2500 | Train MMD 0.0056 | Test MMD 0.0067 | Test Wasserstein 7.0626\n",
      "Epoch 80 | Train Loss 3025315.7500 | Train MMD 0.0054 | Test MMD 0.0063 | Test Wasserstein 7.0638\n",
      "Epoch 90 | Train Loss 2961495.7500 | Train MMD 0.0052 | Test MMD 0.0060 | Test Wasserstein 7.0687\n",
      "Epoch 100 | Train Loss 2901190.0000 | Train MMD 0.0050 | Test MMD 0.0057 | Test Wasserstein 7.0767\n",
      "Epoch 110 | Train Loss 2844111.5000 | Train MMD 0.0049 | Test MMD 0.0054 | Test Wasserstein 7.0875\n",
      "Epoch 120 | Train Loss 2790010.2500 | Train MMD 0.0048 | Test MMD 0.0052 | Test Wasserstein 7.1006\n",
      "Epoch 130 | Train Loss 2738665.0000 | Train MMD 0.0047 | Test MMD 0.0050 | Test Wasserstein 7.1150\n",
      "Epoch 140 | Train Loss 2689879.0000 | Train MMD 0.0046 | Test MMD 0.0048 | Test Wasserstein 7.1307\n",
      "Epoch 150 | Train Loss 2643474.2500 | Train MMD 0.0045 | Test MMD 0.0046 | Test Wasserstein 7.1477\n",
      "Epoch 160 | Train Loss 2599290.7500 | Train MMD 0.0044 | Test MMD 0.0045 | Test Wasserstein 7.1656\n",
      "Epoch 170 | Train Loss 2557182.7500 | Train MMD 0.0043 | Test MMD 0.0043 | Test Wasserstein 7.1846\n",
      "Epoch 180 | Train Loss 2517017.5000 | Train MMD 0.0042 | Test MMD 0.0042 | Test Wasserstein 7.2048\n",
      "Epoch 190 | Train Loss 2478673.5000 | Train MMD 0.0041 | Test MMD 0.0041 | Test Wasserstein 7.2259\n",
      "Epoch 200 | Train Loss 2442038.0000 | Train MMD 0.0041 | Test MMD 0.0040 | Test Wasserstein 7.2475\n",
      "Metrics: {'mmd2_gamma_median': 0.004011356739597405, 'mmd2_gamma_0.5': 0.00016825159990863005, 'mmd2_gamma_1.0': 0.00013004046192466315, 'wasserstein_distance': 7.247520571513044, 'R2_feature_means': 0.7109690879605874}\n",
      "**************** Run: 6 ****************\n",
      "Epoch 0 | Train Loss 3746983.5000 | Train MMD 0.0092 | Test MMD 0.0117 | Test Wasserstein 7.5287\n",
      "Epoch 10 | Train Loss 3616252.5000 | Train MMD 0.0077 | Test MMD 0.0109 | Test Wasserstein 7.3324\n",
      "Epoch 20 | Train Loss 3508505.7500 | Train MMD 0.0071 | Test MMD 0.0100 | Test Wasserstein 7.2164\n",
      "Epoch 30 | Train Loss 3412116.0000 | Train MMD 0.0068 | Test MMD 0.0091 | Test Wasserstein 7.1416\n",
      "Epoch 40 | Train Loss 3323678.0000 | Train MMD 0.0065 | Test MMD 0.0083 | Test Wasserstein 7.0993\n",
      "Epoch 50 | Train Loss 3241594.0000 | Train MMD 0.0062 | Test MMD 0.0077 | Test Wasserstein 7.0778\n",
      "Epoch 60 | Train Loss 3164914.5000 | Train MMD 0.0058 | Test MMD 0.0072 | Test Wasserstein 7.0666\n",
      "Epoch 70 | Train Loss 3092987.7500 | Train MMD 0.0056 | Test MMD 0.0067 | Test Wasserstein 7.0626\n",
      "Epoch 80 | Train Loss 3025315.7500 | Train MMD 0.0054 | Test MMD 0.0063 | Test Wasserstein 7.0638\n",
      "Epoch 90 | Train Loss 2961495.7500 | Train MMD 0.0052 | Test MMD 0.0060 | Test Wasserstein 7.0687\n",
      "Epoch 100 | Train Loss 2901190.0000 | Train MMD 0.0050 | Test MMD 0.0057 | Test Wasserstein 7.0767\n",
      "Epoch 110 | Train Loss 2844111.7500 | Train MMD 0.0049 | Test MMD 0.0054 | Test Wasserstein 7.0875\n",
      "Epoch 120 | Train Loss 2790010.2500 | Train MMD 0.0048 | Test MMD 0.0052 | Test Wasserstein 7.1006\n",
      "Epoch 130 | Train Loss 2738665.5000 | Train MMD 0.0047 | Test MMD 0.0050 | Test Wasserstein 7.1150\n",
      "Epoch 140 | Train Loss 2689878.7500 | Train MMD 0.0046 | Test MMD 0.0048 | Test Wasserstein 7.1307\n",
      "Epoch 150 | Train Loss 2643474.0000 | Train MMD 0.0045 | Test MMD 0.0046 | Test Wasserstein 7.1477\n",
      "Epoch 160 | Train Loss 2599291.0000 | Train MMD 0.0044 | Test MMD 0.0045 | Test Wasserstein 7.1656\n",
      "Epoch 170 | Train Loss 2557182.7500 | Train MMD 0.0043 | Test MMD 0.0043 | Test Wasserstein 7.1846\n",
      "Epoch 180 | Train Loss 2517018.0000 | Train MMD 0.0042 | Test MMD 0.0042 | Test Wasserstein 7.2048\n",
      "Epoch 190 | Train Loss 2478673.5000 | Train MMD 0.0041 | Test MMD 0.0041 | Test Wasserstein 7.2259\n",
      "Epoch 200 | Train Loss 2442038.0000 | Train MMD 0.0041 | Test MMD 0.0040 | Test Wasserstein 7.2475\n",
      "Metrics: {'mmd2_gamma_median': 0.004011356739597405, 'mmd2_gamma_0.5': 0.0001682515954696321, 'mmd2_gamma_1.0': 0.00013004045717693487, 'wasserstein_distance': 7.247520556570192, 'R2_feature_means': 0.7109691381155381}\n",
      "**************** Run: 7 ****************\n",
      "Epoch 0 | Train Loss 3746983.5000 | Train MMD 0.0092 | Test MMD 0.0117 | Test Wasserstein 7.5287\n",
      "Epoch 10 | Train Loss 3616252.5000 | Train MMD 0.0077 | Test MMD 0.0109 | Test Wasserstein 7.3324\n",
      "Epoch 20 | Train Loss 3508505.2500 | Train MMD 0.0071 | Test MMD 0.0100 | Test Wasserstein 7.2164\n",
      "Epoch 30 | Train Loss 3412116.0000 | Train MMD 0.0068 | Test MMD 0.0091 | Test Wasserstein 7.1416\n",
      "Epoch 40 | Train Loss 3323678.0000 | Train MMD 0.0065 | Test MMD 0.0083 | Test Wasserstein 7.0993\n",
      "Epoch 50 | Train Loss 3241594.0000 | Train MMD 0.0062 | Test MMD 0.0077 | Test Wasserstein 7.0778\n",
      "Epoch 60 | Train Loss 3164914.5000 | Train MMD 0.0058 | Test MMD 0.0072 | Test Wasserstein 7.0666\n",
      "Epoch 70 | Train Loss 3092987.7500 | Train MMD 0.0056 | Test MMD 0.0067 | Test Wasserstein 7.0626\n",
      "Epoch 80 | Train Loss 3025315.7500 | Train MMD 0.0054 | Test MMD 0.0063 | Test Wasserstein 7.0638\n",
      "Epoch 90 | Train Loss 2961495.7500 | Train MMD 0.0052 | Test MMD 0.0060 | Test Wasserstein 7.0687\n",
      "Epoch 100 | Train Loss 2901190.0000 | Train MMD 0.0050 | Test MMD 0.0057 | Test Wasserstein 7.0767\n",
      "Epoch 110 | Train Loss 2844111.7500 | Train MMD 0.0049 | Test MMD 0.0054 | Test Wasserstein 7.0875\n",
      "Epoch 120 | Train Loss 2790010.2500 | Train MMD 0.0048 | Test MMD 0.0052 | Test Wasserstein 7.1006\n",
      "Epoch 130 | Train Loss 2738665.5000 | Train MMD 0.0047 | Test MMD 0.0050 | Test Wasserstein 7.1150\n",
      "Epoch 140 | Train Loss 2689878.7500 | Train MMD 0.0046 | Test MMD 0.0048 | Test Wasserstein 7.1307\n",
      "Epoch 150 | Train Loss 2643474.0000 | Train MMD 0.0045 | Test MMD 0.0046 | Test Wasserstein 7.1477\n",
      "Epoch 160 | Train Loss 2599290.7500 | Train MMD 0.0044 | Test MMD 0.0045 | Test Wasserstein 7.1656\n",
      "Epoch 170 | Train Loss 2557182.7500 | Train MMD 0.0043 | Test MMD 0.0043 | Test Wasserstein 7.1846\n",
      "Epoch 180 | Train Loss 2517018.0000 | Train MMD 0.0042 | Test MMD 0.0042 | Test Wasserstein 7.2048\n",
      "Epoch 190 | Train Loss 2478673.2500 | Train MMD 0.0041 | Test MMD 0.0041 | Test Wasserstein 7.2259\n",
      "Epoch 200 | Train Loss 2442038.0000 | Train MMD 0.0041 | Test MMD 0.0040 | Test Wasserstein 7.2475\n",
      "Metrics: {'mmd2_gamma_median': 0.004011356739597405, 'mmd2_gamma_0.5': 0.0001682515954696321, 'mmd2_gamma_1.0': 0.0001300404669811218, 'wasserstein_distance': 7.247520561165076, 'R2_feature_means': 0.7109691247052321}\n",
      "**************** Run: 8 ****************\n",
      "Epoch 0 | Train Loss 3746983.5000 | Train MMD 0.0092 | Test MMD 0.0117 | Test Wasserstein 7.5287\n",
      "Epoch 10 | Train Loss 3616252.5000 | Train MMD 0.0077 | Test MMD 0.0109 | Test Wasserstein 7.3324\n",
      "Epoch 20 | Train Loss 3508505.7500 | Train MMD 0.0071 | Test MMD 0.0100 | Test Wasserstein 7.2164\n",
      "Epoch 30 | Train Loss 3412116.0000 | Train MMD 0.0068 | Test MMD 0.0091 | Test Wasserstein 7.1416\n",
      "Epoch 40 | Train Loss 3323678.0000 | Train MMD 0.0065 | Test MMD 0.0083 | Test Wasserstein 7.0993\n",
      "Epoch 50 | Train Loss 3241594.0000 | Train MMD 0.0062 | Test MMD 0.0077 | Test Wasserstein 7.0778\n",
      "Epoch 60 | Train Loss 3164914.7500 | Train MMD 0.0058 | Test MMD 0.0072 | Test Wasserstein 7.0666\n",
      "Epoch 70 | Train Loss 3092988.2500 | Train MMD 0.0056 | Test MMD 0.0067 | Test Wasserstein 7.0626\n",
      "Epoch 80 | Train Loss 3025315.7500 | Train MMD 0.0054 | Test MMD 0.0063 | Test Wasserstein 7.0638\n",
      "Epoch 90 | Train Loss 2961495.7500 | Train MMD 0.0052 | Test MMD 0.0060 | Test Wasserstein 7.0687\n",
      "Epoch 100 | Train Loss 2901190.0000 | Train MMD 0.0050 | Test MMD 0.0057 | Test Wasserstein 7.0767\n",
      "Epoch 110 | Train Loss 2844111.5000 | Train MMD 0.0049 | Test MMD 0.0054 | Test Wasserstein 7.0875\n",
      "Epoch 120 | Train Loss 2790010.2500 | Train MMD 0.0048 | Test MMD 0.0052 | Test Wasserstein 7.1006\n",
      "Epoch 130 | Train Loss 2738665.0000 | Train MMD 0.0047 | Test MMD 0.0050 | Test Wasserstein 7.1150\n",
      "Epoch 140 | Train Loss 2689879.0000 | Train MMD 0.0046 | Test MMD 0.0048 | Test Wasserstein 7.1307\n",
      "Epoch 150 | Train Loss 2643474.2500 | Train MMD 0.0045 | Test MMD 0.0046 | Test Wasserstein 7.1477\n",
      "Epoch 160 | Train Loss 2599290.7500 | Train MMD 0.0044 | Test MMD 0.0045 | Test Wasserstein 7.1656\n",
      "Epoch 170 | Train Loss 2557182.7500 | Train MMD 0.0043 | Test MMD 0.0043 | Test Wasserstein 7.1846\n",
      "Epoch 180 | Train Loss 2517018.0000 | Train MMD 0.0042 | Test MMD 0.0042 | Test Wasserstein 7.2048\n",
      "Epoch 190 | Train Loss 2478673.5000 | Train MMD 0.0041 | Test MMD 0.0041 | Test Wasserstein 7.2259\n",
      "Epoch 200 | Train Loss 2442038.0000 | Train MMD 0.0041 | Test MMD 0.0040 | Test Wasserstein 7.2475\n",
      "Metrics: {'mmd2_gamma_median': 0.004011356739597405, 'mmd2_gamma_0.5': 0.00016825159990863005, 'mmd2_gamma_1.0': 0.00013004046192466315, 'wasserstein_distance': 7.247520556733222, 'R2_feature_means': 0.7109692066427971}\n",
      "**************** Run: 9 ****************\n",
      "Epoch 0 | Train Loss 3746983.5000 | Train MMD 0.0092 | Test MMD 0.0117 | Test Wasserstein 7.5287\n",
      "Epoch 10 | Train Loss 3616252.5000 | Train MMD 0.0077 | Test MMD 0.0109 | Test Wasserstein 7.3324\n",
      "Epoch 20 | Train Loss 3508505.2500 | Train MMD 0.0071 | Test MMD 0.0100 | Test Wasserstein 7.2164\n",
      "Epoch 30 | Train Loss 3412116.5000 | Train MMD 0.0068 | Test MMD 0.0091 | Test Wasserstein 7.1416\n",
      "Epoch 40 | Train Loss 3323678.0000 | Train MMD 0.0065 | Test MMD 0.0083 | Test Wasserstein 7.0993\n",
      "Epoch 50 | Train Loss 3241594.0000 | Train MMD 0.0062 | Test MMD 0.0077 | Test Wasserstein 7.0778\n",
      "Epoch 60 | Train Loss 3164914.7500 | Train MMD 0.0058 | Test MMD 0.0072 | Test Wasserstein 7.0666\n",
      "Epoch 70 | Train Loss 3092987.7500 | Train MMD 0.0056 | Test MMD 0.0067 | Test Wasserstein 7.0626\n",
      "Epoch 80 | Train Loss 3025315.7500 | Train MMD 0.0054 | Test MMD 0.0063 | Test Wasserstein 7.0638\n",
      "Epoch 90 | Train Loss 2961495.7500 | Train MMD 0.0052 | Test MMD 0.0060 | Test Wasserstein 7.0687\n",
      "Epoch 100 | Train Loss 2901190.0000 | Train MMD 0.0050 | Test MMD 0.0057 | Test Wasserstein 7.0767\n",
      "Epoch 110 | Train Loss 2844111.5000 | Train MMD 0.0049 | Test MMD 0.0054 | Test Wasserstein 7.0875\n",
      "Epoch 120 | Train Loss 2790010.2500 | Train MMD 0.0048 | Test MMD 0.0052 | Test Wasserstein 7.1006\n",
      "Epoch 130 | Train Loss 2738665.0000 | Train MMD 0.0047 | Test MMD 0.0050 | Test Wasserstein 7.1150\n",
      "Epoch 140 | Train Loss 2689879.0000 | Train MMD 0.0046 | Test MMD 0.0048 | Test Wasserstein 7.1307\n",
      "Epoch 150 | Train Loss 2643474.0000 | Train MMD 0.0045 | Test MMD 0.0046 | Test Wasserstein 7.1477\n",
      "Epoch 160 | Train Loss 2599290.7500 | Train MMD 0.0044 | Test MMD 0.0045 | Test Wasserstein 7.1656\n",
      "Epoch 170 | Train Loss 2557182.7500 | Train MMD 0.0043 | Test MMD 0.0043 | Test Wasserstein 7.1846\n",
      "Epoch 180 | Train Loss 2517018.0000 | Train MMD 0.0042 | Test MMD 0.0042 | Test Wasserstein 7.2048\n",
      "Epoch 190 | Train Loss 2478673.2500 | Train MMD 0.0041 | Test MMD 0.0041 | Test Wasserstein 7.2259\n",
      "Epoch 200 | Train Loss 2442038.0000 | Train MMD 0.0041 | Test MMD 0.0040 | Test Wasserstein 7.2475\n",
      "Metrics: {'mmd2_gamma_median': 0.004011356739597405, 'mmd2_gamma_0.5': 0.00016825159990863005, 'mmd2_gamma_1.0': 0.00013004046920062076, 'wasserstein_distance': 7.247520555902264, 'R2_feature_means': 0.7109691247052321}\n",
      "=== Metrics Summary over Runs for top 100 genes ===\n",
      "                        mean  std\n",
      "mmd2_gamma_median     0.0040  0.0\n",
      "mmd2_gamma_0.5        0.0002  0.0\n",
      "mmd2_gamma_1.0        0.0001  0.0\n",
      "wasserstein_distance  7.2475  0.0\n",
      "R2_feature_means      0.7110  0.0\n"
     ]
    }
   ],
   "source": [
    "# Multiple runs for robustness\n",
    "all_metrics = []\n",
    "for run in range(10):\n",
    "    print(f\"**************** Run: {run} ****************\")\n",
    "    seed = 1234 + run\n",
    "    torch.manual_seed(seed)\n",
    "    torch.cuda.manual_seed_all(seed)\n",
    "    random.seed(seed)\n",
    "    np.random.seed(seed)\n",
    "    torch.backends.cudnn.deterministic = True\n",
    "    torch.backends.cudnn.benchmark = False\n",
    "\n",
    "    out = AOT(X_tr_pre, Y_tr_post, X_te_pre, Y_te_post, n_epochs=200, batch_size=4096, lr=1e-4, device=\"cuda\", verbose=True, top_feature_subset=top_genes_idx, seed=seed)\n",
    "    metrics = summarize_metrics(out[\"y_pred\"][:, top_genes_idx], Y_te_post[:, top_genes_idx], median_gamma)\n",
    "    print(\"Metrics:\", metrics)\n",
    "    all_metrics.append(metrics)\n",
    "\n",
    "# Results summary\n",
    "df = pd.DataFrame(all_metrics)\n",
    "print(\"=== Metrics Summary over Runs for top 100 genes ===\")\n",
    "print(df.describe().T[['mean', 'std']].round(4))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8b58ac1c",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/u/jrp5td/here/miniconda3/envs/scgen-env/lib/python3.9/site-packages/sklearn/utils/deprecation.py:151: FutureWarning: 'force_all_finite' was renamed to 'ensure_all_finite' in 1.6 and will be removed in 1.8.\n",
      "  warnings.warn(\n",
      "/u/jrp5td/here/miniconda3/envs/scgen-env/lib/python3.9/site-packages/umap/umap_.py:1952: UserWarning: n_jobs value 1 overridden to 1 by setting random_state. Use no seed for parallelism.\n",
      "  warn(\n",
      "/u/jrp5td/here/miniconda3/envs/scgen-env/lib/python3.9/site-packages/sklearn/utils/deprecation.py:151: FutureWarning: 'force_all_finite' was renamed to 'ensure_all_finite' in 1.6 and will be removed in 1.8.\n",
      "  warnings.warn(\n",
      "/u/jrp5td/here/miniconda3/envs/scgen-env/lib/python3.9/site-packages/sklearn/utils/deprecation.py:151: FutureWarning: 'force_all_finite' was renamed to 'ensure_all_finite' in 1.6 and will be removed in 1.8.\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAYYAAAFxCAYAAAB3Fw9EAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguNCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8fJSN1AAAACXBIWXMAAA9hAAAPYQGoP6dpAAEAAElEQVR4nOz9d5hc5X33j7/u06bt7Gyv2qIKSCDAGANGWCC6abaxsbFjOybJ48RxGt808uR5Ej+xQ4p/iX0lcYsN2BgTqk0z1cLCotiYKsBIqOxK23dnZ6fPqffvj6MZ7WzTrpCo53Vdui7t7pkz98zs3p/7094fIaWUBAQEBAQE7Ed5sxcQEBAQEPDWIjAMAQEBAQFVBIYhICAgIKCKwDAEBAQEBFQRGIaAgICAgCoCwxAQEBAQUEVgGAICAgICqggMQ0BAQEBAFYFhCAgICAioIjAMAQFvAL29vfz2b//2IT32zDPP5Mwzz6x8/fOf/xwhBLfffvvhWVxAwAwCwxDwluCJJ57g7//+75mamnqzl3LIvBNeQ0AABIYh4C3CE088wZe+9KW39aa60GvYvn07//3f/31I933ooYd46KGHXufqAgIWj/ZmLyAgYCl4nodlWYTD4Td7KUsiFAod8mMNwziMKwkIODiBxxDwpvP3f//3/MVf/AUAy5cvRwiBEIK+vj6EEHzxi1/kpptuYt26dYRCIR544AEAvvrVr/L+97+fxsZGIpEIJ5100pxx9/I9brvtNtauXUskEuG0005j27ZtAHz7299m1apVhMNhzjzzTPr6+mbd45e//CUXXHABiUSCaDTKxo0befzxxxf1GmB2juGGG25ACMHjjz/O1VdfTXNzM7FYjA9/+MOMj49XPffMHEMZ13X5m7/5G9ra2ojFYlx66aXs27dv0e97QMB8BB5DwJvORz7yEXbs2MHNN9/Mv//7v9PU1ARAc3MzAJs3b+bWW2/li1/8Ik1NTfT29gLw9a9/nUsvvZRPfepTWJbF//zP//Cxj32Me++9l4suuqjqOX7xi19w991384d/+IcAXHvttVx88cX85V/+Jd/4xjf4whe+QCqV4l/+5V+46qqr2Lx5c+Wxmzdv5sILL+Skk07i7/7u71AUheuvv55Nmzbxi1/8gve9730HfQ3z8Ud/9EfU19fzd3/3d/T19fG1r32NL37xi9xyyy0Hfd++8pWvIITgr/7qrxgbG+NrX/sa55xzDs8//zyRSGRxb35AwFzIgIC3AP/6r/8qAblnz56q7wNSURT58ssvz3pMoVCo+tqyLHnsscfKTZs2zbpHKBSquve3v/1tCci2tjaZyWQq37/mmmuq1uF5nly9erU8//zzped5Vc+9fPlyee655x70NUgpZU9Pj/zsZz9b+fr666+XgDznnHOq7vtnf/ZnUlVVOTU1Vfnexo0b5caNGytfP/rooxKQnZ2dVWu/9dZbJSC//vWvz3r+gIClEISSAt7ybNy4kbVr1876/vRTcSqVIp1Oc8YZZ/Dss8/Ouvbss8+ueBoAp5xyCgCXX3458Xh81vd3794NwPPPP89rr73GJz/5SZLJJBMTE0xMTJDP5zn77LN57LHH8DzvkF/b//pf/wshROXrM844A9d16e/vP+hjP/OZz1St/aMf/Sjt7e389Kc/PeT1BARAEEoKeBuwfPnyOb9/77338uUvf5nnn38e0zQr35++0Zbp7u6u+jqRSADQ1dU15/dTqRQAr732GgCf/exn511fOp2mvr7+YC9jTmauq3yf8vMvxOrVq6u+FkKwatWqOXMkAQFLITAMAW955oqX/+IXv+DSSy/lAx/4AN/4xjdob29H13Wuv/56fvSjH826XlXVOe893/fl/om3ZW/gX//1XznhhBPmvLampmYxL+OQnj8g4M0gMAwBbwnmOuUvxB133EE4HObBBx+sKgW9/vrrD+u6Vq5cCUBtbS3nnHPOgtcu9TW8XsreTBkpJTt37mT9+vVv6DoC3nkEOYaAtwSxWAxg0Q1uqqoihMB13cr3+vr6+MlPfnJY13XSSSexcuVKvvrVr5LL5Wb9fHpp6VJfw+vlBz/4AdlstvL17bffzvDwMBdeeOEb8vwB71wCjyHgLcFJJ50EwP/+3/+bT3ziE+i6ziWXXDLv9RdddBH/9m//xgUXXMAnP/lJxsbG+K//+i9WrVrFiy++eNjWpSgK3/3ud7nwwgtZt24dn/vc5+js7GRwcJBHH32U2tpa7rnnngVfQ9lgHG4aGhrYsGEDn/vc5xgdHeVrX/saq1at4vd+7/eOyPMFvHsIDEPAW4KTTz6Zf/iHf+Bb3/oWDzzwAJ7nsWfPnnmv37RpE9/73vf4p3/6J/70T/+U5cuX88///M/09fUdVsMAfoPZk08+yT/8wz/wn//5n+RyOdra2jjllFP4/Oc/f9DXcKQMw9/8zd/w4osvcu2115LNZjn77LP5xje+QTQaPSLPF/DuQcggyxUQEBAQMI0gxxAQEBAQUEVgGAICAgICqggMQ0BAQEBAFYFhCAgICAioIjAMAQEBAQFVBIYhICAgIKCKN7yPwfM8hoaGiMfjb7iEQEBAQMC7FSkl2WyWjo4OFGVhn+ANNwxDQ0OzFC0DAgICAt4Y9u3bx7Jlyxa85g03DGX9+H379lFbW/tGP31AQEDAu5JMJkNXV1fVDI/5eMMNQzl8VFtbGxiGgICAgDeYxYTwg+RzQEBAQEAVgWEICAgICKgiMAwBAQEBAVUs2TBks1n+9E//lJ6eHiKRCO9///t5+umnj8TaAgICAgLeBJZsGH73d3+Xhx9+mBtvvJFt27Zx3nnncc455zA4OHgk1hcQEBAQ8AazpHkMxWKReDzOXXfdxUUXXVT5/kknncSFF17Il7/85YPeI5PJkEgkSKfTQVVSQEBAwBvEUvbeJZWrOo6D67qEw+Gq70ciEbZu3TrnY0zTxDTNqsUFBAQEBLx1WVIoKR6Pc9ppp/EP//APDA0N4bouP/zhD3nyyScZHh6e8zHXXnstiUSi8i/oeg4ICAh4a7Pk0Z67du3iqquu4rHHHkNVVd7znvewZs0annnmGX7zm9/Mun4uj6GrqysIJQUEzGB3up+x4gQtkSZWJHre7OUEvMM4YqEkgJUrV7Jlyxby+TyZTIb29nY+/vGPs2LFijmvD4VChEKhpT5NQMA7kvk2/zt23sdN2+8kbxeI6VE+ddRHuHzVRQvcKSDgyHHIkhixWIxYLEYqleLBBx/kX/7lXw7nugIC3nHMt/nvTvdz0/Y7kUi64h0kSylu2n4nJzYfG3gOAW8KSy5XffDBB3nggQfYs2cPDz/8MGeddRZHH300n/vc547E+gIC3pZsGXiS6165mS0DTwLM2vwlkpu231nxIPJ2gcZwPYpQaAzXk7cLjBUn3uRXEfBuZckeQzqd5pprrmFgYICGhgYuv/xyvvKVr6Dr+pFYX0DA245rHv9H7ut7BEc6aELjot5zuGzl+eTtAl3xjsrmvy87VAkrxfQoyVKKxnA9yVKKmB6lJdL0Zr+UgHcpSzYMV1xxBVdcccWRWEtAwNua3el+Ht77GPfseQghBDV6DUWnyH19j3BC07p5N/8ViR4+ddRHuGn7nezLDlXCTEEYKeDN4g2X3Q4IeCdSzh8M5UcwPYuoGkERgogWIWfnyLuFBTf/y1ddxInNx1Y8CICnRp4JKpQC3hQCwxAQ8DqZnj9oj7YyWZqi6JYQQsHyLBQUVtb2snHZaVWb/4pEz6wqpRWJnqBCKeBNJzAMAQGvk3LyuJw/aC02MVIcJ+/kEQjqQrVMlCYBKps/zF2ldGLzsUGFUsCbTmAYAgIWyXw9CDOTx3GjhklzilqjhrZoKwhmbe7zlagqQsybpAaCBriAN4TAMAQELEChr4g1bnG/+yi3Je8hV8gRlhGu7P4QV576IYBZyWOJpNaIs67xKBSh4EmvsrmXN/SZXkbZAEjJnEnq58df5qG9W2Z5F4GhCDgSBIYhIGAeRu8eY/jWMfYpg3z/5FsQYUFdJsGUluH7o7ewariHkz98InAgebx3zyD5ySL/PfXDBctP5ytRPaF5XaXHoZykPq9rIw/t3VLlXfznC9cR1SM4nhvkIQIOO4FhCHjXsRhNokJfkeFbx5BSkltWoKiUaBxuAB3qlFpG9Qm2b97FuhOPJtobASC2JULdrbXECzE2rjqNzcdvZV9s7vLThUpUVyR6qryBseIEt+68m/pQHSXHJKKGea24m3a1lZ74siAPEXDYCQxDwLuKxVb8ZLflsCZtQh0GjU49YTtEJpyltlRD1sgTcUIk0nGscYtob6TKkIQ6Q5w5eBprplYQ+0KU7uWdc27YM0tUZxqO8tcP7d3CRHGS0cIEhqIT1kJIoDncOCsPERiGgMNBMPM54F3DQrIU0xm9e4zBHwxjT9rkXynQsCPBWc+djkAwkUghJJz17Ol0Ftsxmg0ArHELt+CiN+gIRaA36LRNNHOCva6yWRf6ikw9nabQV6x83bCjjvWltfNu6LvT/Ty0dwsN4ToMVcfybLJ2jvpQLUW3hCe9oFM64LATeAwB7xrmS/hOP2mXT/7CEIS7Q5gDJqUhizOGT6GeOgZjw3QkWzmu/xjqr0xUwkhGs4EaVbEnbfQGHXvSRo2qFcNRzle4BRc1qhJdHaHwWhG34CJUQf2GBK2XtVTuN3PNq+qWU3JKFF2TyeIUF/aexVMjzwad0gFHhMAwBLxrWIwmUfnkH+oMIRSBFtco7S3xizW/5JGjf0FJMwmZIQodRX7/ss9WHhftjdB+RQvDt45hDpoIVVC3wde8nxlmMkdMJh6cJNRhgAalgRJDN5dIbU3T+ek2Wi9tmXfNeadIfTjBR1ddzEdXXRxUJQUcEYJQUsC7hnLC13ItdqR2Y7nWrJP29JO/9CReyWNieYqH3reFglYiXqgBFR597xOM1I9V3b/10hbWfHklDRvrAJjcMsWOv93F6F1jVWEmNazgOR6u6WEOWSAEUkrcgsvwrWOVUNP0NQsE+7JDCERVkvrUtpMCoxBw2Ak8hoB3JULM/f2ZJ381qvKrTc8x5IwiagTZeI62cCuWZs+b7E0/nUUYAqPBwJ60SW1NI1RRCTOVhkxwwBq2wCsvCKwpG8+WlYR2mZlJ6rZUC1NPpzGaDaK9kWDyW8BhJzAMAe8ayslnQzVYHWtlPJ3k+8/exjHWatauXVO5rvXSFuLr/Yqjodgov975AiKroKkqSMmAOcQy2gnvDFMo+ad7a9zCaDZmhaL0Bp1iX4madVGKu0pkX87hpl3fV/cOrE2EBEKCm3PxLI+ZlD2E0bvH2HHrrkqu4rnLtnFP9JFAVyngsBIYhoB3DdOTz/aoTWjYYDQ8zrPffJH4iVHix8Urp/Dyv1dHduJ4Lt3xTobzIzh4eK7LupdW4/7I4WXrVaQENaSgRlUaN9WhRlXMERM1rGCO2bhZl9zLeYQiwAOhCZSYglf0kOb+kevS/z66oLCnSLgzPCsRXegrMnjjCJ7toTfpDIghbhm5m1BniK6E3/h23Ss3IxCc0Lwu8B4CDpnAMAS8aygncsfTSULDBlNGhqiIUDMeY993h9EbJ9ATGu1XtFQSwOXHSCRH169mLDOBl/T4wM7TUKIK5ogFSELHxPBKHsnNUxjtOlOPF/EcD1xQa1UivRFK+0p4Rc//q3MkwhBIS4IK4e4Q9oSDm3MZ/OEIEw9MVq0DYPSuMYr7SqCAPemQXJeiqJRolc0oQsFybfoyA3z9hf+mMdwQeA8Bh0yQfA5411BO5EoHRsPjqKrCBYNn0TrYhPQkep2KlLKSAC73GXys8RIEgmQpRShncN6vN9K4u57inpK/+QuBdKRfppq2KewsEeowCC8LgwbSkdhpB2H4f25aVEVKkKZEaH64yRq3caYcP+Gdd7FSVlUiutBX9HMVCghVIKVHrC9KxAkzJTLkrDx7s4MoQqEz1j5vj0ZAwGIIPIaAdxWXr7qIY6zVPPvNF6kvJegotVOwCwhdoERUlJCCOWgyetcY6aezuAWXY6KruObyL5JfXiT3kxwNr9WBKpCeBBek4m/w9qSN0BSkIwkvC+OVPOxxC7fkUdxZwHP965Gg1+kITdBySRM1R0XZ+eU9eJqLElPBkbgZF1uxK4loa9xCupLQshDWiI10oS3VwoeNC3kwtIWB3DBSevTUdlFjxIjKSNANHXDIBIYh4F3H2rVraDy3juFbx7DTtp8krtVQQoq/uauC5OYUCNAbdbySh36HzjEf72QgMwzLwBqxAd8oqDUqTspBjaq0XtJIcvNUpQJJiSq4BQ/PkpXntzMOiTVRWi5sJH6cn+QWmuJ7FI4ETeAVPISmVBrkymW0UkqiqyPYSRtFV/j0hR/j3PqNvDDxMt99+UcYqh50Qwe8bgLDEPCuZHrlUWZbjsnNqUp5qt6mkf5V1i8hnfD1knAABEIVeJZHuCuEm3dRdIXeP+1CMZRK4tpoMqaVu2p4cYm0PN9j8AAH0r/MkHs1T6QjTMOmevSEBp7EzjjIggcCoqvCZLflgOoyWifloNX4uZBob4QV+BVLnpTB3OiAw4KQUsqDX3b4yGQyJBIJ0uk0tbW1b+RTB7wLKc9TKG/aB7vOszx2/sMerHEbJLA/ph/pDtN8YQMjd45jTfjegt6o0/P5zqoE8Vz32/3VvZSGSr5R8PDvCyhhQagjhBr2q5mSm6ew0w5u3sEt+YlrBBhNOt3/y3+eg72e3el+Xph4GSkJKpMCqljK3ht4DAHvWGbqE82s8ikzfbPNbsthTzmg42/MLkgpCfeGSW6eQktoGG0G9oQfyomvj8/53OVyV4D89jx7vz2EdCWUG+v2l30oERU37xI/Lk7jpkYGfjDE2H0TvociAM2vQBq8cYT4+njVfeda/3POSxX1WE1R2dh5Gh9ddXFgIAKWRGAYAt6RTNcn0uo17KRd2VxH6scqncKxLZFqcbs1/qar6AoiDNIC6UoiXSGKu4qVxjU1omIOmrO6lOei66plWCmHkdvGKgnrcnWRV3QrYnulwRKpJ9O+V1E2IC5I4eHk3Tmfa7rxG2ka54aNt6DVaWiKxt7sIN//zW1sGXySq9ZeGZSuBiyawDAEvCMpdyCjQWmfiXQl0pXceP9tPNjtj8iMeGE2PnUaZ8rT0Bo07Amb9C8zKBG/+QxPABK9SSdxUi3pX2XnVU89GCv/v170ep3RO8dwMg6eLVHCCmrY92SyL2bZd/0wdtI+EG5S8L0WVaDFZj/XTHG+SabIpnO0xZrpLw6gCQVXgOXZwSCfgCURGIaAdyRGs4FQBaWBEqgCFMm2zt9ws/cTIsUIXfUdjCbHeaDzUVaP99C4oxG35IILSsg/rgsdQs1hOj/dRsPp9dhJu0pDqZz8XSyhJh01oiJdiW4oNG6qp/UyP7S14293ITRQQgrS9JAuFc9Br9Po/HRbZSDQfPIbLUYjITvEeHESx3NQFRUNleZwI8lSKihdDVg0SzIMruvy93//9/zwhz9kZGSEjo4Ofvu3f5u//du/RcynShYQ8AYyfeOs35Bg6OYSIHn8mKe596RHGItMEC6F0PMaTZEGdmt7Gc0mqXfrK9pFUkCo1QBF0PunXTScXg9UVzIdLJk917rKcx6iq6LYkzbpp7O0XtZStcErqqA4aCJNDy2h0XBGHcs+00G0NzIrZ9Kwqb5qBkTLUBMXiLO4t/1hHNfB8zx64l0U3VJQuhqwJJZkGP75n/+Zb37zm3z/+99n3bp1/PrXv+Zzn/sciUSCP/7jPz5SawwIWBQzN87GTXVEusIMhIbZfOLjaFLB8HRcXAbyw7gxl5pIlEQuXmk8E2GBQKDVqrh5D8WoFgeYK/k7FzOrh+YS1yvnKKqG/DQbeK5EOrDiz7srRmlm2MietJncnKpUM5W9mLqT44RDIaJulIJTZCQ/RlusmavWXhl4CwGLZkmG4YknnuCyyy7joov8JFZvby8333wzv/rVr47I4gICFstcG2dy8xRNFzbyynM7KIoiraVmQokQI8o4pmvieC6fW/sJVsR7cFQHZ8rxY/squCVv3hzC5OMpCnuKRJdHKhv3dOaqhoqvj8874W0uqe/2T7cQ7gxX5LXnMyzlaqayEuw9ex7BwKA11kx/Zh9Ft4jl2m/AJxDwTmJJhuH9738/3/nOd9ixYwdr1qzhhRdeYOvWrfzbv/3bvI8xTRPTNCtfZzKZQ19tQMA8zLdx1h5Xw3tOXU/ipXsxNYvORDsiJ3Cky/9+75+wcdlpjH7a38ilJXHzHmpERQ3NnUPY8aVdTDw4iXQlQhU0nd/Amr9bWfn5XAZq+NYx4uvjszb/xk11WOMWUB2m8iyP9K8zDN44gnRllWrrfIalrASb31GgMVxPf2Yfuqrjei5AkHwOWBJLMgx//dd/TSaT4eijj0ZVVVzX5Stf+Qqf+tSn5n3Mtddey5e+9KXXvdCAgIVYaOby2t41fNb42Kyu4I3LTgPwexHE/gqkBr2qi3l6SKg0WPKNguchDAVpe0w8OEnTOQ2EO8NY4xbmiFlloJSwgjlkkt2WndVtndycYuzeZFWPRfbFLPu+N0Rp2EQoEO4OI6UkuXmKhk31VR3aMw1XRT22lMSRLioKmqLRHAmSzwFLY0mG4dZbb+Wmm27iRz/6EevWreP555/nT//0T+no6OCzn/3snI+55ppruPrqqytfZzIZurq6Xt+qAwJmMGc4ZtrGeaG2iZWJXqZq0nQv76xskAs1wU3/mVAFWr2KZ3qggHR82QrpeIzeM4454Hss0pM4ORdUQIA5YCI9GPzBCEgq957cnJrlVeiNOv3fHsSesP3+BQ9Ke01q1sVwUg61x9XQtKlh3uR3WT32ulduxvU8pIDumvYg+RywZJZkGP7iL/6Cv/7rv+YTn/gEAMcddxz9/f1ce+218xqGUChEKBR6/SsNCDgI81UNlTd4O+1Qo8XQLtHgqoXDPkDlZ2hQGijh9Uk/B+Hib/z7q5iy23L71VLBHLDxbA8n64DnN8pFug2ELir3tsYt7LSNVrdfoG9/2Cv9TAY7aSM04XdJA9KWmCMmRr2BZ3kHrYgqjwG9fee9bBl8EsdzCKlGoJsUsCSWZBgKhQKKUl2loaoqnjd7FGFAwJvBzKqh8uZvp2xfoM6WDHx3CIEkuSbNi9FXaE00s8zqqGzQ2W057LSNnXYwWnRK+0y/F8KTB5rP9usYCUPglXzvwRyxQZUoQkGJCNyMR2RFGKPRQHqyUoWU2ZbDTjpY4zaKLlBrVYx6Ay2+/89R8ZvfvKLnV0oJgdGms/ure5GOh57Q55T3KIe92ppb+MuT/pCPrro4mAUdcEgsyTBccsklfOUrX6G7u5t169bx3HPP8W//9m9cddVVR2p9AQGvC/907mBnHIQAEVXwCh63vnAvW2JPkjkhS9gN8cHk2Zzx2il4lsfgD4ZxSy520sEruv7pXZEomoJUJdgSrUnfn6x2cXIu7mtFpJQoIQWhCkKtIQq5Im7ORdbLSs7DszwmN6fQ6zXs/R3QMuXSfnkdsTVRtLiKk3VBkQhdoMVVak+Kk3w4hfQkiu7PgSiHnsr5kOyL2VkhsRWX9gQGIeCQWJJh+I//+A/+z//5P3zhC19gbGyMjo4OPv/5z/N//+//PVLrCwh4XRjNhh+asSUi6s87GG0a48Hen6PlVdqVFpL2FPfV/4wVejctshlhCCJtEaRXwE76JaxCFYSXGbh5Dzvlh4m8gofeoKHFVIp7TXBA2h6h7jDgq68qulKV81AMBbfgElkRIWR6eEUXe8rFybrs/eagv2gB6v4Nv/nCBkbvSfpGYf/63YxLqWiy+6t7fWOnCtyii5bQZoXEltKEFxBQZkmGIR6P87WvfY2vfe1rR2g5AQGHl2hvhJZLmhj47hBewUPRBal4hpIo0bSvEVu41Go1TLZPoV6qov5QQW/QEYoguiJKUSlSsy5GcVfJH99Zr9N6eQuKIRi+ZZRIb8QX1avVKO4sotao4Pghpp7PdxJfHye7LQsI4sfVAFRVT7l5FzWs+GM7DUHs6BjmsIl0qcx5GPnxBEIXB4b45D2wXPQGjVB7iNK+EtaEjdFmzGqeCwxDwKEQaCUFvOPpvqoTgWT0niRu0SUxFSdsh8jGctQWa8gYOYyMTltvCyIqqkpe9YROz+/7VXTTE7+FviITD0xWrvWKHuFlYbr/oLOq3HWuqqeZ1VN1G2qZ3DKF0eBv7KH2EOagWbnPXEN81KhCqH1/z0ajjjliYU/YqBF1yQJ/AQEzCQxDwLuCrquW0bipkcmtKfgBbHp5A5vXP85EYpKQZXD29jNY/v5uzCvMeUtep5++5yuPnd4JPV/V05ovr6yqngJIP53FHDZRImqVFHf5eQZvHPEVWROChrPqeXX7Tl6jjxajkZZS05xhq8BbCDhUAsMQ8I5id7p/3kqc8kY5dtcEH9hxKqsnepmKZqnLxlle04NneYTaQ7NO/fNNTTuYqN5C+kh1JyeqDc3qyKyO6pn3EyqoEZXNDVv5yakPkCsWCDshLhBn8cmPf+iQBf4CAmYSGIaAdwx37LyvMr2s3N08czhNtDeC+0mXHXfvpmY4ytF7V6E36tQcG2PvNwerQj51Jyfm1T2avgHPtwkv1I09nUJfkcJrRUKdBmpYwS15FF4rUugrAlSpsu4c38NtyXtAgxarkXxjgS2nPsl73nMcIc3AbLcIaQYt6aBENeDQCQxDwDuC3el+btp+JxJJV7yDZCk1pz7QHTvv4ybjTnIfzBGyQnwkdCGXtV/gGwXTRY0puCW3Ug46MxS09zuDlZkKC40LLZM4OU5qaxpz0ESogroNs2ftzvQstGk9D0DlZ17JI+mlKGkmbW4zQlGomYqxLzrIV379dXJWnpydJ6yGqAsn+MiKD/L76z9z+N/sgHc8ysEvCQh46zNWnCBVSqMIhZJTojFcT94uMFacqFwz3Xh0NyxDq9O4S3mIV3fvpDRkYo5YFHaXKA2ZlAZKpJ/J4hbcSpWSElGwJmycgoMSUysGpHyyn87o3WPs+NtdTG6ZAkCrV3GKLhOPTLLjb3cxevdY5drpnoX0/E5nKcGzvKqfuSUPS1pIIRmPpMCAMSNJzingeR5Fp4QjHdJWloHsMN/YdgPf2vaDI/7eB7zzCAxDwDuC58deJlma5NXJnbw8uYOdU3tm6QONFSfI2776qCIUaqaiTI2k2f7ILpy0g2d6SCSyJHGyLuMPTOBZXmXDtidspCtx0i7FXQXMIYviUKlysi8zK+mcccg+n8cat3CmHKyUVWVQyglmaUly23KUBkzcnMPebw6SfTFL+xUtCCH4WeIxfnzK/ZTCJuORCXaF+5GqR40WIxGqxZGOr5GEJKQZuNLjzl0/ZXe6/w39LALe/gSGIeBtz+50Pw/t20JDuA5D0bFcm8nSFOd1b6wKI5XVR5OlFHbeZmwySdgN0RJurMxXlkUJAtBAMRQ8W+KkHYp9Rb/RTRF4lodnSbySh5t2Gbp1pLLJF/qKTG5NYacdv4zV3K+bJP37IcDNuNhpe5ZB8SwPp+CCBL1eQ0pZaVQz/sZgy2lPoSZUetNdNOcaiTkRfqv5ctprWyk6/vN7eChCwZMehqpju06V1xQQsBiCHEPA256yJ7Cqbjklx6TolEiZU5zQvK7qurL66E3b72QgP4ziKXwwdTZdspNcOO9rHuE3pymqgkTiTDmoUQU15ktTTG6Z8gf6lJEwuWWKUp9JzbExCq8VfZ2lpAOeRG/QfF0lBZASdF+SQ2hKJQld9jAQvugeiqQ0ZBHqNPAKvnBe/959ZNM5mnONKCi0R1qZSCQ57oxjaMjVc9P2O4lqEUzX8o2QUEgYcerDiUBVNWDJBIYh4G3PdE+gMVxP3ilQF5p7Qyyrj+7dM0h+c4GOfCtKg4Jeq2E5fshISNAaVKxhG+lI3JKLZ0pST6RRIwpuToAGsuR7F0IVOAWHiQcnCXUaRHr3y2mkHP9+qkCJKkhH4u1vUKs9saaypnLyWW/UsScdXMsDW1LaU0JoCsnHUnjbJeHjQmTjOeL5GqaKGaINfqisJdKEIgRSwi9HnuWJkV9juw6GaszymgICFkNgGALeEZza9h62DD5ZNYhnvg1xRaKHFSf0MHrxWFWDWvMptQgguy2PnXaQjgQd1KiGZ7k4Uw71ZySwJzNIe7+isOaf8tWYij3hoIaVA3IafUXaP96KtDySm6ew0w5eyQVFkH0+x44du6rGfnolDyWu4A77U9fQBXqdRmprmhankQ92nM39rZsZj08SKhlcHr+I58ZfqpToWq6F7dnk7SKmZyHxeGjvFpojjbPKdgMCFiIwDAFva6b3LmiKytldG/joqovnNQpVDXCX9hBfH2f0rjFSW9Nkn8+hRlWaLmzEzTkM3TTqaxQh8RMPkvjaGHpcY/KJNHbKRlF8SWy8/WJ2JQ/NkxU5jYYN9UR7IzRuaiS7LcfgD4YRhqj0NUwf+9n/7UGccX8+s9AFoVaDcGeYYp/vOZzx2ikclV/JWClJg1nH8nO7+dvt/4xE0hiu5/mJl7A9B4EAIG8XKDmlYKxnwJIJDEPA25a5eheeGnmWj666eM7rpxuRiBfm8vhFXBQ9m/TTWYQhMBoM7EmbifuTNH+wEa1Ow806eCVASpSIwvhP/e5kPa4RWx3BGrFxiy5CU6g7vRZr2J5XTsMat5CurGgiTe+Ejq+Po0VVRLOBNWUjJNjjtn9dQqNxUx3JzVM07WqgNdpM+xUt7GnYR35Hga54B2OFCWzvQO5DILA8m5JrUnLNYKxnwJIIDEPA25Zy0rkr3oEiFBrD9ezLDs25CU43Ii35RsYmk/zQu4PEr+O0jjYROzrmVxzZHqV9JsO3jaHoAuIaQtlfoWR6eLaH3uSL5jkpl8SptaS2ppGOxBq2adxUR/y4+JyyFAt1QpeNRmRFBG1cpTho4lkS6VJpomvc1FjVcZ1PFyu5FU96+/0Ef7CPlB4SGM6Pois6z4+/zKltJ70xH0zA256gXDXgbcv0pLMnPZKl1JyzjXen+/n54BOkSmnqvFrsEYdaswYzbDEVS+PmPcxhEyfnYA6YvsJph4GW0NBrNXq+sIzak+LYUw7mmEX+1QJOxsFKWqS2plGjCpHeMFJKkpun5tUqKvcrCCH8TmghKl5FldFoNgi1GYQ6Qqz48+5KZ3W0N1KlsVSushII0lYWRagoqAgpkPuf01ANGsJ1PLR3S9DPELBoAo8h4G3L9PLT+ZLO33rxB9y5+6cUnRJpM4Nj2rS5LWTjOSJeiNZ4M2qNinTBHDKRHkS6DbSYhoz40hQAuW15hALSA8/0MAsWQheoMY/wsppFz0GYT3hvTrXWT1ertc7F5asuYryY5JYddxHRwliuha7omK5FW7SFzppWwlp4Xk8qIGAuAsMQ8LamXH46l6Lqt7b9gG9suwFXehiKTkg1mHKyeHFJ3IlxYXITLUNNiBZB9x90Yk/aDP5gBLF/fGY51APCzyu06JgD+5vSBChRgVeSmMMmofbQLJG8+VRZ5xPeO5ha61zsTvdzy467mCilEPi5BUPVaY40UGPUENbC83pSAQHzERiGgLc9KxKzZxvvTvdz566f4kiXkGIg8XA8qAvX8lHjIlY9spy2iWZEVFTPUZDMmrEQP64GNaoicxI0v/u5XJJa2udPW5uZcC6rstppB6EJWi5povuqTmB+gwHVRmOh68o8P/4y48UkuqJhaCEsx6RgFzln2Qd4dWrnosp3AwJmEhiGgLctC81eGCtOMFXK4Houea+AQKAIQbPWyCVnnUfb+1qWNGOh/YoWnr7zeUaUceqLtfTGu/FKHqEWY875DcO3jmGnbH/qmi0Z+O4QAonRZMyS8W69tGWWEZhL7numiuvudD+v7n0Nz/OQAr+zWvj5hZPbjud3j/3kvO9PQMBCBIYh4G3JwWYvmI5FyS35HcEIPOniSsGZHaf5m2SCeU/hc4V6tq59mhud28mksxgZnXN3b+Ts5IZZU9vA72S20w52xkEIEFFfBmPkznHUiIowRNVEN3PCZnJzqmIEGjbVM7k5NWvyW3x9vLKuO3bexw2/vIXJ7BReSFJyTVzXBVXQHGnk+KZ1c3pSAQGLIahKCnjbMbN/QSK5afudVVU3Ic0gpkcJa2F0RSOkhkkYtRzTuJqnRp5ZUoXOK6/s4PvP3oZEsqK7h1BXiJ+/7wnsj9rE18dnXW80GwhNIG0JmgBHougCz5I4Oaci46036Nhpm7F7JipGQErJ2D0T2Gm76jq34FZE93an+7nxxduxJm06cq002HUoKIRLIZaF2vmj468KDELA6yLwGALediymf6El0kR7rJWSWyKiRSg6RQp2ie++/CMcz513wttMRu8e49mHXyR9dJrWUjN2u02NFWUgN8zLP9uBepNK/YYErZdVN7O1XNLEwHeH8Aoeii5Qa1W0qP/nNr2PQWi+htJ0I+Bk/E5nc8SsTHSbntQeK06QtwrUFWvRwhpdpQ4UFM577QNcetkFnLBq3byvJyBgMQQeQ8DbjsX0L5RLWcNqmJyVR+wXijBUY14vYyaTj6fYd/0wiUycqIgwZWTI9xUYnUgSKhrUDEcp9BcZunmUV//itarhO91XddL1u+2E2kNodRpGvUHnp9vo/HQbQgiKfUXcgkfDhlr0hHZg5sOkjZ7QiK4MYw5a5HcUMQctoqsPhLdaIk3EjCiZSA7HckjrGWrMKCelj2fNshVH7H0PePcQeAwBbzsW078A1aWsL4y/wk3b7ySihg/aJQ2+p7Dv+mHMIZOmiQbObfwAD614jLHoBCE7xNmvnkHbuJ8MlorEs71ZeYCuq5bN6lYGMCf2h48cSea5PNHVEQqvFSuVTeUcQ6jDQImoeEW3MgM62hthRaKHT6//KDcUb2E8nSRkhbhg8Czee/H6RZW4BgQcjCUZht7eXvr7Z5+wvvCFL/Bf//Vfh21RAQEHY6H+hemsSPTw3PhL3L3nQSZLKVJmmu54J4aqoykqQ7lRdqf7qx5frioSmj+bQbqS054+iRVDPUxYk9SVaumYasOTvsKqoijoTTrOpDOruW1mIrvQV2Rycwo1qlTCSYXXilWVTda4xfi9E5UZ0NLTZjXOTZcPr8slWHPZisAoBBw2lmQYnn76ab/yYT8vvfQS5557Lh/72McO+8ICAg7GfFU308tYAW7afieGatBT28Xe7CB9mX3Uh2rRFZ1vv3RjRVDv8tUXVcTu3IJLqDOEooqKblFHro1O2kCAJeyK6Kpaq+Kknao8wFwcmO5mE+mNVHVLK4ZC3cmJyrXzaSrNev0nBEnmgMPPkgxDc3Nz1df/9E//xMqVK9m4ceNhXVRAwKEys4z11Lb3VCWqE0ac/uw+dEWnLpygZipaEdRruK2O9168vjIfoaxb5LkS6cCKP+/GTvqloygCZ8rGLfkzoZ2UP6sBYOrpdGUTL4eRsi9mKw1vdtJhMLaH4ooSteM1dEbbqzb9ueQxGjfVVaqSAs8g4EhzyDkGy7L44Q9/yNVXX40QYt7rTNPENM3K15lM5lCfMuBdyEJNbHNdO1OGe8vgk2iKWpnuVnRLxPQYjudQ59VSHClRK2sYj08yGZqqmo8wn25RfH28MlvBTtu4OQ/pSlK/SJPd9hu0mIpregjhq7L6E95c9DqNSG+YLd1P8NDyLZhhi0h7mMtrLqJ3vAs4sOlPb7TLbMuR3Jxi7N7kvM1uAQGHk0M2DD/5yU+Ymprit3/7txe87tprr+VLX/rSoT5NwLuYgzWxzWS+MtYP1J7KLyeeY29hgJpoDR9Z+UEe2ruFifwkYTdUEdRrCTdW+gUW0i2K9kbIbsthpx3cgofQAF3gZT2cKQejScccMQFBbG0Ue8LGTtqE2gyGIiP8rHcrWNBV20HKS3NH9j7a/qOZLq+zatMvP+fMZrfBG0dAQPy4+CzvYcvAk+zK9LGytpeNy0477J9JwLuDQzYM3/ve97jwwgvp6OhY8LprrrmGq6++uvJ1JpOhq6vrUJ824F3CXKf/g00imzn7OVlKYeR1TvjpOtaXjiadyHLUppU0djeQsbI8uucJxmuShJ3QAUG9qMBoNhbUKRq9e4zBG0ewp2xwQYQFQu73mhVwS54vTSEl2BK9ScccsbCTNsnWFAVKtLpN6FGdWH+U0cgEuWUFZL9k+NYx9Ea9KhFdzncIReAWXMxBi77/2EeoJVRlSK55/B+5r+8RHOmgCY2Les/h2tP/5oh+TgHvTA7JMPT39/PII49w5513HvTaUChEKBQ6lKcJeBezlCE8ZWaWsUa8MJte2EBHvrWSxP35k0+yRT5JUSnhqS5diQ5WvNLD8n1dlfkI5XzAXDpFlYolw5/ZYO6zkEUJIfwqJgFqWAEp/c4JXeAVPYwmHUVXqBmIEmkNkavLE7JCTKlpIm6IulwCvUEn/2qe3V/d699nf26hnO9wCi7mXj8s6+ZdrJRVCX09rT3PfX2PIJHU6DUUnSL39T3CBT1nBZ5DwJI5JMNw/fXX09LSwkUXBQPGA44Mc53+FyMdPb2MNbwzjLvTQe/0u4rHOiZ4oPNRDEdHC2vsyQzS7w3yzNHbeHTdE/zBis+ypm0lO/5217w6RdNP8EbT/gTzkIUW11AMgZTgFT20eh0hwJn0q5W6/1cn8fVx6u9KcF7fRh7s3cLewgAh2+DM595P3e44+WgeN++h1UGozX/u5OYpGjbVM3F/EmvQTz6LsPDnS2dcbMXGGrfYFevDkQ41eg2KEES0CDk7x65MHxsJDEPA0liyYfA8j+uvv57PfvazaFrQHxdwZFhsE9t8j12R6KFQKrIjuqtS9jlWSpIz8iSUOAPZYVzPQyJRFZWUTPPDsds5xlldFbqZOXxn5nhOLaah9qh0fqad+HE1wIFKpOn/L6uupp/OcpY8naPt1QyPjFCbqaUj34rnSrwpF7VGJdRW/dy1x9UQatbp+48BnLyDogrQBF7BQ2h+yGml1osmNIpOsSIBogmNlbW9R/JjCniHsuSd/ZFHHmHv3r1cddVVR2I9AQEVFtvENh8zyz5fW7uHTCxLqjSF47m+URAqIUXHkjZZK89UTZq6aO28PQRzlZJ2fLqV1kuaq553rv9P9zY6Mq3UDdcipSTaHUboAnPcRo3M3b9gNBuEWgyUlPClvAseQhG0XtJItDfCRk7jot5zuK/vEXJ2rpJjCMJIAYfCkg3Deeed5yfVAgLeAF6vdHS5umjHwG5eHHuFRllPqpTGxgFAQcHFQ0qIGzG6l3cSuyIya1jPYmY2HIzp3oYyLQ+h1Kh4RY9Qk0H8xBpSW9M4mSJqWKVuQy1QbZBQRGX4T9f+4T8A157+N1zQc1ZQlRTwugliQQHveKK9EUrhEsWJEqviyyk5JkO5EYYKo4DEcm2aI41ctfZKViR62L2xn6nujC81sWxuqYn5xnMebB3lzd1JObPyENHVEbLP5ZCOxCt5eJZkcssU6aezlQT4wQzSxmWnBTmFgNdNYBgC3hXMTGYnwrVE9DCXLj+fluiBwTazeifaPsLlHL4ii5mbO/ghJs/y2PvNQaSUGK06+VdsQBJeFsIrVQv0BZ3PAUeaQHY74F1BOZktEOzLDiEQXLX2Sj5/3Kf58MoP+p7CIgYAHQ6ivRHqTk5UNvm6kxMohoJbcNEbdLAlkv1jOvfPapg+qCcg4EgTeAwBb2uWIplxYvOx/qhPCSc0+x7C9Ea2sfDSeycOF1X5h4ji90BIidDEvCJ6AQFHisAwBLxtWYpkxsxrJZLYlkhVI5t6uULMWFrvxEId0kuhKv8w6aA3aEgJTsqZMwEeEHAkCQxDwNuSpUhm7E73c90rN2N5Ns3hRopuiRtfvJ3wAyE6ZGulkU2/Q+djX7yEW4buoj87QMyI8lvrL5/XWxi9e2zBDumFDMbMnxf6ioTaQ1VzGYBZ9zhchiggYCECwxDwtmQpkhm377yXvswAqqKQLE7SHmvDsiySTNLTsKyqmezUX55I7cs1JJmkkQbeq62HVbOfvyyNMVeH9HySGuVNPbMtx+TmVOXn5Qlu068vz2aYvvkvZIgCAg4ngWEIeFuyWMmM3el+tgw+iSIUVBQksDc7yLJQO400VDWTCVXw8nPbScUztIabaRlqmjWus8z0ZjWv5IEisKdsstuycxoMa8IiuXkKc8LCmXLQEhqxNVHMEZOJBycJdRhzSnCUWcgQBZ5DwOEmqEoKeFsyV5XRXJIZY8UJHM+lO96JEAqu5yKlx1nL3897L16PEAJz0EQIwdObnuc/Trye7x17M19b8d/8YvUv560GMpoNhCrI/SZH7jd5Cq8VsJMO6eeyleqisidipx1G70lip2ycjIO0JPaE7TfQhRWkK1EiKkIRKGEFK2mR3Zater6yIZp+36BSKeBIEXgMAW9bFiOZUfYsJJKj61cxXkxiqDofXXUxrScd6CkYio1y/6uPwiA0ZxvIxnL8tPFnrGzq5ZjmNbPum30xi511cFKuP+LTAKNeJ7ctj1BFtSeiCdyii5t3EQpIFXChNGIS2m9gnIyNnbKwkw5IGPzBCEgqoaKZGk1BpVLAkSTwGALe1qxI9HBq20kHleIWCJKlFHGjptLhDAd6CjINWYpKiZaGRlShEs/WUFJNlPPFrFBNOayjhhWEIRAhgaIpGO0G0pXUb0hUeSItlzShRlSkLRGGglABBaQjQRHEjo5gjdpYQzbSkmh1KsIQDN86RqGvWFln+xUtVfcNKpUCjhSBxxDwjmcpnkWurkBdbS0TxUnqtARHnT0781wJ6zTq2JMOUkqk64eHtBqN1staaL2sZVb10MB3h/AKHoquoCQU1KhG56fbGL1zHL1R870FFbyCRFmm4Ew6FVVXOHSNpoCApRIYhoC3JYttbJt+3altJ1V9//nxlxGCihxGWeZ7yB0lFptf5rsc1vFKHkabjjlgggeKrlSd4qdv3N1XdSKQjN6TRDoeekKn/YoWQu0hP4ndFsLNekgp8RyvYmRmhooCSYyAN4LAMAS87VhsY9t8192x8z7+84XrGC8mkUBzpJE/Ov6qRct8VymdOhDuDFO/IUHrZXOHdsplqo2bGmnc1Dirf2ExRmYxlI2g6ViENOOQpMoDAiAwDAFvQRbyBhbb2DbfdU3hBq575WYmSil0xZ/FmSyluO6Vmyv3WMxmutiwzsF6D5ZqZOajbARH8mPk7DwxPUp7rHXBbvCAgPkIDEPAW4qDeQOLbWyb77pdmT5ydh4pPYSioSBwhUvWys/ZHLeQkTpYWGexvQevN3dQNoIlp0TRKSGRlFyTkluatxs8IGAhAsMQ8JZhMd7AYhvb5rsupkYpOiUsz8L0LBQUVEUhbsRm3WMpWkzl9U83ItOb4OYaEzqdQ8kdlENUe/VB8naBGj2Gh0dEi2C6pj/3eR6DFxCwEIFhCHjLsBhvYCmzoE9tew9bBp+sXHdU3Up+sP02MmaO8gxCiQQJl/SeN28oqjFcz3gxWRVumslcRuTC5k1HrPdgeogq31TA2KhTrCmioPjznhV//vPBRAADAuYiMAzvYJYiSf1mszvdz1BuFE1RD+oNHCxJPH2T1hSVs7s2cErre/j6C9/F8mwMTUdIHU9KGiP1mK5Na6z6OcpGSlM0+jMDOJ6DJz1u33kvf3nSH85a+5yezoZjZ82HPhy9BzNDVB2TrWx6YQNbTn2SiBYmZ+cJqyHCanheoxkQsBCBYXiHstQwyJvJ9LVarkXeLlKwiwt6A/MliefapJ8aeZZuu5NsNkdDuI6kmEQKsB2T8WISgeC7L/8IT8rKe9QSaUJTVPZmB9GEgqqoSA+2DD7JR1ddTFuqZcE5DrvT/fx88AnO3Ph+1qxfuej8wWLUU+cKUX1g5ymcdenplFaVgqqkgNdNYBjegSxFkvr18HokoKeXVs5cq+Va/O66T1b6C5bCXOGovuF9jPx8DGWZwpSSobmukQExgiNdDKHTU9uFoepV79GKRA8bO0/j+7+5DVeAhkp3vBPHc9j+yE4yP87OO8dh59QeJktT3PibO7h3zyO+UT754EZ5rgqm/MbiLM9oPnmMNctWEG0LehwCXj+BYXgHshRJ6kPl9UhAT/cQJJKslWNd41FVa22PtVbGbS4lHDYz6TyeTqKlVFZPLUeJKfy08WeUshZNjQ3kZYE1dSuoMWJ40pv1Hn101cVsGXyyao6DWlSw73VwS+6sOQ63Je9hd7qfydIUDeE6Vtb1LNooz1XB9KOtP2GL8yRFpVTl9U0vcT2cIaqAgDKBYXgHstjKnaUw3TsAliwBPZ+HMJQbIW8XGMqP0BFrq6zVdCz+5Zn/YsvgkxTsErqq8ZGVH+T3j/vMguucmZwOOSHO3bmRnnAXvalujsqvZDQ9TuMnGvle8UcU3RJRGZnzPVqR6OGqtVdy0/Y7SZZSGHmdM3/1fqwxi+drX6ZpooHeWBfOpMOF6lmcsuFEfj74BDf+5g5W1vUsySjPDA+NdUzwQOejGI5OV+Nsry+Qxwg4kgSG4R3IUip3FsNM7yBxcnzRZZgw20PIWXnWNq5BEQodNW1MmRkcz62qHvqXZ/+L/swAHh5iv9bjN168AST8/vpq4zDTq5ienK6djGPdbWFLP+zSMtREq2hmzYqVlJzSQd+j8r327hkk/40Cr8Z28p1zfkhJNwlZIc5+eQNnWxswmo3KY+/d88iSjfLM8NBYKYmpm3REWuc1MIE8RsCRIjAM71AWK+8AvjewY2A3UzVpupd3Vl07V4gjtTU9S1p6vjLMcr7DNV3aZDOjcoKcnWcoN0JHje8htMVa+JPjf5dkKcV4Mcndex7Ecm0AXM8DPGr0GCXX5M7dP+W8no2VNc5Msp/XtZETWtZVtJF2R/r51WXPMfWrNCtHe+gSnZWwy+Us7j1akeihwa7jF6WneHj9Y3glaJpqIBPNsfnYrRz9zGrWcXTl2k8d9RGue+VmdqR2EzdiVWqu8zEzPNTQVEc8UcOUkqFRHh6vLyBgsSzZMAwODvJXf/VX3H///RQKBVatWsX111/Pe9/73iOxvoDXwWLkHUbvHuNHW3/CA52PYuom8Zdr+O1TPl6pzpmvSathYx3pp7MLxrjLlTnJyUlax5opuSa1ag2p5jCOdKtO6hOlSW7aficTxUkmSyk6Ym0oioLneSgo2J6Noeg4nlM5Nc9Msu+c2sM3tt1AY7iB+nCCo+pW8tTIM4ybSeR6SaPawB+s+CwnnHrskt6j3el+9uqD7Grtp0iRxkIDqqqQMONMNKbI1Gawxi1G6scYK04wXkgCIMTSPq/p4aFjmtdQcszD5vUFBCyFJRmGVCrF6aefzllnncX9999Pc3Mzr732GvX19UdqfQEH4VB7FXan+9m7Z5DdP9vLg6seRWjQnG8k4+b4/jO3snKklzXLVsxbATOftHSZ8kl+Mpdi3JkkHy/QabVRlCb16QRXH/v71HbUVE7Af771/yGRLKtpJ2WmGS6M0RSqZ9AZ8ZvQPIirNdQSrzxmrDjBlJmmPlRHqpQmbWZxPBdD1cmYGe7pewgpQRMKnoCkl+KHY7dzSvrERb9XNz/1E27e+xNKoog4XmBnbTLRHIlCDdm6PBE3RCMN3O8+ym1b7yFVSpMsTdIQrmN13YolV4RNDw8t1qMJCDjcLMkw/PM//zNdXV1cf/31le8tX778sC8qYHEcaq/CHTvv47pXbiady+Id5eEYDssL3SiGQjwbY1wkefFnL0NB0n5Fy4IVMHPFuKef5MMyjIdLOpQlbxRpsOr4yMsf5H2nnkBdmz/w/qmRZ8jbBRrD9dieTVu0maH8KJ7waIk0YRcttLxOxAyzcfA0YloELoXnx19mojjJaGECRQhM10IgGMqPIBBYrr1fC0n4MxOQjObHeWHi5UVttk//+Dm+P3ALniepcxIUWovoDQYiJJmIpAg7Ic4fOIuWCxv5XvJmJJKYHmG06JAqpSk55uuuCFusqF9AwOFkSYbh7rvv5vzzz+djH/sYW7ZsobOzky984Qv83u/93ryPMU0T0zQrX2cymUNfbUCFpfYqbBl4kl2ZPmJqlG++9AOSpRRCghNx8fAYN5I0ZupJG1nCbpjWRDMyLxm+dYw1X15ZCXF4lodiKBT6ivMmPsvlso3hevrcfRiegSNd4m4NIdvg6OKqqnxES6QJ07XYlnwViYcnJXG9hj85/vc42lrFnq/tZTI0RUu4EWfK4eGHH6NlWTMPDW+hIVxHykxjOiYSiSoUIlqEgl0AJC4SRQrAj+vknSLfePEGFKEuaEwLfUUef+Jp0qsytNhNCFsQHY1gdVv8/gc+S0OujrpcgjWXreDF8Cvkn/K7pAdzQziei43D7nQ/daHaBXMDr6cXJCDgSLEkw7B7926++c1vcvXVV/M3f/M3PP300/zxH/8xhmHw2c9+ds7HXHvttXzpS186LIt9tzM9bPTChH9aXlbTftCyyGse/0fu63sER/rzhF3pEdUjGIpOyTMxPZOSZzIWnSBkhbgodTbLrA4GOoYYTY/DgOCEDevIvphdVO9CuVx2vJTEFS5CF+i2Rl2+lkwkw/i5yVmboAAcz8GVLgBZO8dYYYJN5ukUJ4r0dC5jc83j3L/8ZxRkEX27Tl71heNcz/UD+tK/k+ma6IqOIhRM10LiAaAJ1Q9LCTGnMZ1eUvvYS0/yUO9jpI0cGSNPu9qCYimEZYQTmtexYtWB97glfaBL2l+Er8Q0XkqSsbOzdJjKvJ5ekICAI8mSDIPnebz3ve/lH//xHwE48cQTeemll/jWt741r2G45ppruPrqqytfZzIZurq6XseS351MDxuZroXjOUyZaVJmmu54J4aqz3ky3TLwJPf1PYJEUqPXkLNyeHg4roOhGiiaQHVVrui9lLXGauxbHTryrfxs9S/4SfMDFLUidSN1fOapj3HMrasW1bswvTLH9TwUVSGh1zBkjODhcVvoHuI7ayon9bHiBCAwVB0pNXRFr1QgfWDNqahRlf7SPu5v/hmu69FqNZPRs0wVM0yUJhEIf8MHFCHoqunAkx5CCCzXwvZcavQoObvAeDFJjRYjbWaIqGGSpRRjxQke6t/Cnbt/StrMkHcK2K6NElWJ2RFKuslgeJgGUcclbefO2uTLXdLfe/lmvP1GCEAgSBi1bJ/axe50/0GrvQ7WCxIQ8EaxJMPQ3t7O2rVrq753zDHHcMcdd8z7mFAoRCgUOrTVBVRGUF73ys0YqkFjuJ5tyVcB6Ii1MZQfYU+6n86aNn7/uM/O2rR2ZfpwpENEjWC5FqpQsaWDI12KdhEJtESbuPR0/1Q7Whjj1/e+yP+03kU6lEXRBDmzyDdfu4Grrc+zPN49Z+/CfL0Et++8l4f3PsZwfhShKPTGuzFUneteuRlFCI5v8ktLdVXD8VxiehRnWgVSpiHL8iu6eOnhVynIIq1WM5H2MFqNxp7ivsrrVFD2Gwjf22gMN/Cpoz4CUDGoIFEQbE/tRBEKnpQkQnF+/NpPuX/vozjSwfEcwD/ze6pDWmQxHB1PlVhRm0ezT9Cxs40Tm4/lhYmXkRJOaF5HT82yKqPg30MyZWbwpDfLk1uKJPdifkeCBHXA4WRJhuH0009n+/btVd/bsWMHPT3BL+ORoOwlJEuTTBRT9NZ2oQnVL4OU4EoXVag40sXx3DnvsbK2FxBk7GzV92v0KFE9Qo1eXWffemkL2ZY86dcy6EInZIQolUpMuJNsD+2i+ZVGwl0hFF2p9C7MlwRfkejhL0/6Q1bXLeffn/tvltW0U2PEGMyN0J8Z4N+f+2+aIg2c172R97Ycz1De74I2FJ1EKE5dKEFLpInWS1t4z6r1JF66F1OzqE3EGcqMogoVQ9ERiIqX0Rxp5I+P/x0/3LP/NZ3YfCxfefrrPDnya+z9G39Zd3uiNMldfQ8hpZ+fkMxAAcvweyos1WaylOI/XrgOx7NJmRkE/mjQlXW9c77/rnR9L8+xqr4/X7XXUiW5305iiQFvH5SlXPxnf/ZnPPXUU/zjP/4jO3fu5Ec/+hHf+c53+MM//MODPzigit3pfp4aeYbd6f55f15OLnfG/DzC3uwgjuciJXjSY6I4uX8wS5iIHuam7XfOul9zppGwrN5sFAR1oVr+5Pjf4z82fmXWRhJq1kEIhCqQjoe0/JkFEom0JMU9JaTlVyyN1I9VJcElctY6jm9aR1OkgaJbImfl2ZsdRAiFZTXtTJZSfOPFG3hq5BnqjFpqjRrqwwkawvVVdftr167hs+/5GGpIZV92CEPVaYk00RiuR1VUSq6JKhQ+sfoyPrLqg1Un533ZIZ4ceeaAUZiBK/0EvC3n/nkZiUeqlGY0P85kMY0uNHRFY6KU4sXxV+Z8jEBQo8cIadWfQbmhTQiBOWgihDikGc8He+8DAg6FJXkMJ598Mj/+8Y+55ppr+H//7/+xfPlyvva1r/GpT33qSK3vHcliTnkzhfC64530Z/ZVpBZM1yRtZQipIZbF2mmNNc9KPo/ePcazD7+Isc5AD5moqoqma7iei+XZdNS0zhl6OL5pHc2RRpKlFI7r4gmXRCHOUbmViLAAG5o/2EjrpS2VUtPFDtcZzA/jSY/e2i4UIUhbWVzpUR+qw8PDci1+Z+0nq078ZWZ2cz83/hI3bb8TgbKgltKvRp/D9uzX/bnZrov0bKQEVaooJQXFUHBUkxLmnI9pDNfTEm2asyrp9eodvRFiiQHvTpbc+XzxxRdz8cUXH4m1vCtYbJnpTCE8Q9XpqV1WkaPelx3iK7/+Opqi0hprniWZUE5u1kcSxGSUrMwhLIGHhxSSqJydqH7llR0MjY3Q0dLGHx1/ld/rkM6gpTUufGkT7ZkWPEXiCQ8n61DoK9JSv78CKZ2kTtYyJTLEQvMP13l+/GW+98qPMFSdomtiuX4+IaKFCWsh9mWH5jRY0+Pop7adBPgGZzENYLVGHIEff5sVKloCjnTQHJWYFaEUMrGFjXSkX0mlClRU//3d/yyqUEiE4gt2LL8evaMjIZYYEACBVtIbzmJPeXMJ4V219ko+vPKDlZ+XZSTmkkwoJzd76ru4eOfZ3Lr8HtLRLDiQKNZy8vYT2C52wjnQlmrhxvtv48fW/RS1EpFtYT7edin/cd5X2LtnkKl/ytCytwFPSDzbr/aZeDhF+ldZ2q9o4RLzHG4ZuZs+ZR8Rz3/sXJLZ5X/lkMdkcaqyeYa10Lwb20Ie1kINYOUegTNip/Bd40ekrQwcqmmQoHgKp21/L2uGV/CT9z1AKp7GUzy/T8KDxlAjBVnAcmwkHmc3nsHVp3/+iJ3eD7dYYkBAGSGlfD2HqCWTyWRIJBKk02lqa2vfyKd+S7A73V+Rfyif8gSCr274v/NOJFvoRDzfzwt9RXb87S5c08UcsRiKjbCzqQ+hCKbiaZ5d8xKmblITi3HCy8fyq+Zn8QTUy1rS4SwKgn87+0usXbuG0bvHGLxxBHvKxsm5GA06kRUR7Enbzz8Aw3WjZJpz1I7X0Fls5zdf3MltyXvmDZeV1/382Ms8tG/Lgtct9v2a/l7EtkSqegSeu2wb3/ZuJGWm8ZZoHEIYOK5LfS7B1Xd9nvapFl7s+g0/OOs2hIBaUcu+yBDosIw2SnkLzVH5/7b9Pu+9eP0R700IqpICFsNS9t7AY3iDWeop72CSCPP9vJzc3Hf9MNKSdLittKdaGW0f51vrf4BQBE2ZBjKlLFs6n8RULerztZQ8kzqnllF9nF3P9NEb7arEwie3phi+ZYxIb7hSYlnYVQQpWVbbgRyUCE2wVxng5r0/wQnZ1Ogxik5xVrisvO5T207ivJ6N825si/WwpnsVES/MxqdO40x5WqVHoOXhZurPqycej1Ojx+jPDlBwitQaNQCkzeysHoRySMjFQ5MqZ/7mNDpyrUhPorsahqfTajYxGUuDAjY2/d4gCSXObyUvpyPfuujehNezuQeyGQGHm8AwvAkcTBL7cJ0AWy9tQW/U2f3VvUjXw0k5pLQpcqECtW4cS7WpydYw0TZJUS+RjeTQXJUaO0q8WIP3uOTVu1+jfkOC1staaNhQz8QDk1UlllpMxZq0yb2UA0UgBEwcNcW4O0Epb+Lhq6NGtPC8SdGFNrbpcfQ6r5aJ4iQRLVwVbpqZtxlNjvNA56OslatZZnWgN+gkzUkcx6GnfplvYCL17JrqZ2WihydGfj1nD4JAoAmV1mgzFzln84H+07CxEbqgwUsQlmHGoylSkRRSlb7ohoScUSBlpNEbdPak+5nom2RF/fyvcbpQX020pko6PNjwA94MAsPwJjHfZni469IbTq/HTvpdtV7RY2dLH+lohikljY6OUdQxNYtYKYalW9iaQ0bNsWnnBrrcNkqDJkM3j5Lamqbz020VQb096X7SDVlaapuIPRLFr/SUSBUc1SbnFkBIIlqEolMkZ+dn1fIv9n361FEf4YZf3sKe9F5CdogLBjdUhPRgtlfRFGlgj76XsVKSTtqxJ20aYw1Vc5mTpRSaovDy5PZ5K5Z0ReOy5Rfwu8d+khWJHgqnFMluywKCE487DmvE5bt7foTl2LjSwxAGhmNQ0kpsbtyKm3f51Zrn8MY99C06GztP46OrLq763GcK9Q23jPKNqQPS4UFfQsCbQWAY3kIsRRhvKeJrrZe2kFw1xUuD23lu4iUaZQMZL4stbeyITcgM0ZPsxFQtLMUiF82zfKCb0oAJGggh8GyvIqi3ueUJbt77E4puAWVU5ayj3s8Zu05BeIAq0D2dmIhgKhama6IpGmE1NKuWf7FcqG0ivCVUEdJrGWpi8MYREIL4cTWVyqjypj+lZIgnamgw6zAnfEXY9168ntLaA/MN8naBolMia+fmfV5VqLw6tZPnxl9iRaJnVgXRlb0fomNZK//nqX9mojRJVI9iSQvd1ih5Jj9b9gtCNSEUKdibHuT76VvZMvgkV629kstXXUShr8j2zbsoHmPSJpsxFdMv3zVcGiJ1eNJbkmR3QMDhIjAMbwHKm/xefXBR8fSliq/dsfM+btqzfwiOm6KntovuUCdFp8TI5BhaQSUby1GTq8GqsaixYtRlavFMiaIoCFWgN+o4KYcdA7u5ZewuJB5tsoURb5yfHbuVlYO9tKdbGK4ZZbImRX2oDmFQ8RjCanhWtdFijZs1btE20UxP5zKEJSjZJUr7TPq/OYDRoNN+RQufWludt/nkyo/w3trjAUn8uHhlYltTuIEbX72dJ0d+jbdfins+aqklmZ3k35/7DhPFSc7t/sCsBP/x42s5p/4Mbhm+m5yVJ6Qb1BoJzJBJSmZQpMAq2GieipCCfOpAvqVhvI5EOk6UMGk9g9BUbMUhJAwiaoiwFp71+QeJ5oA3gsAwvMlM3+TzTQWMjTpJff669KWKr+1O93PdKzdjeTaN4XpSZpq92UESRhwPj9aGZs7qeT8P9v2cpDFJVIlw7s4P0JFv82vyHUmo08Areb6Y3WsDTBXTNOcasaRFvBRjIjHJVDzNrrY+fnb8VtwmFykk0vZHc86VYF+KcZsuH6GEFcwBE6EIQh0GXtH3ZC788iZO3ODnbdQnFfT/1BkoDCFUQf2GIq2XtXC/s5nrXrmZ3el+HOliCH3+D0bCqD0GDiDg357/Nv+z4yd88firuFDbxMAPhkj/KsOW3qf4+TFPYsR1TGGhuSqxcBRP81BLCp7tgSJxFBfD0YkNRUmTZqw4QVtzC11eJ+ft2chDy7eQUTKoUqFWixPWwrM+/1ljTLs3ckJzkIsIOPwEhuFNZOYm3zbczJm/ej9b3v8k++y5K5aWKr52+8576csMoCoKSTFJwoiTMtMM5ocrQnOXr7qIs5vO4Nlvvkh9KUFPuAuzzcSectBiKjggDEHjpjqeff4l5FGSZG2KxnQdmWiOsBPCNRw2n7gVFOhKdJAJ57Bcq9KQtxhlUb1RRzGUWR5E1TzkIRPpQaTbQItpyIisvP4VvT20pVrYcccupJSgQWmgxNDNJV5+bjs3nHsLk3oKV/qJZkvO0w1ddiLEga89PMaLSf7rqevZs3UvPa8tA+COo+4jE86C7V8vbXjfb07g8WOepkbrYNAc2q8GLqktxjF1Cy2lUzsZJ7rWf11n3Xo6q0Z7SSeyjGwY44maZ2ZVrM01xvS/XriBRrWOulCCT6//6GHJRQQeSQAEhuFNZfomb41ZWKMWp+57D6snl6N/SOOos1ZV/jjLYRfP8hYtvrY73c+WwSdRhIKKLxCXtrJ0xFr4g+N+u0p2Yu3aNTSeW+dvvkk/Lr/sj9urJBvueO0+bsneRcEoUVSL5Bry1E3Vcu6ejcTaajAjFq2lZrSQVgmDtcdmdzHPZdzyr+bZ/dW9CMGcHkS5ZDa7Lcu+7w3hWRIn72CP23iWR3FfibqTE5V7a/UapX0mww1jpMJpMvEsk5kp0rHcIfU/CwS2ZzPkjnLbsfdQuyJO13gH6VgG1VXRXR1bsykYBTzHQ0up6M0qvVNdDEVHKIZ8yQwFOHngeJ4d3Mb20E6O37iONetX0jveVTGI0zdn8CfcDedHK2HGklNiKp/BdR3CE2GKaokbire87lzEwQofAqPx7iEwDG8Shb4i5oiFUAXmsIk1aiFdiTAEnaU21B+rtJ3UAonZYZfo6giF14pzjtqczlhxAsdz6Y53MpwfxfEcpPQ4t3sjH1n1wVnXz6fdU96s7sjeBwr0Ti0jWZPC9Vw+tv0STkqvZzAyTMQLU2gtUhuJLyjPMFNZ1Bw2cfMeWh2E2uYPj0V7I2RfzOLZEjtp+8lxD1Bgz1f3kt2WY9lnO/x7J20eW/kUP1v3C0q6haar5LQ8NoemmSSRvkYSktp8jGKoyLaeVw8YGSlB+sOC4mM1nD21gV+c/TS5pjyN4/Wc9OJ6ViWXs7urn82rH+fHI/fDqKA50sgfHX8Vl598YAMuV6xN36g1RcV0LZKlFJhguRaapxHRQmgljfF0kr17BllxwqFt2JWQo+v3nmStHNe9cnPF2CymWi4wHO8cAsPwBlPoKzJ61xiprWmkK/EsD7fk4Vm+UYh0htCbDYp9RSa3pigNlqrCLuaISeb5HJ2fbiPSFV4wcVvuAZBIjq5fxXgxiaHqfHTV/FpX82n3jBUnKColWhoasUccGjP1jMWSNJ3YQO8JXawwesi5RW5L3rNg497udD9j4QnUyxX0O/wwmJSg1qiE2hYOj5VDUHqdhhpVKO4qAaDEFLAlEw9O0nROA+1XtPD0nc/zs7W/QAKtVhODoRFK+txCd7PYL2te9fV+PCkZaZhAComruGiOiqd4WLqNRFKbi7N8XzedhTbO+cQHMI+xcH/qEno8xGB4mO+vupWpaBpd0VAUlWRxku/8+iaOsVazdu2aqvdpZoVawS5iuRZ5s4CQCjVuDMMzSMcyhKwQdbnE4l7fHNy+8176MwO40sOTHqpQUITC7Tvv5aOrLj5otVwg//3OIjAMbyBlaYnivhJCEYSXGWgRDaG5KGEFNaKgNxsUdhdwUi7Dt4whNIGbd4kdFcUetzBHLKQlGbxxhK7PtVN38vybwfQu62QpRdyoOWQtnbKRydUVqKutZXRsnEguhPaMyt4nB2m/ooUrL/0Qp6RPnPfUWLV5GFE+9sVLuFA9C8/y2PvNwYOGx6aHoMwhs7JhCwSEBV7eo7CnyLLf6sCIGth7HJom67E0m5yeR1X8YT4uc8+uqGKaMdCE5nsEDjiqi+d5/nNKgeEYhK0wjuYQMcNc8OxZtE+2IGoFy2u6qWtLwFVQ2FTkpy/9jInUJLZ08DyJ4YJ0JOlcmme/+SKN59ZVwmdzdXznC3mujHyYlmgjW3/+NL9qfZbR0ARhy+D8vo20r2hZcBb3fJRDjgh/SqNE4koPRShsGXyS1XXLp4WxTBQUUuZUpVpqqfPHA976BIbhdbKQ+zw9L2BPOgzeOOKL0KkCFIk5YhNdE8EretS/L076uRz51wq4GRe9XiPSG/bDLDmX0t4S1qQNrkQxBEKlKtwy3zouX3URx1irK6qpa1etmfkSFkWVdHZpGKWgcP7AmSxP9FSFflb0zt24N9fmcVvyHk7ZcCIrEj2VJrz5wmPTQ2/2pI0SOTBKRCKhJEEBaXvsvW4Q+ykHY51BOpzDiOu4uktIC9EcbmQkO0bRK1U8A6GIg+QdJCER2m8YCjiqX3WleIJiuMTxe9Zx2vaTqMsnaE/7G7sa8ZPolTnSmsUvvWdRhIIiFTzpUfJMNKERk1HqS4mqz3OmcurI8BhKSiFxTw0tXhO/tfojnLLtRJJMUpuppTPXykD/0CHNji6HHJvDjQzmRypyIM2Rxsr8j5geZefUHtJmFsuzEcDdux+iJdIUyH+/AwkMw+tgIfe5nBcwxyzcnIswBF7Jw2jT/QE4EjzHw57wpZuz2/JIR4IAJaIQWRHxSzLb/Zi7k3PwTA81pBDqCGG0GJVwy/3O5gXXYd1qkUjXUtCK7L1kkO6rOud8PdP7CoBZuYaylMfu5/op3lNkeaKHocgIyZ4UNQNRese75j2tHmzzmJ7fGIqNsqdhH/l00R83Oi3H4lkesgBqSEGtVfEKHrK0f95zVGHkznHspENLfSMXdZzNT5ofIOVMgS5IGHE6tTYmSikUw0Tiz1WQjsTV5vciHOkS0QU1xFCLCgWjhOIJf4NXPHa39XPhs5toT7UwXD/GVE2a1Wet8D+XrXeSK+TwHEmePD21XezNDmC6li9/7oS5bOwClnkdmEmT7LYs0d5IlSHeOzmAklI4f+8BQ1x4rchpf3AS9qTN4A9GEAlR8bbmK18u9BXJbssxvbcDDniDquJLl9iegy60/VLoYU5oXsdYYYJvbLuhUtHl4XH37gd5YeJlLuk9b0H57yD38PYjMAyHyELuc1vKL610TRcn74ArkZYfm7DHbPQWHXvMpizPI6VfDhpp8z0EM21hjpiE2kIUdhfwShI1piIUDyWiYLQYlXDLUGx0wXVUVFELLrgw8N0hBJKuq5ZVvZ7pm69reggBiqHMOoGuSPTQ1tvCDm8XD4e38FDvFgqUiLSHyLlFruRDc75fi5kdEO2NHDByO3wj97HGSzjm1lVVpa3SknR+po34cXFKgyXSz2RJ/mwStUYFRWCN29gZB+lI3zuTEBVhHM/hhamXyYbyled0lIWntgFoqOiKQawxRsv2Rl5pfw0hBEIKmqca8TSPqdj+Po4TtuIkHKJtEfIvFKkpRqkbTTBuJMnG8sSdGo5rPobB9DDOpMPv/OaTHD+ylvxAHunB03c8jzfh0b26iwuX+b0Z0w3x9ByMYiiE2kJIV2I0GAvmZ0bvHmPvdwaxJvzku96o0/P5TlovbakyQlEtUjlghLVwJfR4Qss6GsMNRPUIg7lhhBD+wCfX5qF9WziveyMP7d0yK78U5B7engSG4RBZ6ATcMF7nb7CW659mBeBI1FoFryiRpiTcGabmuBh6ncbEw5PoDXrFQ3DSDm7RI/9qvkrmuri7iJ1yKPYV0RN+x++ehn3kd8y9DnEXFPeWkLYfZhEhgXQlo/ckadzUWNk4pvcVaPUa5m/ygGDqpAxJUgzeO8JZ60+vqlKyL7e5Z+hhbOlQbyfwmjx+OHY7oZ3GnNPXFqMqO5exvXnvT/g95cpZm2KoLVRJlCuGwuTPU+gNOl7JQ9EFg7ER7m/dTJESpmbhSpdsMT9LLG8xuLiEVZ1wJERmWY76qXqElCTycfKRIhLJaGKCx054ilCrQU/7MvozA4wXJqhN9mIpFmE1Qsg1MEsWE+EkjTUNXKKcw/HD6yo5p1++/xke7Po5xYxJ5NdhLrj7LD654UO8f/3J7PB2zZuDOVj5cqGv6B8QJh38nj6Bk7IZvHGk4llMF3Y0HYtJM4WUcELzOsA37PXhBFk7h4eHKhU0RatM+juheR3ndVer5Aa5h7cvgWE4RBY6ARvNhh8LTzq+UfAAAW7ew2jW6fn9ZZjjNpObU9hpBzvpIL0C0RVR7EkbYSgomsAp+o9VYwpCEURWRKCvRPvHfaXTaG+EfLo45zpqJ+Oktqb9xe5PpEpTooQVpONVnSinJ3WdjANC8Njqp/j5micoGRahksGOV3ZzWvikyh/9o91PMJobR0ElI7IYmkE+U+Dfn/kO9aKOK7s/xCnHVCeiT2w+FoGvwDqz6Q3mNrZ7CwOkE9kFN76Z5a9qrUo6kiWr5snpeRRdQdd0Mnb2wPsxT9XRXEhgqDBGsjRFZ6KdjYVTeDr7AsONYxQNE8PSuPuUB/HikhPa16EIheZII0O5EYYio1i6hSMcFBQ27H0fl6w9jxUn+rmY0fA4/d8cYGLlJA+tfAyvKGnKNpCvz/Ng16OsuXcFZ60//UCD3xw5mPLPin0lhCZouaSpyluwxi2cvIuUEkUIPxfjCZycU/V7MFeZ7PRT/qeO+gjXvXIzruchBXTXtFN0S5Xf+5nCkEHu4e1LYBgOkQVPwAmo35CgeHPpwAMk4ICdssnvyJN5Lo+UkkhvGDxZ8QTUsIp0JeiCULtBcXeR0oCFWqvhFT30hFYxCgutoyPVyi63D6NdxxywKhuhEhboCX3ejVUJKwzXjrD52K0IRSFeiDEcG+O7yR9x9+MPUR9OcF7XRr9xTlGR0qPo2RSsIgAi6WsZ/cfk97h+ZwxRI4jpUY6qW8n2qV2VzcY7Ss7aHOYytjXRGo7atBJxh5g3MV3VGT1oYtQbHL1pFZFImCknTdgIU3JKVc91MGMwF6ZnkbNz/O4nP4V3p+QuHsDFoxh2KQkTKeDVyZ0c3bCKoluiTk+Q0tMgJLrUiVoRtrfs4vMtbZXXHj+uBqNBJ0mKoijRWKj3p9q5tYyFJhgzJ8huy87bY7I73c/Y+yZwJ1xC94SQjsfk5hShJr0S/jOaDfAk0pK45n7vURVoNdqsyq+FTvllr+L2nfeyZfBJHM8hpBrzVroFo0ffvgSG4XUwc64C+F2qLZEm2i5rIfloqiLhUEaWYOjWMbQajZq1sVmeQHF3kbGfJnEUUCYV1LiKk3YxhyyMBp2GTfVY47589cyk8PTTeaGviBpVkVk/hOVmPVDAqDfm3VgHbxzBnDTJNOcxwxaqqzKUGMHUyobFF527c/dPsV2H9mgL/dmBAzOOXYW0kSHqtTAVSqNlNFYmehh1x7mv7xGW1bQvGFKYz8idvOpECicuLLg318b50Rcv5hvbbiBn51GFgoqKEAJFCKx5pLYPRo0eA+DZ+m2IjECpDPTx3ZDJUortqV00Rxq5bPX53PfKI0RTEQzbIITBZOsUmYZs5X4j9WMMf3gU+zGHsGWQjeZIWHHSZDByOvHRGIM/GAHpv8bpr718ss8Vcih5hfNXncm5pY1zJ6BVgdDw5dElSClpurBx1nt5sFP+ikQPf3nSH/LRVRcfNKEcjB59+xIYhtfJQu73e846lqGbRg5cXD6leuDm3UqC2Z600RMa0eURxu6aQCiAKpDSw80KQq0GXb/biTVuktycYvzeiTmTwtP/4KK9EYSB3x2831uI9IY5+l9XL1znLgTNSgOhuMGwMubr/3hyf+16mvZYK2kzg65qSCnRFQ0J2K6N5mg4wmFSmQIgUYyDDZFwBEc6RLTIQUMK8w0xmq/xbjozr/n99Z/hpeSrPDr4OLZrgxBEtBAJI0GyNEnRLS1wt7k5senY/eWdDqqi4XgmqlBwpF/V5O0/BZzXvZHzujfy1MizuDGXOlnLlMhQG4rTEmlid7r/wMlbdYlsDHOsejSvDO9grDRByApx9osb6JIdCEPM2uinn+zblVaGvTEeWr6FdX1H0dHQVpWAtsYtv5rt2BrcvIv0wCu61B5XM+v1LXTKnynVMZO5qo8ONpQq4K1JYBgOA/O538ecu5rQ5hClIRNcDtTNq6CEBNKhKjyiGArSlYSWhbBGbKTrb8qNm+qJH1fDjr8dXbSq6uTjKfKvFhE6CF1B2h7moEVpsDRnGePwrWMIQxBti9A52c76vrUMrvBr2hUUdEXzheRKSepCCc7r3sg9ex4CQEFQV6glbxSQQmI4Ool83Pc0dCg6RTShUXSKeDLBeDpJyAlROxmHttnv56GOqpy5Me1O9zNUGKU+VEfayuB4DiXXpE0LLegx6ELHkfacnQ29iW5aIk3UhRKYrsVE0aoYBU1oGKqOKhTu2fMQ53VvrJyYhwujhGWEj3VcwnPjL3HdKzfTnxlACIXueCdCF+wy+/nYby5FyQhiw1E6s214qp8XclLV+YDpJ3sp5f5xrBMktRTNk42VPMz0/g+v5GE0769oMw6EE2e+b3Od8p8bf6ly8LFcCwmEVKPyc2DWwWi6QTi17aQlf54Bbx6BYTgMzOd+ZxqyLP+dLvr+awB7Ylp5quurcDZsqKXhAw2V0Ecl/CMl0dUR7KSNoiu0XtayZFXVwp4i0pUoMRVFEXj6gc7ghtPrD1y3X3rDTjtVs5w37jqVbSt/g6P6PRij9gSu52EoeiUZeV73Rm7feS8Pv7AF07SIFiOs37OWD+w4hZ1tffxi4y8ZckeJ6VEu6j2H7VO76Bveh5ZSOWf7GRRuLrLvkoFZpbOHwlweW2dNG1NmmrxdwPU8PCnxpMtr6T3z3kcTGkL4xsH1XNxpVUy6ovE/O+6q2vBt1yZr530ZCUXF9VwmS1NMFCcrchLZ57JM/SrDytEelLDKtzb+ACtq++KGispwfoSj61czbiXRigrHq+soFIpIxUO6YCftWfmAqpN9pJ5Ca5FIKkTNQBQhBO1XtJB9MTur/8MrelV5mpljRcuf7fSmyHBziD/f+v+QSBrD9WxLvgrAcY1HU3RLXPfKzQAYqkFjuJ7xUpJ/f+7bJEK1OJ4blKm+DQkMw2FgIfe7HPvu+6+9pB7zq4QUQ6DWqmSey9N2eVtVGWg5ieqkHLQarSofsFhVVYDo8ojfSFfy8MIKsuR3XEeXHzAi+64bYOTOcdyCh1t0wZNEVkSwJ23a0i2c/8yZPNi7BVM36aht4exjzqgaTbki0cMXG69i7WOrGc1NUFeK05ZqBUfSG+7i0lPPJ9OQrZxEX3llB88+8CI14zFaB5swbZN93x1GIuZtulsM83lsf3L876IpGgWnOGeZqoLAm+YXKCg40gEJKqovAWIX0PdPoHP3S2+/MPFyVYikPz3At176AWPFJIaigRB4rsdtr93Lg7t+jjvhEu4O8cHY2dQn68imc7TFmkkqKaT0cPZ7YjEjSiMNeEWPUJtOacDyw3i6MisvVD7Z3/ji7fRnB4hFo3xq1eWcceqpld+JHX+7q8rDdPMuDWfWkTiplobT62eNFS20+kOEurd1oN+hU19IYEUt+j+8j7zqH3zSZtaPiAqw98/4eG1qN1JC3BD0Z/ZheTaW6487Xdd41Fu6TDVovpubwDAcBg6WZIv2Rui4oo38qwW0Oh01rPgDZ+Y48S+kcLpQyeJMGk6vp+n8BiYenMTL+0ah6fyGirew97pB9n17yO+23o81YYMiUMMKUsJZY6ezTjuKsVKSBrOOszacTjQxrRqmOEF4IEx7ppUOrRUzY/sxdiFo3FTP8rXVf2gd+VbyfStxpvwyXiWq4BU8xu6ZoGlTw7yv5WCT3ubz2EKawZkd7+eGzP/MvqnkgFEQoAsNVag40mVZTTsIKNpFiq6JrmioqobrmH40UB743Fckeji17ST6cwN8/ze34UjXL+f0PKasNHkKtDstoMP9rZu50v4QwlUYKyZ9qYvCOJ70PbFPH/tR3qut9+dzOxDuDFG/IUHrZXN/zt3bOjjzydPwLI816RWsO/EojMv892jq6XSVh+nmHUpDFmP3TJD+VZbc9gLbnzswVhQJ0dEIKXWK7U/sYp08qmJQvAclkQvCDOVGUIQv5yFQ0BXdrxzTY1iuxd7sIJriT6oDKNhFSo75li1TDZrv5mdJhuHv//7v+dKXvlT1vaOOOopXX331sC7q7cjBkmxGs4Ge0P0TYFhZ8MQ/X6J1PqMxH2v+biVN5zRQ2FMkujxSMQqFviKjd45VGt/KuQ8pJU3nNhBdHmbg+mH0Bp1lVgedtGNOHDBi0/+gFEdh3frVnDV4OstqO7AnDoS/ZmI0GwhNwbMlSlQBRyJ0gXTkvCGxxUx6W8hj29R9Onfu/ilpK+P3UADSo7J5eYqH6iq0RJsZsybQFI1aowZPSqZKaSJqGMuzce0invSIGzU0hutnrfOjqy7m7t0PMVlKYcsD3dSaUBmLTdBSaKQUsnlBfwVTN0k7GcgL6kO1XLri/AOe2CoW9Rnf/NRP+P7ALRSXm0ScEPazDvU315Hamqbz023E18dRoyrmiImTcbAn/DW5RQ8r5RuIeLimMlY0QS1TMkPIDpFIxysNl3qDTttgMyvVHh7OPeZ7VAhiaohkKUXEC/OJ+IfYU9jLLd5dOJ5EVVUMYeBIl6JTIu8UjkiZ6us57QfNdwuzZI9h3bp1PPLIIwduoL07nY65fikXSpou9cQ/H4upzplOw+n1NJxez+50PztGdtMSaaJhvA7POnBarjThAdHlYeLHxVGjY3OGrab/QWmKxl5zkP7lAzzT8BIX7djEOTUfmPd1RXsjtF7SyL7vDuMVPIQu0Gs19MTsenpY3BjTQl+RhvE6PtZ4Cbcl72F3uh9N0fjIig9WPovueCdjhRBpK4PrughP4fzxjSwvdvNKzQ62h3aR1XM40kF6Hi9P7sDdn1BWhUpM8+P2pf1VTF9/4btMlCZnnS41RatIZYBf3GtJG1d3GYyPongKW9t/RXOskd6mrioZ9JkVZTMFBKcbit3pfm7e+xM8T9LqNJHy0vzs2F+waqyXHnvZgfdodYTx+5PI8metg1DAzbjIGHTSznl7NvJgzxZGlDEibpiPGB+kPduGWTxQMTfSNM4ut59l8fbKDG+B4GPmJTQ/3EhTXwNRNUzLBU24hkdLvJGJyCSTpSlS5hS1xPlI/IO0pfz5IoeD13vaD5rvFmbJu7qmabS1zVFK8i7iUH8pl3riP1zcsfM+bnzxdvKWL3f98Y7LOCqxAnvKZroCtV6vV8TV5jNiYyP+H1RjuJ7+zACKK5CuwFFsHl71C06rfW/lRD9XCKjrqmVIBGP3TCAdiZ7Q5jUkB0u4T/cmjomu4v2XnMRDocewXYeH9m2hOdpY6di9afudfsWQq3LaC+/lw3suQG/Q2fDa+3iu6WVu3vBjQoRImxlKnj+3IayE/JCSW6IhlKAl2khHrG3O0+VYcYKQarCmbiW70n3Yro2De0DiW4GwGiKr51nZ1IsiFEKqwaQ5xfPjL8+7Gc3lMY29b4KSKFLnJBAOxM2a/XO3M6yq0XFSDtltOQqvFdEbNewJx/cO3f2zhCyJHlFpvaSR0+94H13bOpiKZmj06ui0O3AtBzfv4aRdQi0GyvmColKiK9axP5SUYO/kANpjKm2pZsyiRZvdzAXPncnmEx5nQk5SF03wifUfomdHJ96DkraJZnZEdy1K+fVgnsDhOO0HzXcLs2TD8Nprr9HR0UE4HOa0007j2muvpbu7e97rTdPENA8MSMlkMoe20rcIr/eXcqkn/tfL7nQ/N/zyFqxJm7piLZlIjh8V7+T/fvLPCH3PwBq3QUq0ep3eP1xWWVvrpS0kV00xNDZCo6wnVBOi0Fekpd7/gxovJrEdG2ELVFQaqGfKyLDnub2c2HdcVUXMzBBQ91WdNG1qmNdAbhl4kl2ZPpapHbREG+f0XGZ6E/2lfTwy9gsinWG663w5im+/dCNN4YZZYb5YKMJw3zSj94FmFFXh6Pgq9qT3sjc3iED4hkTRyFhZHOnSEWub93Q5fShSTIswMa1Hwpex9igJE8dz+M3kDmzPwfIsQPD/e+7bSOSsw8V8HlPtqjg10Rq/Emko7M/dtg1aoo14Jb/qCKRvVNtCuFkPT3hIS+IVPRRNofWSRho3NTJ+/yS9dhdCFZSGTGxsatbFcNIO0oXuP+ik9tg4sa3Vm2hYRkik46hhBc/ywIPTt72PlSO9pBuynPSF41kTX8GOH/sJcL1zYeXXMos5dB2O037QfLcwSzIMp5xyCjfccANHHXUUw8PDfOlLX+KMM87gpZdeIh6Pz/mYa6+9dlZe4u3MYn8p3yrVDnv3DJJN52g2G9HCGrVmDWNTSSYik5xyzXuwJ/1W2OkyzLD/D3TPnaQn0mgpnXN3nsE5U36Y6FNrfc0cDw8EtJktmKpJVIZJpONkt+UOGgKay0DuTvfzlae/zi9Hn8WTHprQ2HTR6fz2fR+f5bnMTK5mmnMUlRKtspnRwjjD+TFMz+Qrv/56JexT+RwurY7j19Yf2Phqjfj+TISfiyh6RTRFJa7HFjxdljea6165mbTldzbraDi4+/vFQZX+gJ+8U0DiV0Vpik7GylaN0Swzn8fUkW+tbGrpUIZQVufsbRvoSLchon6pqt6g+7OF0g5Gm445YIIuMJp1Wj/SQtdVnUw97U8RVAxBadAfAIUAa8wi0hupKLjOtYle2f0hurxO7Ox+r3N/KLJtrIW28Rbiv4xhhZZWYr3YQ9fhOu0HzXfzsyTDcOGFF1b+v379ek455RR6enq49dZb+Z3f+Z05H3PNNddw9dVXV77OZDJ0dXUd4nLfWOba3BfzS/lWqnaoyyUI2SGysRwJu5a0kSVU0DFvsdmrD87p2pf/QM1xk7q9CdLRHA/1bmHVr5fDrXDhlzdx4sZjufmZn/Cz3/wCR7gY6Jy3ZyNdXieV0+oiNwTw37Nvbfs+/dlBACJqGA+Pzd7jfPDqc3ife0IlDzH1dBrP8qrKd2vHa4i0hxmV44xmJ3CkQ0gNoSnqnJvLdMO0ggMbX2YqS8yMUtRKFN0imqpxycrzeG/r8Qc9XV6+6iIUIfjXZ75JrpRH2OAqHlLxN1xbuqhCwZO+lHfMiKEKhaJbImvlZx0ujGaDkaZxJpmixWikZaip4jFd3lu9qbVtaqkYuuyLWfZ+cxA37+LmXNSYQrgzPKvCqSz2WBowGaobIdWcpi6boHOyFSWkVBVHzLWJjo6Nse/64eoPcn8xQ2prmvpTE4tSfi2veyy8uEPX4TztH2oz5Tud15U5rqurY82aNezcuXPea0KhEKFQ6PU8zZvCfJv7wX4pj2S1w8HKNudizbIVXHD3WTzY9Sij+jhGXufsl89gebwLr+jN6dqPFSfIFXIkRmuRQMKsYbxmkqSXojPdhjVusaK3h/+96U84L/0Btm/eRSIdp8vrpP2KlgWT13NRfs+Krh9yFPiCdTEtStEtMhAe4ry1G2fF26OrIxReK2IOmnRG2/l426XcrN2FWTIJqSGWxdppjTUvKsxw+aqLiGdqeGrLM7QnW1BDCgM1wywzO/nEBy4j2htZ1Ony+KZ1tGhNyIIkp+fxlHJXIwghKpIZQii4noMrxP4egNisE+/9zmZu2HgL2XSOkB3iAuHLcM9UQwUgQaVJsuytxY6KYo6YeEWP5g820HjWbG2kyMowD0W2sHn9VkqaScgOcfa2DZyTnV1EMHMTbb20Bb1RZ+eX92AlbZSQP+5UCJCOh2IoCxZczPw81csVYsbiPIGDnfbfKh7725XXZRhyuRy7du3i05/+9OFaz1uCg23uC/1SLhRqKv/8UH5Z5yvbPNgfQLQ3wic3fIg1965gzJwgPhpjRW03WruGjMg5T/ItkSbCMkI6lCFu1ZAJ5wg7IRKZOGL/yMoyJ3/4RNadePQsg7WUCqzye9YcbmS8kAR8zaGSW0ITGitre+eMtxdeK9L9B50ohr+mE3qPZcVAL1/59dfRFJXWWPOiwwx37LyP656/mWxHjlCTr1N09rMb0Gr1yvszc2Ocy1CvSPRwefwifrjvDhzNxZI2lGX2hC+4F1HDxPQoWSuHBJojjVy19so5Z1NodRrLa7uZKE6ypetJLtt4/oIHhHL4aWLlJCkjTU0uQv2+eoZvHSP5s6nK703596nfHWDze7aCAm2yhSkjw6PveYLzjz+T1rMOPh604fR62j5WYOC7Q0hbouig1qoVBd+6kxNzFlzM9Xnqd+h87IuXcMvQXX7TnhHlt9ZfvqBI31w/Oxwe+7vdsCzJMPz5n/85l1xyCT09PQwNDfF3f/d3qKrKlVdeeaTW96Yw1+a+a6qfnw8+ARz4hVyK1PDz4y/z0N4th/TLOl8ScnPLE9yWvOeg92y9tIWz1p9OdlvWHwOpC6Qn5z3Jt6Va+LByAbeIu5iomyRkGmzadjodmTZar5h96pwrX7CUCqzpidvmSCNjxQkkEiEEF/Wew8ZlpzH1dBo7baPV+cN4pk8xqzv5QA3kxmWnMVGaXFKYYXe6nxtfvB07a9M01UAmluNn67eycriXjmwrxX1+Inn661iov+Ly1RfRcFsdOxK7uXPFT5kMpRBS+J3owj9s/OV7/pBkKTVrNkV5QxrOj1b9DrZGfc9n+yM7yfw4O29fh9Fs8NiqX/Jg16OUNBOjXudsuYEL1E14JW9/85xk8MYRhAqFNUXMsEVTqgEMqNNrmWydwjzGmvf9mkn3VZ0I/AFQ0vEqQ6QWyifNzJ8oYQVzyOSkh46jdqCGJJM00sB7tfWwatFLWbLHPpeRfSuFgt8slmQYBgYGuPLKK0kmkzQ3N7Nhwwaeeuopmpubj9T63hRmbu47p/YwWZrih6/ewb17HlnwF2WuUNN5Xf7Yw0MNL2W35bAmbUIdB8Y37tlfy67GlEXds/LHKVnwJF/e8NYXjqEhVMdkOE1DqY7OUhv1lyeIHxen0FdcVChrsRVY09+zmB6lR+tkXeNRfHjFB9m47DQAMtty2EkHa9xG0X1JEaPemDM8tdSk4lhxgrxVIFGIg4Dawv7yz5o0HcVWfxMVVN6v+Pr4gsn1aG+E9168npbvNGKOmdz13gdIx7IIKWiJNnHV2isrr2s60zckTVExXctvIlPDjJeSaI6G95BcMKk/Uj/G5uO3IiehaaqBKT3L5vc8wYkDx9FRbCP/ap59/z3oh340hVgkSnRZhGxtjpaGJrI1+YoC7FLoumoZjZsaFx3qnD4DxLM9zAET6cL4/UmaGxroWbFsUVVMc36Wi6xYevrHz80Kg+Y3FqsMy/Tqtrk+s3cqSzIM//M/c0gLvAOZvlHtmupnsjRFQ7iOFYmeRW3qMzemseIEP959/yGV143ePbZ/LKONM+UQXmYgdIV0Q5aSKNIdXrakey50kp/pmXRPdtKV66DzM35eIbl5iqnH0/N2IL8eFtrMC31FJjen0Os17IyDZ0tkyqX98rp5N4ylJBVbIk3EjCiZaI64HSMTzRN2QtSVEn5uQKPS7DV8qy9FXnXajSiYQxbZbbnKeuLr46gRlXPSZ7B++1q2R3YhgPM+dyZrV62ZtYa5TroFu8h4YYKUmUEAjWo9r0Z20hNaNm9Sf6w4gRWz6ahtxctJxD7BmJYkqaVoHK7HzXuIkAAHPMej4Td1bIqdzs/WbWUsMkFNqOaQE7kj9WOMhfd/fsz/+PIpvXFTHeP3T1LaZyIUgdGiYY3buBm3yiucq2hhvnDaYiuWKp3jx5hECXPeno2cdevpTHVnKoZlvuq2d0OY6d3ZtrwIyhvVzwef4Iev3sGKRM+SNuCZG9OhlNeVZ/V6tofRpmOP2RT3mkS6why1aSU1Rs0hleyVT7W70/2MjRz4BZ+vPBIEyc1Ti5b8PlTm28zL64qsiBAyPbyiiz3lEj9u7hLpmRwsab8i0cOn13+UG4q3MKFMEioZbHppA91eB8QFobbZ70f5tCttryJ2N/iDYZCS1kv9CiHpSsJdYXqUZXSXOiulpnMx10l3qpRBV3Q6Yq00RxrJFwo8vGoLR7+4kp5w16yO9PK85pgeZYoMjS31FN1SRXVVuiAViZedJijowfufO5n3v+9k1DPUQ97sFht+mRmCqzkuhpN3K96wPeUbfrfk4RZcRprGyep5utOdlXUtFMZbTMXS9M7xNtlMWs/w0PItrBrtpSu3jJgeZSg3wnB+bFZ123ghyUP75g8JH0qByEwOxz1eL4FhWIDyL9O9ex456Aa80CniUMvrRu8a8+PbCiiagtGi45mSzs+00XpJC5/aeegle3P9IV/YvGnO8sJDKT89nMyc6ezm3XllNGZ+DovRWoIDB4G9ewaJ9UXoPnoZeoPG3m8OVr0fQvWFpcqn3eI+E6FAuDuE0A8M1Jm55oNVZs110tVVDcdz6Kn1vcKoHqGvsI9UOE3bYAsjTeMo5wueyjzLQy8f2KzKY1T3ZYeI1R9QXS3uK7H7X/r9ngOVSu+BGlc5+rhV1LUdXK9irk2r7O24pkubbGbKy8zpVc+VK8tty6PFVH9sbYOOXqthpxycKZutxzzN5uO3YvXZxAb3/45qmw7aI3OwUOJY8UDnOBIS1DKijZNOZDl72Qo+1fYRvv3SjZhedXXbrql+7tz9UyJeeM7XudjftYXYd93ArFzN4fTMF0tgGA7CYjb1xZyWlhr3nnw8xdj9SZASIRSk9LDGJOHOUOWkXL7nCxMvIyU0husro0XLg2rmer55E3Qbjp2zmqhcfmoOmygRFa/oLrjJLZaFjOnMDWgxVU7TP4eI9/9n783j5CrIdP/vWWrfuru6q7d0p5N0EkhIQoBIMIFIAoSIASXKDDK4IONyYVxmdOY6zrhdRrzz06szOqKjAsoAKgQQDHsSgkEQZEkCAULW3rfq2pdTZ/v9cVKVquqqXrIAgX785IPdXcupU+e8+/s8Ti4z1rLktwsQ7MKkMp3ZgZnMPn0mnH7kd2pYLbyvkTMwTei5pf9ItJvUcLQ4kL0ypnFkyqtmWWBKk1mVrrPL57yfR7u2lTgLf62PMz63mOd7d7IxsYmYmSC8a5Q6Zw2dNbMIZyO8Ht3HF5Zci0O2jzm3kkdCUzXLKRzmyZJ9lZ1sOfJlTS1pUcLnyfr2HzxEdCRO/UgtWV2xqD4ao2Oy6moZad2qGmLPJVB6FWy1Nho3hBidF2Xb0NPIDplGZ0PhGp0T6JhUkDJeKTHkqi9sjrsHXUTNOC7dwfzVc3B3uNjAJdQ768ZMt9kkGSWhUDPkJ6sr2LERbojQdaCXpppQicNS+hW6b+nHFrSV6J+Mh66be+n5Rf9hqnVrQOREZOaTwbRjmATGM+pTmYIY72ItNpKebS4O/qgbNWwxYpqagSALIJjUrgyUXCR5Za38FIvX5qHJEypEjZWc1XgNuuWXnlmZ9nuui5FHRjF1s0DhfSwX63jOtFLklVqVIdoepyYZYN6M2RW3pvPfQygVZGg0zG+0+/AlPMzyt5cYkcSuxKRT9XxPJrErSe+v+xHsQiEDSO5KIXtljKyB6R475TVVbqxK11mDK1hwFi7DyeW+9wNwb+4hBIdAnVDDUGaEmJIoobh2yPYxqmn2BjvOZgc5WUCNqZgaCKIw5pqqhPTBDId+1osWsSRStZjG/h904ai3k5LTyGeKRKUYtbYaokIcOSKNUeirlkU1XhYqiFHlz1PfwACZkSxtztJrNOqNUeP2TzoTq4RiJxx1xHCaLq5s/yDLli8tPKbSdNvF0vk8En6CqBwnYPiszzkikfxhksGlQwWHlRvKkRvMYeRM9n+vCzWsThj1pw9mLP4w4wjzsB7XUUX1TcvMizHtGCaJaka9mpF9afiVSWcHG/du4ubdd5LIpfDg5sKnzuXsxBlHmE8PU2I7mxwldNZ5Y5jVs2R1BROTlJpmKD3Cnug+2n2tFZ3VVCgF9scO0XWgl1RvmqbWBiSniJ41SL+RqTqdVB7tV2IHreZMmyKhMaWCO7bfxzbtaTJi1nIiTZezgcr8OS1SI5mBLH7Ty7AnTMybQOlRkH2WAdcVg95fD2Dq5oTpfvFxO5rsmLqJvc5eNdqtlBVMlRur/DrLO4vXH99bIKN7oXUn8bMTdDS3kdWy2CUbOV2dkOI6n3l1/Xcv5mEaCxOT8JYI3rnucY1XYlcCNawi2AREu4CeNTHiOrpbp71uBqt3rWTLou0MuoZxGU4ueP08WpaX9lMmyvwm00Run9WK5wpXxdeYSm1+Mhl88WOkp0WEX4vkfDm2LN7OoG0Yp+Zg9SsraYqGiGyPIUgCSr9CbtDqLwl2AUGmatRffLy54RzmYRp6NBNkwWIglsVjzsyPBtOO4RhR6QLO6Tl+ufuOScka7o8d4kc7bi7MtEf0KHfN/QPtr7XSkmjCyB5pFAZX147ZUE6pabx2D4ZpICKS1bPksiqaqZFRlYoN8/HKY8XR+pOdf2bLku1kJQVxgcDFfatZE1mJXFO9x1BtO7mcHbRaxlI3XFNSKhhqGeHh1q3YNRttwYn5c0ZSozh1BwlfEhdOGn1BTAOUPgXZa8l2CnYBe5193NJSea03uLpm0tHu0aJaaa0pEiJ+b6JARlebDSBHJIbdYRoCQQJ2X4HiusYRGLfX5FvsQ5AEBFEAu4kgimgRld7bBiYoWQiH/2uxPGGaltKdR0Jyiqx6bTmdgx1k5ynUZgK0ZporGrTJZlHjlnAvHatZcTT1/clMrs0OzKQpEmLPxn3oss55B6zPGXHGqFUDtCabsc2zoY1q1K2qIfxEFCNnOQVXqwNbg73ivVJ+vMHVNdgCMhgmalzDTBsIokDj+rF7Q28Gph3DMaL8ApZFCRNL/7bZUzvheOuOkVcKvPx20YZi5oi54uwPddGStDhrTM3E0WQfI36TN4YZLQNY9NAAsiih6zpDmRGaPSEyenZMFFkpYipuDo7MGeWRtq2Yo9DgrGNYG+Whhs3M3N3CDLOl4g5BeXNR6VcYeWQUR6u9pMYfn5HExKQvNVCgsc4fX3m5YSgbtqRFXY2T4s+5befdDHvDODUH68Krac21YLZZDXsQ6Lmlr0SEptJNW6nWG94SpW51LaNbIhNGu0eD8Upr5bX5mc42Lty7im2Nz9Cd6KNGCHBZ3cUsbT6NkL2elkhj1WwuN5zDUE2QQHLKWIMFJrnRHIldiaqfw7fIi73ehjqqHZaBFRBsVibbXzNIX+cg/gEv8w90MtQ8Qv+HBvHX+iqOrU42ixovqi9+jcnodhwtyjXRRUmg6ZBFFCjIYGuVS8SpAmf52f+9LutvDfYxpa70wQyJXQm6f9mHoRvIbgld0Rl+aBTvIg/JXQaIAoIsEFpfT9sxSN4eC6Ydw3FA8QXcnxrkp7tuI+isJatlEQWR0Uy0KiWGaR6OxUwTBItnRpAEJK+IqQKY2OpttH2qZcxFXuyUbKJMBkssxibKBOx+IkqM3lQ/QWddxSiyPGIqNkARe4ysPUd9pBY9bRAwfAy5w4zaYzQNNFbcIRizzeqSrJKNUywY4kfEJ9i29xkSepKkliKSitLsbzpyfIFSOo26+hp8AS9RMU7QrF72Sh/MsCayklNPmcshs7tQdhHcAi1XN9K4PkT6YGZSpG7Var3+Rd5x6cLh6KgUJupTVarNrxFWsu601Rx69chn1RWDlJBin/1g1ajZ3mBH9khoUQ0jp1tOQgXdsEpsmFSMtN0dLto/3Wo1n1M6skfCe5qHTanNPNy6FWWWgtfjYX7NHPbph6yy3/Zj3xouv0Yrnd+JdDuOFvmoXo2pVr/PMBkMDTHQPEIg5qc5HiLboyBIAragjcTOBI2XhkoGFvLZgLUHFCa8JUqmO4N+eGQ4J6gW8SAUmvr1F9RWlXN9szDtGI4T8hfw/tghPDY3e6MHiOUS5HQVSRD5n1c30pceHBMRnt6wkAZXkJFsBF3NYAIhTz3rrltDcF8tlSix4cgNsrThNJY2nMZLw69w065bQRBocAbJ6FkCDh+fWvBRTm9YOCkjVWyAap0BnDk7cVcSf9pLwpvEjZPmhhA21VZxh6DcgBkZHUES0LMGsmFyKNvNY4u3IaUl2sItDNvDGJLBx+0f4bLOiwuvU1xuOLVhHllNKd0kb19VcLSVxgTPuGIRvn8+Ot3siWq940W7R0ulMNG2brXj9rl95O7NYZomYq2I8moKEPAscFclSHR3uGi9uomu/+5FGc6BBkjgmlk6blvpM5aXgQZqh9i2+Wnsmo0WVyOD5jCPJZ5khreZNq/l4H71/O9ofjrEKbM7q07nTNaZVju/fZ5BXm19g9psoGS/w8gZRJ+LHVWJrzgLcXW4MI00mxu2s2XBdrJ2BZfp5PwXV3Dua2fj7jzCKOBb7Cs5T/FdScJbIgzcO4IaVpF8oiWalIdAQSzLVmcDIPZcoqI07puJacdwnDE7MJOL2lfxk523oh8WeXfbXDzZ94x1w1SICK9fcg03776TpJrCa/NwzYIrre3YBZXfI9+sLn78hs5LMDG5/fV7ClH1NQuu5PLO9495frUbsdgA1e+rY61wPo/OfoIhcwRH1sGa3SsIDdVjq6083ljJgNWvrSswoEZaY6h+Ff+Iz6KHMOsZYJjEE0nSZ2bGGLD8z0tjpyEeZiEdSo/waNc27t33EB6bm48E13Pq7zrpdfUTb0/iH/bC76wadDGHUh4T1bgtbe6p13qPhVV3MsMAlY67WJNCi2sgCJimiZHUwSaiDOcY3R4pnM/y1xq8f5ihP4zgbHcge0rHbat91uLvZWhgxFJ2C1oOzZV1oZkaLtmFKIg4u+30KoPs3rYHBqB+bR3zvjGn5PUm60yrnd/hTJhHu7YRPzuBHJG4cO8q1ggrcc91WdTjR7lTUJ6FRBbE2NL8FCbQZISIyQm2LNpO51AHs7LtyDUy2qhWOHf5czS6JWL1hmokDqndxLyWvnZT9vCx5FuIEgiygOyX39QdoWqYdgzHgGoG9vSGhdS76qh11OCSnWT0LBElWrhhyiPC8loqULKPUP6eP95xMyPZCAIwmo3x4x03s7ThtElNWkx0IxYboFm5dmb9qJ2R3AiBWIDmaAhFytG4oXqa61vsA8EapcpnOvnpC5fHxW9fvJ+oGKOGAHFboiDuU+1GyB9vJBvDxEQzNRpcwYJxuLPrPpa0nMpfOnaQlRScMxxc8Pp5dAy3HdWNVezcplLrPRZVsckuQJZnK8UZmui0GsKmZpLpUTAUS1Wt++Y+Rh4eHWMY3R0unDMcIFlGUHJJUx79LHdoGS2DLMhktAxK2M1IMoITO3VGDWAy8sgo9RfUFTKHqTjTSud3f+wQ9+x7EI/NTUdzG8PuMNsan2F1wwrSv8wcU8+hYq/LkSOUCiKIAgF8DMhDROwxmnsVhL4ctrrSgKnYuWwO/JFNSx4nKyvWNNNLK1m5+z0IdgFTNREOO4ajGb89EZh2DEeB/bFD3L33D2zrfbri5FHIVU+NI4CBgVN2MKpECjeMYQaqKoDNDsyc0HC/NGw1q22ijF12kNMUhjPhgm5wcU223HFN9kbMG6DBB4ZoOFhHyBZEkMB0gKmbOBpsFc9LeUkH88hruTtcKPcrnP+X9/Jwy1YG7MO4VScXdVviPtW2mG9//R5GsxFiOWtOXzd13JKLBlfQYr1NHeTJjmdw6A5CWj0RIcZjndtY51lNTQXl+clMrxQ7xz7PIH11A2gxbVwDf6yqYkejJlbsxLSIhuSTMaOH9Z3z282qiZ7VxxjGPd/ax8gjo5YspwnqqIZ7pmvcJbxyVHJol3RcwOvRffTE+pFMkQv2nEdrsgnDadJr6ye8P8LC0+YzOzBzSs600vmVRRlV1wj6ahEFkYZAkO5EH2EhQm06cEw9h/Lst66+Bn+Nl0wgay3FCTEcqoPabMCa8DJNTLP0NfLO5VC2m4dP3QoKNMSDFnvvku3MGbTYeyW/hOSWrO9wgkXINwvTjmGKyJdxDsZ7EAWRdl9roYSTN7Dj3TDjRYSTMdyCYA0MWv/Halibh38sP85yB9PqbZpiVHtkRFGQJExJP5z6CmMeWW0yJNwZJV6XwD/qI/e7HOebK5gbncVQYoRA0s9s30yarz5yI5QrekWysYJUplt2kVCT9Kb6qXfVkdGz2Ow2FJ+1jYoONZJFGx2vS1Q8xjz3lC1oQ4tpVbdT3R0uHtK2WOdwz8Q9g6OlPSl/jck8vvgcFTsxZSBH13/3Itgh159DcIqYionoktBTesEwjj4VYeSRUcBE8ksYaQNTNWm8vGHK9AuVHNr+2CH2PneAxJ1JmmMhDKfJk+1Ps/m07ZiyiW+7j6vmX87ShtMm7UwrbobPfj+Pdm8b8/yWUBM5d27MkEGfZ5DXBvZO2vFW63VFHTHsGTvve/a9zJ0x2+pJyQJaRCtxPnnn8vJjr5E2MzQaDUgBkUDax7A3TGpGmoZZQWZ8rAVgTHkzr30+x9/xpjO7TjuGKSBvuHOGiiSKSIj0pwY5pbaTcDZSYmCr3TDjRYSTiaCW1FvN6nA2gm7qmCbUOvyYpnV842UGX1hy7ZSi2vIRRQEBe72Mb5F3zGMrTYY8Ij7BtpefQbErOHIOVgWXs9Z4H7PFDtrFGSgppcD7BJUVvWySTE5X8djcaIaKTbBhYtKb6sdr83BWwxJ2iK+geHLUmH5G1Shu0T1m6xaOcE8JkoAymDs8iy9U3E6tdA5v3n0noiCU6CcU483QEK6W8eRLdraAjK7oIAmYWQNBEsZQmKQPZCytZ4+IKArgFjFSFmnd0aDcoc0OzGT2BTPZ85SVlfTa+tl82nakgERrqLkk4JmKM624Ge4Ojnn+gs55DF4xVNLrevGyXTxw4PGqTn68vluBg4kj7+8f9ZF7NIdhGuNuYPsW+5hvzMGf8pHMpnD1Oom5kzgUO7WZAOk9mTHvA/DVp77DpoOPo5kasiBzSccF3Ljin4/q+zkaTDuGKaCwUCZ7GDSHMQQT09AYzoTx2b1jDGzFG+YYyhHpgxnqhmv4TOvV/M/Q3SRyKXTTwCbK/Ozl26pmBi7JSW+qn3A2MqUbsdKIYuvVTRXT3PKabH4CySE7aPO1MBwLlzCDGlkDe9BemG6qpui1/tMX8vPU7aTUNHbRRoO7Do/NzZLgQnaEX+H54Z3k9BwmkMqkCw3I3P05Bq8YKhj79MGMtZ0qWipqpmrSHxgiVhen0VOP/DuppNRS7qRzusqheA8/ePHn1LvqqmYPk436jwajT0XovqXfogGvogORL39ILgk9qSM6RSRn2c7FLJclGpQ1MJxiwYFIbumop3gqYd435lB/QR3h/RFM2aQ11Dwm4JmqM622GV7+/GIKFXudjQcOPF41E5/KNFnh/ZsocT6CJFCz0l/y2LwTF9Mi5898Lw+3PcGwP4xTd3LB7pU09AY5NKOXkYOjzK498rl++/rvuf/AIwgIeG1eMlqGTQcf5+KZ579pmcO0Y5gCQq56cnqOvtQguqGhHRZ3t0u2o+awL8ZkN5JPdXfyzQ3/QM/Sfn7xyh3YJXvBkZRnBjldpSvRi2Ea/HL3HVyz4Eq+t/Lrk74RJ7upWl6TjbTG0Gp1ZgSChfpvMTNoeS21khiR0qtwtevD2BbZuGf/g2iGRo0jwEXtlvBR8RJhNqOwdscq5kZnWSOLZqnRLNBgz7CT7c3xx/l/ZvOS7eTcOdyia0zDuthJuyQnXYleBEFkhreZjJ49bjrek8Xg/UN039KP0qcg2AVESai4VVv8fRk5oyB7Wvy91a2opX5tndVjSFlOwXOKi8F7ho+JGbQS6lbUsvC0+fi2+6oGPNWc6UQZdr6k1tQQYnZHKallnmLGZ/ewynXOuHK7RztNVpju+v0Qke0xRrdFiT2XKIg5FcqW9TbOO7Cctt2txP0JarN+mlNNbJv9NFuX/glz2MC73dLByI2o/HjfzeQMFRGRnJHDJbtIqkn2xQ+yimnH8LZA8cUJh+v7gCzJCIaI3+7lY/M/Qqu3qVDKmSqKa8YTbSQXR9O1s2rQDJ2Q4EeP6dTY/PTpgzhkO1fNv5ybd9/JoXgPgiDS4W/DLtkKF305wdp4mOymarFRcnlc3HPgoYrMoC2pxjEymZXEiPKp+Wc7PsZFM1eVCh/tKxU+OpTooS5Ww0xnW8WGYz6jMU2TyMIoW+ZuxxSgUW0gKsfHNKyLnXRvqh/DNOjwt+G1e3CbrjFSr8fynU+GwK7/d0MIEoh2AVM3yfQqGIf5nsrLF+XfV/pgZkwmkI/m0wespb/Be4ZPmN7G0fRfJoriB+8f4i9/2HlEAvQDiwsa6CUUM0qM+/c/QsDhL2tcS/QlB8dIqBY7jabI5KhOYs8lxtCsxF9KFCjz1VENW1CmJdZIm96MqZj0evrZsmg7thobzXWWQ/rvZ25DjWmAgOARMEWDjJpBRUMSLe3zNwvvOscwlc3U8ovz1NpODNOgM9CBLErYRBsH4l388tU7ERCOSh+2Us149qXVN5KLjV5NMoA9ZaNvdBB/xkvclcReZyPkqmd505mIgsAPXvw5M7zNBT6lrtEe9r94iKaOEzP5kDdKNQS4yj7WGJSrlxWMnl3A2e5A6VEKYkQtVzcWjrE8qhxTcrO7CVJXdau5OKMJE0Fx5WhIBBEMgRqjcsM676RfGn6FX+6+A7tkwzCNKUm9VsJUeX1Kvn9JQOlTMHImpkZJ436q71W3opa6FbUl+xAnSm+jWslnPH2HalF8+mCGO7bfxyOLtpK153Dm7Kzdfj6fWXw1O/RSipmcoRJR4pzXeg6vRfbSnehD0XMIwM9evg3DNEiqKfqSA7R4j9CzSE+LFj/SBN9RpXszczBL5Nk4gojV6zENcoMqco2MzSejZ3VSTRn0Bp32GZbWRo3hZyA1iCnDTGUGSi5L3J48XK42OKdnGae8MAdmHJevY0K8qxzDVGqJ5Rfn3ugBfn/gEQzDJKLEDk8jpUmpaWodAVq8TSX6sG2+lgkdUN4o6oqO5BErjhVCdbridqGF1TtW8kjbVoYDozhzdlbvWEnTihAErEZ1fnrHbboY6B9CjIhkHsiwx9g3oUGaKKqdyMlOpn5cfmPJPhmlr7QpXY5KEejfLN7AWfLicbea8xkNPQK1QzWoaHhMN1EhXlXnOO+Q8pNn40m9wljKk0rndKq8PsXfvz1ksbyaOsz+cvu4XP+Tfa+piAodi6xluXOv5rQmGsLY07Ofh1u3otpU3LqLrC3Lw61bOb9nBWZTKcUMpokAnBVawrULP8qOkVf4xSt3YJomWU1hKDOCZuokckmiSpwmT4iPBNdj+7FtUt9RpXMnyNbWvGOGg9yAainnGSahdcEC4aLH4+H3Bx4pBDcjmVFcOReCLNDt6iMnqgi6RXa4Mnw2n3vlY/TvfvO0Gd41jqHc0Pelxhf5Lr44s5pCTElgmtDobmAkO8rBeDchVxCvzUOLt6lEH/Z//+nfcEoO7JJ9XAeUG86hDOUOk5IBIkguaUykVo0SQbSLnLf3bBaYc4nYY9TmAtTvqys8P7+Ffc++B4kkozgjDtZ2vY9ZgZkTGqSJotrJOtmJmrFjaDTKmtLF31+xQarodDrHsm6Ww93h4vSOhVy998Pc/vo9DKjDkyptTCT1evfeP/DMwAsTno+j4fWp9v1PJAAz2feaDFUIHD3lRyWM57RCteMPYUS9MUZdURRJwRRNBEPAITqIemMVKWYaXMECLcxQxhqBTqlpsnoWE4tfrN5Vh12y84Ul17KkfwH70gcn9R1VOne1KwNEtsfAsHRM1PARkr1qGbVLdrK++yIitij3dD6IgYFDs+PQHbzif50HFjzK0tdPO+rFzaniXeMYig39YOqwEdctke9XR9/g9NDCkiiouPloNYFU7JKNVm8jzZ4QB+PdvLf5LF4aeaVEH9Ym2kioSZJqmkXBU8ZtVB5IdvGy/zVqJB8tSjNm1kBP6tbSURkqNYHzpHChvnpa65qtaMUtFCK9jXs38fAbW8kpOSRdZFnP6VyYXVXxYi/ODoBxI81joX8ox0RGabxlwkpO53gwdxajUoRcLvUqixLbep/GLtknPB9TlfwsHMN7RvB3+sb0Z8bDVN5roiGD8b5zmDhTKsd4Tmt2x/g9CaFeIOvMYmgmDtWGIqtknVmEeqEqxUz+uYqWI6mm0AwNAxMBAd3QqXXUkNYyOGT7lL+jcm6k0S0R9JSOntTJjarYAjYa1o0Vtiq/Bj2yi8cee5JAxocv5yMtpRnxRoiKKne03MfDwa0k9QxX8sFJneNjwbvGMeQNfV/qsBE3NByig7Sa4Se7biXorKPWGSgxOvmLM6JEkQSRgN2HU3ayN3qApJriT/3PY2Ki6EpBH7bWHmBEGbW2SQ2t6hLZxr2b+NXwXcRWxXAodta8ci7n7T0b0Ski2sWKn6Hc6FUyqnkmx4Ppbm598bfkRlUaMkFijgTPNr3I8t1njBGSL88OAst8hZvWUAwEEdTokeWdY6F/qIRqRik/XZJvoOeXCW/ebfV0JksOWA0TZTPVIuRyo7Wq9Rw2d2+n2VM74fmYbHQ+7jF0TC5Kn+p7jedUq33nk8mUKjnXSsZ3oH6YhC1Fe6x1XMftkO14nR6yqoJm07GLNpw2Bw7ZMtwTPVcURDTT2tkwsUpNSS2Fz2aNnLubpnbe8ufuYLqbF17cSa0rwMz5baT2pNFiGrpdJ7wlir3ePqZ0W3INXgpndC4m+HKQnJgjmo6jGqqle2GKxNwJ/mfobs6OLT3h03DvGseQN/Q/e/k2FF3BIToIuesZSo+gmwZ1rhoM0yiJglq9TQXt3JeGX+HRrm3sjx2pMc+psWrMyVwSp+TAZXNSYw8wnB0FwCbKDMfCODRHycJVPvoSZGjMNRC1xdl6xlMsMObSplemh4DKNf9KTI5DfwjzSmgP8YUJQko9slMmkPMx5BwhLEVLxkWhNDtQ+hVGHhtFkAQy+zMWkZxqIogCiV0JapYFjpn+oRLKjVJhmVBXEQURSZToTw0QdNbRlxrkP3b8vEAnfiy0ztUwXoRcmdvqhUmfj8mOAB+PzGyq8qLVUJmSYuJMqZpzLXdaeVGo3EEVT29pVliM9MEMzh4njXIDOac1ypnRMjglZ0WKmXIomrWUZhctWhfFsHZgMM2SrGSq523j3k386uW7iJ0Swy24uLjvfJalTwcsJzhZ/eYFC+bxcftH+NnLt5VQcudEFYDB1MhRB2BTwbvGMYAVSRREvgUJl+yix+jHLtlwSQ6csrNqFPTZRR/jPGk5W7ue4i71ATprZhX0FjTDYF3H6hLjIABDIyMlC1ddq3vxL/LSZeu1oq9AC2qzCv0w6Bwm5k/wngsrRybj1fzLmRwdrQ5qIn4cWTtxd5Ia3U/Ck8SZc3LqxZ3Mae2oyNBZrFUreUS0hA4CiDYBuUYivCVKcHVwwlS/GEfbrMxHqEGhhhFzFEM10NHpVQeQRJFWT+k+Qf45ipbDIduPefN4oqyo3PBMdRxzMiWv45WZTba8Nh4qNfwnypQmcmx547unZz/bhp5Gdsg0OhuqOsDie2BV5zlsWbKdpCc1JfoRh2zHa/OQ0bIYGLgFFzbJxucWfWIME/Fkz1tJoJdtIGqP81DjFlo9TbSmmpCcIqJTnPSkV95Ofe2Z7zKYHi75W1JNoWi5CY/pWPGucgxQKvJdXiIaLwpq39XC0ENhFJeCOE/kNWUvKVu6oLfgt/lKFseyvQov3LSzwBGfiWTo+UUftqBMqjmNfZWNsC1CsLGWqDNOQAtwxvmLaVwwdhJnMtMlY1W+ZrD65ZVsXfonBl0jOHN2Lu49n4WXnVJxIqVcq9Y0AR1Ll0AQkDwSevoI387xYHItR7ETCbnqsadsREbj1OoBBn3D6KKBYEJDogGbKONuLC1n5GfSvTYPTZ7QlLOJ8vefSlZU7XwcyxTPicjMjgVTzZQm49jcHS6yziyZkSxtzuqPK78H3td7DvOis/H8L0sHerxzW/69NnlCZLXskWxDdnJ6w8JJn4fyzL3wOYsDPccwMW+CdqkV0SlW7FOMd22smnEO57Us5669D5S+uQCjSmTSx3q0OCbH8N3vfpevfvWrfOELX+CHP/zhcTqkE4PiL6H4As+XiMaLgrpGe/jtK7/n5UWvk7XnyJAhpicObz3bCTh8PNq9jYtmriosjkX3xMj0zi3U6dW4hmmYyDU2WlKNrN6xkm3Ln6Zb7cPjcHPV4rEz/nlMZrpkrEiOwZrYSk57aT6j9mhhCagSWV3d6lr67xpEVwxEh4ijwYYyZKWuiIAASo+Cs9VZcmGPV6Ofahmk3Il8JLie1TtW8nDrVjRBoyFWz+yBdvY3dyHpEumeDDlnHFmW2HrgT+iaTlZXMDHJaFmy2tim/3g3YiUnNtUsoPx8HOsUz9Eshp1oVMqUbt59J3si+/HZSxu9k3Vsk3lcpXugqbeBOWoHNYGxLLpQfXAhf06TaqpwrdXtqSHdUFkOtRiVMvfQqqLjLwr05qzsgMcFUq+nsQXkkr2cyVwbZ4WWcN/+hxERkEUZ3dDR0RlKhyf9fR0tjtoxPPfcc/zsZz9j8eLFx/N4TgiqfQmzAzNZ3nRmQRGsWhQk6BIvBF/GLttoVOrpdwwhkKBdbiVYV4dTdoyJcIoNtSCCqVr6wQCIAu/deRZnL1pKbpE6YcRTTaGq2EhXajS2f7qVhYtPGVMnLb64jZxhZQcm1h6BW0L0yJhaDsFuyYyaukXjXLsyMO6NU86MOt5GaWJXAhDwLfIyUDs0xonc2XUffztwJZ+PXUP/yBCBqJ/mWIinljzH5lP/yJBrhIAWYFluCU+MPo0z50TzazhkO6qsHaYRSBW+k3KahGsWXFkIDhQtV9GJfW/l14+aFO94TW69GcR8xwPl7L4wecc2mcdNdVKo2uBC+fcqPS1i+7GNfenqcqj569rIGRUz93mL55Qev8PNev0C3I+5UEfUw/fWkdeb7LURdNYSsPtI5JLk9ByaqSMKIvcfeIR6V90J6a3lcVSOIZlMctVVV/Hzn/+cG2644Xgf03FF8ZcQdNYynAlz8+47S76EierF54XO5rHRbfhSXkS7SE3ax4g9jCppOGVHxQin2FCrUc3ibJchezCDnrVEVKRbJGpa/HiucMGllY//zmfu486u+0idlcQWtxUUqipNSVRrmJXTI+QvbrlOJrU7DZh4TvWQNUzUmJXZCKKArdaGvdleInZeDZWYUT32I1FgvgmvP6jz8sbXUMNWRmKvtzHyyVFSUqkT6Ur3EAskOHVgLjUDAcysCQKsfO09dPbNJDUzy6LTTqHvzkGeWfQCOWcOwRTIaIqlf61lCt9JJZqE7/7lx4TcQTRDx8QkmUuxIDhvjBNb3nRmoV5eTTypEo7n5NZE01NvFfL3ll2yM9fTWNHAFW+PC4K1dFmOgl73rLnE6xIVz3G1CStgDN1HtcGFU2rnFliQlzedSVMkxJ6N+8Yt0RZf16YJelLDc4rH0jR3iih9ColdCTasL2VezX4jSzaSRbBbolVaXKf3tgF8i33jBk3lTXvd0NFNw2qYSzZm+kqpbU7UdXFUjuG6667jkksu4YILLpjQMSiKgqIohZ/j8fjRvOVRI3+DyqLMoXiPNb9sGty99w/845nXVXxOxVpq//PE9ST+jBfFpVIv1+Fw2seNhIoN9eiTowxsHClE3+OJqOTx3L0v8que32IYJkGtjlQwzbYVz7DutNUVexFwpGG2P3aIoYGxUWZxSq7HNWtcTxDIDeUwMgamYYIsULPCT65fRRu1BMrHG9erxuX0kevXc1f4AQ72dyNHJNbsPhfzeZMcOUSHCJgWpfcDOq5LnSVZmtftZf7qOUgbJWSPjKZZDgsd2sQZzHx/Kw6vA2PE4P3hNTzUuAWH6SAtZHAJ1iBB/ju5d9+DJTQJWV0hlovjsbnorJlFX2qgIi1C/rs/mpLQ260/cCIwWef36KFtBRJEPz42+C5hw1xrMqk8oJh1RVtJxF5Ne8LeYCexM8GefxlLW5E/rjw9vWkaaBgMZ8OFkVSYuERbfl0rAwp6ykDpV0C0yqumAb2/HgCTApVNtDvGnuQ+EAREuwCI6Bkd7bAmht/jw5FzMBwL0xAIjrk2igWqVMO6Rw0M6p11tHqbMEzjmMbDJ4MpO4bf/OY3vPDCCzz33HOTevyNN97It771rSkf2PFCyFWPLEp0JXqRD0cPpgHbep/mw50fqHpi87/PO4dPnP1X3LbzbqK5OB67m08v/ptJpfjFkw3hJ2L01w0ynBqhVquhaTg0RkQlj/TBDK9v2UfmVIUmswFM8ITdRD2xiiI0xRjPkBWn5P11Q/S3DBJI+mgZbQQTRIeI7JXI9au0f661IjtnOardYOuk81kyawEvPLwT77CHUFc9hmYt7xmCgeSSMHSD0FADG3yXcG/uoRJHu6xzKemlR9J4dVSjWDJ09+49vNr6BnN6Z/LF1N8ylA2jihqzP9leUp4zzVKaBMM0MDHx2jyIgkiLp4lINoZm6mMc/dGWhN6O/YHjjck4v5/u+jU/2WXpn8uGRDat8D/dG6m7q4bFy08hvCVaNWIvdhqCJFC7MkDjZSFqlgUmtTk9mozgN70M66MggF0sZUGeqDxVfl07mhxoMR09a6COqAiigKvdjmAT6P/dELagDdEuYuQMZK+MFtMwcgAGAgKyRyKxK0FuS47zat7DY51/5EDtIQL1gZLr7YnePzGcCZPMpRAE8MhuEmqSoUyYFk8TGT17woOMKTmG7u5uvvCFL/DYY4/hdDon9ZyvfvWr/P3f/33h53g8Tltb29SO8ihQ3Ghc1XoOv3r1LnQBZCTafa1ohjaux61kXP/fmm+OEd7pTw3SlxwExmfZtDfY2X7qszzcupUMWZyqnQt2n8ua+LkVa6W54RyBmA83TmI2S0A8asZxmq5xL4gx1B/JUuqPfEp+x/b7eLh1K9n5WRw5B6tfWsF5B5bjanUw1BZmMDYMpsDpyypPaxTXXdVRFUESKt5gLcONpA7OQYtqmMV7eyroqo4gg+yR2DD3ElbULhvjaKuNDG7cu4nbD9xTKgIfsUpsjaeXZlNjaBJME5tow8BK0cPZCM2exsLOSvH7H0tJ6GTpDxwtJnJ+eU1m3TRwiy5ySo6kLYUoiow6ogw+EMbUTFwdzjEROxzZrxFkyPYo9N05SGR7jNarm3A0O8bdnG6Pt/B45o9ogo6IyBn2RXx71VdKvoOJFgArOQ5HyE5wTS39dw3haLEje2RMwyT1Wor93+tCEEByS3hP86Cl9aKSqUz9uiDhLRFykRzn7DuTma+1EfXFmH3mTM4MLebOZ+7jrvADRJUYo9kIumHgd/hQDRW7ZMcwDHqS/QU9kBN5PU3JMTz//PMMDQ1xxhlnFH6n6zpPPvkkP/7xj1EUBUmSSp7jcDhwOBzH52gniXKjflH7Kjr8M8gZKg3OIBk9i0OyVzWwxca1RWpkJDXKbTvvpv49dYXtyo17N/GjHTcznAkjYPGxXL/kmqqqUNTCliXbYRRCiSAxe4LHT9vOwtdOKZkWysPeYKfNaOWiA6t4dNY2BuRhXLqDK9s/OO4FUUL9UcTf9G9/+Q9GsqNs6LyE1KoM27SnsWs2WlzW59sq/omF4jwOdHTzYHAzCTmFu8/Fhl2X8NlFHyt5j3wkpwxaqbXklRBsAkIGjIwx5gYTZBFDNREcghW6m0UvJgjUrwta3E5MrpZe/P0Ui8BXK7FVoklYHFwwRmq1EmfWsZaE3q79geOF8ZzfUGYEVdewSzZUXUXSRbI2BUkVqU360WIaklusGFDko3W5TibdrYBklTwN1WoAN17egGmC0q/gaHYUghNlIMeLW3fx2she6sUgbsFJ2swykgyT7VUolwEvL09Bac+ikuPwLfYR3hyxSq8uE6Xfug/kGnA0WceSfiND5z93oI6q5IcscsM5Bu8dIjeoggHNSojm0RAcgs3P/pFbV/8WuUZidttMcrrKUGaEZC6FQ7ZT76zFY3Nz7cKPVlUQPJ6YkmNYs2YNu3btKvndJz/5SU455RT+6Z/+aYxTeCtQKfV/tGsb6zsuKujD5p1FXqij/CTnjWsoFSQzkMWpOzhY0823n/g+steGLErElDixXBK7ZAPTZOSw9GM1VajlTWeQ86i0+BtBBY/upk8fxLPaPSbChSPRzPm/W0HnYAexQIL5q+ewbPnScT9/gfqjiL/JITmQRalQAhnKjJARs7QFrSi40d3AQa2bfQ2HeKh2K3Fbgow7SzQX5yc7b2U0E2V1+wprBjxi3Sh6VsfIGoCJntFxBOwICLR+rBnfIm9J47txfZDuX/RjZCzhecEGiAL2Rhvo4K8gFVr8fZYbnfIoPi8CP16J7WikVvPXxju9JHSsqOb8Qq56ap0BTAxi2QRZm4JgCLx3x1nU7AigCRqGZqWRlQIKyS2hjqgYmiUmJEgCtqCNzIEMXTf1YGQNDM1Ei2kgCmCYdP13D7ub3iCzOEuTHkJExIeXAWmIvqEBFiwYOxKez0qrLZFWGugodhimCZJXwtFUmr2IdpHG9ZZGxIFMF36Pz9JbGEuFxihRMmRo6AqioHBKWyfqiJUpyIJcoOv50Jz3j33yCcCUHIPP5+O0004r+Z3H4yEYDI75/VuFaqn/6aGFBbGXl4as3YV79z1UsZkYctXjMpwMjYbxm17C/ggZOYsv5qEl0EiP1s9oNoooithFJwgCupopjEfCWFWobb1PI4sSUTFOMFBLNBvHh5f2Wa1VP0v+ouwYbps0nUEJ9cdh/qYZnmYaPQ2FEkilKNhf66NpQyPKAYWsoSCIAh7RTVxJcPuejTze8yQ1jgAfsq9jQXouosfq1eQF5yWniJ4ycDSNPc62a2ZgIjB4zxC5YRVBAscMB6JNtKLAnFFRUrJar6RaFO8f9RHttl5noHZojNEvN2CTjebf6SWhE4VipyogImjwnueWcsmzF1j7MTKgmQiSwIxPtowJKJqvCNF728BhQ2pin2EnfSCDHj2sTW31ddHSOlj6NohZA7/Di0OxE7HFqBUCRIQYLsNJS6ip4nHC2AEKZUCh+5Z+bEEbdStqx50ANHIGXTf1Vsx8iq9hh2LnvR1nsfLl94x5/5pUAKfmIOZOIg6KxP1JWr3NFcubbwbecZvP+WbzoURPoWyUT/3zJ/bR7m3jNhNnB2aywXcJ/2NsZNg3iomB23ARTNWBCg3uIL2pAQzTJGeoYJqYgNfmKSiMpdQ0QWctMSWOS3KSVjOsaVvJMwMvTCnynAqdQT4CXtpwGl876wsW9Yco0ehpoC81YBH+abmqUfDShtO4eeBOckkVj+gmd3hhzMSk1lGDgcHGxCbq6q+iabTB2s/IGiAJ6Flj3Lny9mtaqV9dV5BBNDVrw9o910XXTb2FKK1udS3+RV76PIPcfqB607f8+NenLyD3nRz70geP8O541KNaLKuUSbzTS0InCsVO1bnXiXK7giqrSC4JQQIja2LkKgcU5dKZuZHcEaeQhwFmxgQJJJ8EmknTYAMXvH4emxduZ0AewmU4+aumSytmC3kUl66yh7Io4RxosP97XahhtaJuSfG9qYbVkpJT3epa9vTs57ahuxEcAqFUkMHhYTaftp05vR00R0pfrzkaYs0rK9m8cDtDHmtH56rFlcubbwaO2TE88cQTx+Ewjh9eHH6ZtJphOBOmLzlIgyvIZ1qvLmw2TmaGGGDD3Euou6uGUUcU029wR9O9xF1Ji2dFz9LoqidnaESVWKHHUELvq+fYFX4trxVC0FnLhzs/wIc7P3BCIs9K0fVnTrua21+/h1fCrxeoIv5jxy8KvYZKUfDlc97PT3beSkpNIyIiChbXvUt2Wot8ah/iWgHpXgnxcJYguSQkx+QYKGd9YSaNl2VKIq18lJbZf4Q25PWOfcTPTtDR3Fbxeyo+fv+oj9x3cpimycicUR5p24o5Ci3+RqLEpzTzfTw1B6ZhIe9U09kMrwXeQE/qmLqBqVs3h+yVqwYU+WvG2TbIof/qLu1RFfeqdDAzBoJLxEgbrB5ZyYVLziMsRGgJNY3rFMDq6Rk5g+SuFKZqvbDgEBAkJkV+V5xBJHYlCG+JsPOZ3URPj9FiayQ3quLXfAx5Roh6YmMcAyKct3c58zOdJBtSnLF6cVUmhDcD76iMId9fqHEGaPY0MpwJI6YEMndmuT/zCEHqCK0LlixeVWsmujtcnPUBSxGsO9bLabb57Op8zeoLiG6uX3INSxtOY8fIKwXD75Dt7I8dAg6PR0Lh4s3/PFHkWTy3XakcMt7nLizxZa0lvh+t+rcCaWBeZS58uBciCgJL6heO0X7+7KKPgQn37H+QjJYlpsQJOHwli3zzz++k6czQuILz4yEfaRUT+JXThtRmA8gRiWF35Vnv4nMZ7Y4VhFUi9hhZe46GmJXdBQOTnyI6njoT0xgLd4eL1qub6PrvXnIjKmBiC9povbpp3GsnrwmuxfVSZ1AMCQzFBF1HlEUa1wdpO39qOphaSi84BQBTM5FrZLRRbcw4ebXPBxRGcBsDDTgUO0PpUfxZD3F3EkfOQU2qtAPuW+JBjeqYmkGH3EbzhaGqe0pvFt5RjqG8v+BQ7exKvcatc3+LKAk4NQdrnz6fj1xuLV5NVNLxLfbxYG4z9ygPodgVbA4bq1rPKdl/yDea/2PHL0oazXbJzqLgKaiGhk2UCxuX4xmY4ubXVMohpUt83Wimjm5YS3zntS5HQKDF24QoiOR0lQPxbv7v8/9F0FnLNQuuHPO6n138sdJ+TPe2secqwKQdQTVUow2RnCIznW1cuHcV2xqfmfB7Kn6dWmcAZ85eyO6mMkV0vHUmpjEW+cg6sStJ8U5KNRQ0wceZaxHcgrUZfxg1K/y0XTM1p5DYlbCGIxyA5bNAh1x/zmIAGEdIqRjFuw/NsUZWh1ey+dQ/MlIziiPnYM2OlaXZggRNlzcWppbyAeFUtuxPBN4xjmF/7BB9yUFkUSpkA91DvWTkDN60m2C0loQ/xcOtW/m/2r/wvZVfH3dNf/D+If7yh538ZuH9IEJjXQNJd5pnBl7gw50fKHnfao3mjJ6d9Ihj+mCG3tsGMFSDoRnhKZVDSpb4RAkJEVOArQf+xMLYKYiayKF4D16bhwPxQ2iGTjKXIqVm+PGOmyu+bj4aX950ZsFJHO8LtRJtiOSXCmyUa4SVLOo8lR5nH3P8HVXrrcWvU7+vjrXC+WxZsr2Q3U12iujdsK38dsBU+mZ5Qyu6RdArPEDCmnSTLa4vW72NXL9K+uDEhHh5pA9mSB/IWgueNgFTwnI0JiBQsURaTQ+9PNhZ+cZ7mDvUQbIlRSqWQc7I9NcO0RwPIUgC9npboenu7nBZpcztb30p8x3hGIrrwjk9R0yJM5IcRc/puHARTNZaY2tRDyPBCFFvjH3DB6vWkvNRymhtFMWZoyFRhzqgUeP306cPlkSQQ5kRkukkzWIjpmkSdFlR5lQbzYO/HyLdlUEQBPptg2ROzRJK1U+qHDI7MJNVrefw61fvQjcsDdtWvYHMUIbNL/6RWFuMmDuBIRhopo5dsOGxe8hpCsOZMC8NvzIl1tDjicZLQ4Q7o4wMjeLa78S+xV5o4L142S4e6H+88B3leyPVXidf4z21YR6X1a6dsjObHk09MTgW6vG8oc2FK2sQiA7RyjTtIo4WB/aQfVzdg/JjyWfpasxaRDMUE0EWEGQB2Scx56sdY7S1u27uZeiBEUzNxBaQx2ijlAc7M8wWnml5gU0LN5MVrPH3C944jzXDK0vKaG+nUuZJ7xjKT+be6AGiuThePMiGDbsIcXcSf9pLzJ3E63Ij1AvcvqP6F5CPUkLNQVyGg4QniS/hZSQzisdTGkFKT4uIAyL9xhA1mp9UMIXgFGhXWzl7yRmTGjVLH8ww9GAYNEtq0B/xYU/biTkSVcsh5RHLhzs/wLbep8npKkGhhkh3HFEU2THjFbyql/rRIEOhYcJaFFmSC/QQJpWZMaeKahHURMhvMKfUNJ5GNx+5fj3rpPPp8wzywIHHp3STFEeik12WK8f0aOrxxbE28/OGtvvmviONOjsIhgAitH68meEHwwgS2EP2cVlX88cSycawSTLr6y7k3N+djWmauDpcmIbF3SV7ZWwBmdarm8Y4he6be+j5Rb9FNGmzdifKm9Pljeidz7zGptrNIEJ7XStxV5I/tj7LJaddUNJLeDuVMiuLC58EyLNdvjT8SmE0NKspxJQEpgnNrkY8hgsEE92lM1I7imwT+ZsFH8Yh20mmk/izXsyM1bBNppPsf/EQ6YOZQpQS6qtn3eBq0GDYG0aSpZIIMn0wg22jjbVd70OyiQzahxlNxoiNxvnpC7/me9tuojc5MOGXmtiVQItph0VxoDkSYs2OlchuiT59EAGh5H0H7x9iz7/sY9+NB9nzL/sYvH+I2YGZXLPgSnx2L+FsFAw4K7EYQzSoNQO4ck7abTOQkNAMjYyeRTN0GlzBiqW0qaDS8Uz2Oyx26iYmd4UfYHRelHhdovC95m+SlJou7ImcSORLaNNO4dhQ6fu9/fV7CgMak0XjpSEW/HA+Nef4rQ17w1p2a7g4yMy/nUHbJ5uRnBJKr4IgCBVLP9t6nuZnL9/GYGqYaC5Gb3KA/z70P2xs3YStzoYgCrhnu7HX22j9WBOn/H9zK9JvDz4QxjRMRLeIIIAa11BjWoHGIw93h4uaZQHarpmB53+5MRoNWjoacTW7aAgEUezKmIXM4lJmnqrlrSplnpQZQ3EUIosSOT1HOBtBRCRnqNglGx6PC7fPSTaZ5ZJXL6BRrC9sDj9374slUX7Cn0TMimQeyLDH2EfzFaFCOrhi5zLm1HcgrhWYv6azIlPphdlVnPLaXF5O7GbToi04bHb8io+4nuS2nXdPIhW0QiHBBqJTwlAMzn3jbNZ8eCX6cqMkch2PPCwf7XYd6CW1JQ2mya6lrxERYtRIfhQpR7MnhCxKKLo6RlhlKpiIo36i8T4YP0Karvef/DieEbC7w8VpPz6V0acipA9kcM9yFaL5ifSZN+7dxM9evo3uRC+GaWKTZDw2N6lciidnPsPyF88o6JzYAjbqVo5daAPY07Ofl2tfw696aI43gixgpg0EWRi3Od0+qxVfr5cocYJm9Wu5UilzPIaGE4mTzjFUqsOl1Aw5PUdGyyIJIj7Bi9FtMpIaxWk6mRufxennn0b7h1rpv2cQ8+cmF3aex+On/pFBeRhHzM7a3vOZFZiJOqrSe9sArR9rKrCLntowr+KFUtxoahJD9Kb6MSSTgO5Hskv4M16iufiEN4JvkRd7vY0uvZeIJ0ZtKkC71MrC00/B3VT6vhNRBc8OzGT26TMZ/IBVO73g9fN4rHMbo41R/A4f18+/poI849QmIMbjqK+kLlcN4xn/6Xr/yY8T4dzrVtSOKe9A9Yb2/tghbt59Jzk9hyhI6GYOVdfICQp2yQ4+iDhjNPWGxlByFGPj3k3cNnQ30bNiOBQ7a3atZMXu9zBQO4R8qY262hpmc+y0KiXqkhMwNJxInHSOoVIUklYzXLvwozR7GnnioT+xNfYUPfRZ46l976PNaGF0S4Tk7iSRP8YwVZOzh85gzmAHiaYUrtcddIY6EOoETNUg061w6Cc92IN2mq+waH7Lm1b5iLludS2jWyKoUZVaLYDLcBB3JvClvMRdSTz2iW8Ed4eLlz/+Gr/tu5+MmLU2NVsuZSGnjKGKqEYVXE4rUUynsc6zeowASiU+p8lceONx1OfJzMbbgC7GRDfMdL3/5MbbwbnfvfcPHIz3IImW/ocJmBjopkHQGaDOWcsZn1tMS6qxan+s4FwElXp/HYlIis2LtxOvT/HswhdQvDm82zwlo9/l9mIq1/JkGRpOJE46x1AtCllSvxDbRonzfrGcOd4Ooq4YtUqA1nQz4jwRpVch9UYaBGt9HgPq99XRkmxES+jofgMhpZHtySGI4GhxYGStMsmW0J+4K/xAwYCuT1/A0t8vKtA4BFfX4Fvko3FXiK7X+ni4dSvD7jC+gJerF3+45IusNKGxP3aIB9yP42hz0Gg2EBXi3JN5iNB3GmgaaSjhoq/E+JinlVBjGoIsEFpfT/s1rYUoqqacUpKjn4CoxlFv6lSkLp4IE90w01QUJzfeSue+P3aIbb1PWypuiEiShGFa2gg1DsspXDV/rNZ6/h5VtBwO2c79+x/lULwHURAJSxFqG2tIqykeCW0la+YQsjCajRVGv18cfrmqlPBkP/9b3Yg+6RxDtSikKRJi9wN7MA2TVqXJorMFDJslqgFgmiaSR8LMGYe3JAET6tfWkX4jg9KXwzRMnO0OZK+M6TY5EDvEnV33IcjQZDYwmo3y2+H7CbnqmVlr1SXDW6IEVwdpv6aVzxy8mvN7VhD1xsZoOVeL0AsXQcC6COpSNRwY6CIsjBLK1ZEb0cjcmS1w0Vck8Iqo1uawatLziz4EzHGXfI72wqvGUT9ZUR8YO8E0bfzf2Xgzv9/iwGsoM4Jm6LT7WulPDaIZGpIgcenstVw6+6KKjip/j/anBkmpaVyyk5SWRjcNbJINRVPoVnsRBQHdNLALNrwOb2H0+7GuJ3mk64ljjvTf6h7bSecYoHIUEn0uhqkZiDbBWnhxHN6GPLwU4z3NY5G3ZQ0Ep4ig6SAKtP1tK82XNzL6VITY8wnCm0cRbeLh0TWVWF2CZCZF/UgtWV3BaTqIuxPEG5IIqbE1dXeHi9M7xk757I8d4radd6NrOi2u0qW18otgJDOKPWvH0+NGGVDprx0i6o/R6GhA/J1YaOzmaSXUmIYa1xAEENwWV8zgA2GCq4NVaTUmuvCqjZ9W46ivVPethGrUxtOYxrGikg6Lx+bGxOSU2k6GM2EAljUuqegU8ll0Vs+SPUwgmdYyGKaBJErohoZqWlKbfrufiGKp/um6Vhj9jquJ4xLpv9VluJPSMcDYKMTeYMcWsGEaJnpcBx0Em4B3oRs9bpDZl0V0WkbTTBkIkkj92jqaL28sbabqJmpUK/DDt81pxhaRicpxam0BYmYCh2LH2+vG9Jvj1tSLjevrz+8lOhijIRkkI2XxNrkZ8oQLwuTFF4HDtLNm50qaRkL88dQ/s3nxdhS7gltyceHeVXQMtxU0aZUBxWJ3VU0EtwiaRSthagYb39jEvbmHKvYQxrvwJjLeE02BVEP6YIa//GEno7VRQs1BQn31k55gmsY0xkM1HZaL2lYVdFg0Q8MEfrrrtoo9tXwW7bV7MEwDl+wio2WQBAnDNKl31TOQGcIm2GjztpJWM+QMlYyeRRQkGlxB3hNaytP9z9OXHCg8/2gj/beyDHfSOoZylGwciiqCLFK7MkDixSS6oiPIIrJPQvDLBNfUETjTR92K2orjn2bOpPVjTSjDKrl7VM6vfS9blmxnwDWMW3Cy+tWVNMebUBLVa+rlerU5WcX5XgcJXxJfyktkNIbL5SxcMGPoifsUuoO9bF68HROT+liQZE2Sxzq3sc6zGuV+xSIWS+kYikUuZqStjEnySww1h9mY2ITgEKqmtJUuvGJqDlu9DSNjVDTeU6E1yGPjG5v4n4UbUZw5XIaDdZ7VrNi5bFITTNOYxniYSIflpeFX+OXuO7BL9kKGXH4/5LPojJZBFEQyWgZZsEZb47kEOUMriObUOgPUu+oYyY7it/modQa4ZsGVrJpxDg8f2sqmg4+jmdbjL+m44KiN+ltVZj3pHMN46/XlkWxuOMfgfUMYmSO8J6JLoPacADXLrIZsbjiHGlORa2wYWaNQGgKBgY1DqKMqK/vfw5y+DmKBGI2+Btr0Vtq/XL2mXu5sst1Z6rvruHj2+Twy4wmGfaM4snY2+C4p+QzF9MR7QvtIezPk3DmCo3WIgkhA8RNpiTGUGyH13ynUUSutxQBkE9lnQ7QJ2AI2xLWCpdLmnFpKO/j7ITLdWRBBHdVwNNkwNI7ZeO+PHWJjYhOI0JCoI+FJ8mBwM3PqOzi14a2jF57GOwMTjT7n+w3Nntqq90NxFu2UHKQMHZfsJOSu52/mb+D00MISUsk6Zy1/PfeDnB5aWKIK+Hp0HzN8zbhkF7FsjOeHd7Kt5+m3TFvhaHBSOYbJjFcWR7KxF+IY6cOsiyJggJE2yXRnC44j/GQENayRG1YL0ba91k5uREENq9ampQTNsRDNkRD2FjvNnx6/pj5m1yBoQxnIsfKV93Bqbi5D2TB1Sg3nX7ai4vML6lV/GMCpO0kGUtR76kjVpfE7fHgOusiNqAg2GPAPMSpHqUkGaKeV+gvqaLwshL/Wh2f72BtF0XKFvYXy6YmPBNczf/tsBBGQBEzTINuTw9nqmDS7ZDXk5URDdUHUAQ1fwsuwN4y4VpjOFqZxzJioJj/ZZm5xFp2fSioOQicilSzXXA9no2M0108GnDSOobgx5LV7yGiZCbv92Z7skR8OZwwAsWdjDN4zjBpTUcMaokPA0MBQTcyITvOGGuz1jsITRYeEaeqgQvOGiZul5ZM7RtZguGOUeE2C2kE/i4xTaL4iZDWGB6pnP+cvXsHoG1E2JjYRFiOFi7391Vb2cpA/dj7L46f+EcWu4Mg5uOjQKi587jwaLwtVvFHm18wp0IPLokRKzVDrDBRKTXd23cffuq6kZUYTuQEVUwfTMKldGThm452/MZM1aWr8fkYyo9TIAeav6Tym153GNPIYryY/lWbuROWb8f4+Gc31k2EC76RxDEOZEfpTg2R1BcM0EAURp+SoWhoZvH+I8JbIkV/IIGBxEcV3JRFtIqJbxBw2MTRwz3KBaaJGdXyLfNgb7NjrbaijGnpGBwPkgIRnnrvwkpOd3Hmy889sXvhHspKCS3Dy0TmXs6/+UEV63ZJSWcdMru74MCtiy0r7AHKG4Y5RNi/4I6ZpUh+rI+5JsfnU7Zzy7JxCc3qdvJo5gQ6i3hhCvcB/7PhFoTl3KN7DcCZMi6exkFp3pXuIBRLM0Ftwz3WhhlVEm0jjZcc+NVR8Y/bpg3g805vM0zj+GM9ovxnN3Px1Pp7m+slwzZ80jkHRcqTUNCZmodufMnQUbSwdb77GL3kkbA0y6rAGOcAOzhkOst3WKJqANdpqqqY12WOALSAXDH37p1vpvW0AZTiHoRggCNbOQNjai5jM5M6env1s3vdH1LhGbSZA3JXkVvM3yEMydsle0hgezoR5tGvbhJvI7g4Xjo/YUPpzBMO1iIJErelnSAwTCySwN9hLmt81bj/9HxokJR1pzjW4gvSnBhnOhnHbXISzEbxuL/NXz0HYKKBFLJbJqSyrTYTpTeZpvNV4M5q5GzovKSgn5jXXTzaur5PGMThku6W3rGVRdAVZkHHJThzy2Np3YleS3KiKo8WOd76XXH2OXH+OmnNriP0ljqlbzWjTPNyUlkGN6gVu9WL6XFvQxv7vdSHI4GhyFLiUAAS7MC5xnLvDRTQaI5lK0aAEkZ0yfsVLT6oPWZaZ39BZiNb3xw5xz74H8djcpc4iHebR7rHOYv4FndRuriHrUvCE3cTsCVy6g/mr5wAUmt8jc0YZyoZRn9RwrXIWaqxRJYbP7gXTLEmtl3UuJb306Ci0J4PpZbZpvBuwasY5jGRHT1qur5PGMYRc9TR5QmS1bCFjcMrOMR44rw+rjqpoUQ3nDDuCTUR0icT+Eic3lLOmeIogeyXaP91aUFIqhmi36HUdTUdI69L7MmCauDvdExLH1SQDOFRL0yGg+kl4krhyLmzYSxphsiij6hpB35GpiX3RQ9yzf6yzWNpwGgDvnXkW2+SnifnjOE0XV7Z/kGXLlxa0lJ9a/BwPNW4hIyo4snZOk+ZzgG52h/eQVFN4bG7skn2MXOnRjKK+nXAswjDTmMbxwsmcIZ80egz52p1TdpJUUzhl5xgPXNCHtQs42x2ASaZLQU9ZLKCSU0SUj6jSCHYBwSUgeWQcTZWj4+JGcn4bWvZIyF655HfVltzmzZjNxb3ngwaDjhHQYH33RXyi4woEBLoTfQgIXD77/dQ6AyVc7DZJRjO0MZoEd+/9A1/e/m02d28H4ILZ5/HDtd/iyuUfLBzzQP0wDwY3W1uaaQ8Ze5bd6husbl2BJEqE3EEWBudjl+w8M/DC8f2yJoG8nsZUufknwsa9m/jy9m/zjWe+x5e3f5uNezcd19efxjSmgpNV2+OkyRhgYg9cPiYq+2SUPoXg6lpGn4jiaHZg6ibZbsXKGgSw19gKfYVKKG4kH4gdIlaXYP7qObRLrWNoISo5FneHi4+u/CDz/jCbMKMEqeOsDyymcXmIs2NLSz5LgztYknpePuf9PNq1rSyzkNjW+3RJf6Jch9rd4UJcK5AdVJBUiX7/EIbNYDgzwi2v/paMnsEhOXBJrrekKXasql7V8HaSRpzGNE5mnFSOAcavUVcaE7UH7QTO9BN7NoE6quKc4URP6Whxq6dgq7VN2GBtvDTEltCfuLPrPrJCBq/dy1XzL2fdDasnVYvPj56WPzb/WfLR89KG06h31rEvfpA5/g5WzTiHBleps1jVeg6bu7ePu6gDMP+CTlyPuejJ9CEJEgig6zppNY10WNyoJ9WPjv6mNsVOpPF+qxkppzGNdwpOOseQR6U68ngEb2pYLfzeEXLQ8tc1hbFUd4dr3Lr0/tgh7go/gOQRaXfOOGLMVp7G7I7JGZxqdfvi6Dmn5zABh2THY3MXFmLGCuu8MOGizuzATM6f9V5+9epd6KhousUTo6FjGNbSn2ak0Qydqxa8eU2xE2m832pGymlM452CKTmGm266iZtuuomDBw8CsHDhQr7+9a+zbt26E3FsVTFeKaLx0hDhzih9QwO0hJpItWZ4ZuB5lDNymO0mNckA82bMLjHSE5U2jsWYVdt1gNLoOeisZVf4NQAWBU8ho2dLIummSIhct/U6k13U+XDnB3isaxu9qUFsokzOOEw/jolNtL76T5165Zu6jXkijfdbzUg5jWm8UzAlxzBjxgy++93vMnfuXEzT5Fe/+hWXXXYZL774IgsXHpug/GQxUSli495N3H7gcAQezmG+CoquWKyJNg9NnhBXNV3OBixjuHv3Hn718l0IMrQFKpc2qhkz/6iPaHesailpPJbS/bFDPNH7J6JKjNmBmcSUhLWYLYBqqCXOx7PNVfI6K69YxtJVE087zA7M5ML2Vfz61bsOa1cdXvIDREHEZ/MyM1Bds+FEIG+8b959J3si+yvqTh/LVNHJPAkyjWm8XTAlx7B+/fqSn//t3/6Nm266iWeeeeZNcwzF0XtWyyIKIqOZaEEwuzwCNzGQBAkTk4yWJasdicQ921y88NhOYqfEaMw2oDarBBvHJ9fKR6Lr0xeQ+06OfemDBaOfWpUpGKSmSGgMa2t+1+EhbQu3v34PkWyMcHYUwzRo8TRZptsEm2grcT6VXmfe4jmTKmN9uPMDbOt9mpSaJqLE0A0dQRBodDXgd/je0jKLIIz93fFoTE/vSkxjGseGo+4x6LrOXXfdRSqV4pxzqrMGKoqCoiiFn+Px+NG+JXAket8bPUAslyCnq0iCyEtDr3B6aGHBaeQjcNME3dRx29wouoJLdpFUU3Qd6KXmd35qXQHcgouoPQ79EHXG8TjGJ9fyj/rIfSdXYqzv2H4f27SnyYhZPDY3H7KvY0F67hEivcO7Dnt69nN7zHJec2pmYkYNRrNRiz3V4UM3dPpTQ9Q6A1w1/3JaIo3sSx8c8zqTZTudHZjJNQuu5PbX70HRc4XMye/wvSVllnzGZ5fszPU0jtnNmJ4qmsY03npM2THs2rWLc845h2w2i9fr5d5772XBggVVH3/jjTfyrW9965gOshizAzO5qH0VP9l5qyWtJ9oIOHw82r2NU+vmFko+LslpReACSIJU4FbPC2fUJAPoaY2ZtW2sG17DQw2bGXQOE9ACXLV4fHKtaHesxFgPtYzwcOtW7JqNtqBl0DYmNlFXfxUto40FCUzJLRH1xkiNHOlXdNbMYl/0EIvrT+WN6AEyWhZZlLmobRUbOi8hfTAzRkqz2s5ENUzEGDkexuuRHA3G69cA01NF05jG2wBTdgzz58/npZdeIhaLcffdd/Pxj3+cbdu2VXUOX/3qV/n7v//7ws/xeJy2trajP2Lg9IaF1LvqqHXU4JKdOGWHpXwmH2nMhrMR6p21mBzpMViPtRbj5smz2ePehzqqspoVzOlpJyxFOfXiThbKp4z7/uVjsUPZMIpNocV1hJCuW+1DXCsg3CuUTEj5Z/nw9Jb2K9w2J29ED2CX7DQfjqIf7d7GRTNXMbtjZsVJq/GMdKUafVMkRN1wjWXgZ0zOwJ8IGc6Jms+TkRs1csak9aWnMY1pTB1Tdgx2u53OTosq+cwzz+S5557jP/7jP/jZz35W8fEOhwOHw1Hxb0eLkKueGkcAAwOn7CgxIMubzmRpw2m8NPwKggB1jlocsr1ipFxscBuUekJCPeKvRfbcvY/mK6rLV5aPxdbV1+ALeImKcYKmZdBchpPmxibaP9cyxoiV9ysm2k2YipRmvkYfVWLIoszls9/Phw5ePGUDX0nZ7njIcE40OTSR3KgyqKAldSSHdU5br26a1oyexjSOM455j8EwjJIewpuBiYzLi8Mvc/PuO0nkUoWpl0oNzLzBTexK0PvrAQT74Rr+gMKB/+xG9koIojCh7vGpDfPIakrheOwpG6t3rETfq9Hl7qX5ilBBMQ7GTs50J/p4vPuP9CUHaPE2VRzhLN+DqFTi2R87xM2772Q0GyWtZtBMnf/aeSt9rw3yIfPiKRn4MWJDU+xtjIfxJoeqyY32/24IPaujJXXMnImm6OjZLF3/3TutGT2NaRxnTMkxfPWrX2XdunW0t7eTSCS44447eOKJJ3jkkUdO1PFVRTXjsj92iB/tuJlwNoIgQESJ8aMdN1dtYLo7XOSGc5i6ib3OjjqcI9unYGZMtLiGo9WBaJoT6h4vjZ2GKAgowyq+mz20pBqxtdqqGuJ8vyIf4SdySVJqmqgSt0Zqx2kMVyvx3L33DxyId6MZKgIidtGGZqg83vIkK5SzmJFrmbSBLy+XHU1vYzyMNzlUXPYiAIldCXLhHKJDxMyZluDS4X+5EZXEruS0Y5jGNI4jpuQYhoaG+NjHPkZ/fz+BQIDFixfzyCOPcOGFF56o4xsX5ZQSIVc9O0ZeYTgTxi7ZsIs2cobKcCbMjpFXJqTSUAYUlIEcpnJYDlQDpVtBaHMgCEJVY1o8YunIOVjlWs4MvRktriE6RbSIVvG5xTsZC4Pz6UsNoBk6X1hybVV92GolnnBnlG29Tx/eUhAwMMgaCgICEVeURz3b+GT2rydt4N0dLupW1zL0wAhaPDuGkvxEodzpuee6SL6cQo1omIZZosQniIJFoX54R2Ma05jG8cGUHMMvf/nLE3UcR43yufdTajotu2Ga1qC8aYnwpPZnSEuZqkR3zVeE6L6l3xLkyRsfATBA6c/hanNWNKblC3fDsTCPdjxBx6OtNMebwDSRa20Vn1s+odPiaSo00auhWomnb8hyKi2eJg4legqPlwQJSRJ5seFl3rf/EG1C66QM/OD9Q4xuiWBqBoIsElxdc8Jr+eVOT+lXGH4ojK1extFkQxlQrUU9EwTZ+m5tQRu+Rb4TelzTmMa7DScN7XYllBtlE5Md4VeocQTQTIOMmkHVNAIpP4HbvOz5l30M3j9U8bUaLw0x+8vt2II2SwbUmQ9LAZOqusd5456nxq6z1ZC1KUQ9CczDqnBmziDbmx3z3OIJnTzV9kT0EJVowCW3REuoCY/NjdfuKXm+Q7Izs6YNOSThusbFvBvmTKnx7OpwIblFwluipA9mxn3esSLv9Gx1NgRRQM8amDkTdURDHdWxN9mQ/RK2Bhu2Ohlnq5OZn2mdLiNNYxrHGSe1Yyg3yi7J0mo4r3U57b4WArKfUKKej+z7ALMCMzEP9wqqGbi6FbU0f6QRURJBB9EhINdKuNqdVXWPy437SGYUp+6kubkRR70dJNDiOvu/1zXGKeWb6MW6DBMtneWzG0GwxmAFQaD5ihALFswrvJZNtOGQ7NQ761hQNw+7ZMPr9jJ76cxJGdFyA22rs6GndXLDY2VUjyeKnZ6W1FCHLW4nBEttTx1SsTfYmfuvs5j37Tmc8v/NnZ5ImsY0TgBOWnZVKDXKOV2lK9GLYRrsGHmF9R0X0TnaQWZThlmBmSVll8SuZNXRz/ZrWhEwGXwgjKkZ2ALj03KXT0i5ZCcX966kcaAeJZJDMA8LAklUbEIfDbdPtfHV4td6aegVHu3eVshCprLlfKIbz9VQPAac3pfG1EwQLU1uUzcRRIHalQHqVtSe0OOYxjTe7RBM03xTO3fxeJxAIEAsFsPv9x/z623cu4mbd9/JoXgPJpazcNkcOCUnN8z6pwJ1Rd7AaTENySVh6ua4M/1T3fgtXirzbHPRfUs/Sp+CaBdwtDiwh+wovQpzvtpRMrp6InEsZHQnYrltshh9KsLeGw+ihlUEWbCazjo4muws+OH86dLRNKZxFJiK7T2pMwawomRREPi3Z/8D1dQYVSKIORGn5CBel2DWFW2FRTRBEqyetF2wRlPHmemfqu5xyfjlpWAL2tj/vS5M3UCwi2QOZRAkESNnjP9CxxHHQiY3laW64w3RLiLZRaQ2B7mBI1ThwfNrp53CNKbxJuCkdwxgbTfnDGtixSW7yGgZErrKjuHdJToMzZEGem7pL6mdH6+lrTHHtKKWkcdHGXlkFKUvByZIHpGum3pRw+pJURufqnM8XsiXskzTxD3XhRpWEW1i1T7PNKYxjeOLd4RjcMh2vDYPGS1LRsug6ToGBje9/Cs0Q8Nn89LsaeQjDes51d2J0q8guiS0mHrCovj0wQzpNzLYGmTUYdWaTjJBV3R6bxsAAXyLjv/G7vEmvXuzUH7c+V6DFtGQvW/ODsU0pjENC+8IxxBy1dPkCTGYGiamJtDQAIupU0AgQYpaPctd4Qe4fsEncf7BYTmDClH80dblyw1bfrJH9ttQRzREt4CpmBhZg9ywytO3PU+yMcX81XNY9qGlhdc5WfsCR4P8Z5WeFrFttI057reqlDWNabzb8Y5wDCVU3IZW9leTnJ5DRCSZTtLXP8D8hjljovj+3w2xJfQn7go/UCISs05ePaFxqmSQfYt9SG4JPaMjiGBmDUwBcsMq20/5M1vPsrQbXD0OPv7MX3Hl8g8ek0jNsZLevdmZRv6zJtNJxAGRtc3v48LsqjHHPe0QpjGNNx8n9R5DMfJU3M2eRoTD/7Ng/TeppnCaLgIxH7LfBoKA6BbBAMkp0i32cmfXfeiKTpPegK7o3Prn37L1O0+x78aDVZfj0gcz9N42gJbUkOvkwq4EWOytklNCdIqAgGQT6a8ZZMsZfwLJpElrwDBM7uy6j209T49Z1rv99XvYHzs0qc9/LLsHg/cPsedf9o37OY8nihcTm8VGDMPk0Vnb6HMNHPPORJ4eZbLnbRrTmMZYvCMyBjhCxZ3VsgWOJLCmWSQkPDY3V7Z+kDajFT17JIpHsjZsY00JUpkkwZE6srqCHRtD7hFGHVFmts6oGoEP/n6ITHcWRFBHNRxNNgzNMtTF5RAjZ6COauzZdICsrNCkNkAOagQ/USHGvvjBYxKpqbZ7YOQMos/FqmoYnCh67fFQTAVimiY1mp9B2whhOULDaLDizkQ+o+nzDBKvS1QstR0PWdBpTGMa7yDHULxo5rf7iKsJnKKDWmeAC9tX8eHOD1i8/kNW2Ud0iugpA8klITkkZi1txxa3EbXHqTUDxMw4joydILVVJ5jSBzOEt0Qs7h5BwDQNsj05nK2OgmErL4ecqnXi6nEQMWPUCH7SjRm8bi9z/B0VRWr8oz6i3bHC61Ur95RrRAiSgK1J5uAPu1GjqvVZvRKOkL2k9zAevfZ47zcZVCtPlYj1uGpJN2ZwRRx4e9yFTe7ix3ff3MPgA2GeaH2Kx+f/Ea1Wx1/rKzH85fQo07Kg05jG0eMd4xhgchKW5VG8aLemktIHMlzwwrlsXrCdIWkEh+rgfS+fQygWxGwzS7Z/8wZv6KERlMEcJiamYiJIAv21A0jvk6ipDTCbsQZp2YeW8vFn/oo7u+4jKsTwur1cNf9yVs04h5HsaInGxPr0BeS+k+ON2H70jIEoW9oQ1RrL+c82+PshwlsixJ5LHP6LtTWsZ/RCPyWfEVTLNOK7koxuiRx1I3u8RvgYPY1aN1d1buDc5ctLnEj6YIaeX/cx/GCY/ppBHu3YBhrUDdaguHMlhn88ydBpxzCNaUwN7yjHAJNb6spH8dt6nualP72M7ykfi/afwnvDy+gcmUVmdhb/sJeGkXpEt1giqZnYmaD/d0OoMZXciAqGiegQMA3445w/s/WMP2GGDLzbvVVLGVcu/yBnn7p0zPRR3rHtfe4AztecuB9zktNyaFENI2stqDvaxar6EHnEnktYrRURBAQMxUTwCqBZ/RQ9pRcyn/JMQ3JLBFfXEN4SOaZG9kTlqYmoQAbvH+K5e15iIDZMwOMn5ouTteVoiNeBDWpMPwPqcMHwTyQZOo1pTGPyeMc5hsni+if+ma3dT6HrOvISmZVt7+FTW68k1FWPLSVbHEmfKR2ZBNjzL/swTRO5xkZuWMUUQTAF+v0DbFm0Hdkn01I3cSmjmgPTfqjhfcSDoRpkdQUkEG1CgQJcHVKxn2pDG62s8VBoQtfbUEc1DN3a0TCz1ga2njXG1PCLs6g+zyCvD+0n88RYjqnccI6B2qEJx2knq/5WfA6K9ZxTe9L85i+/5/ElT5IRsjhyDs7cuxinZifmSlKr+4kK8RLDn59Mu2ffg0SzcWqdgSnxQ01jGtM4gnelY/jOs//Jo13bMDERENAFne3BZ3nv6Wdx6s55NP9ViLqVR+gX8v8dfGCY3KiKo8WOIAiINgFDBWeLnZQ/jeJWaQ01H3UpY/SpCCOPjAImolvESBigg2E/TGclYlFtj1ibwMqAQvqgpTFRbFgFSUAdUbEFZXKDKqZoaVPk+ymVlsXcHS4e0rYcGSFdJrK268gIqeSWeEjfyl3bH5iwuTsZEr7i/kM+C1MGFbSYRp9viEfXbsPUTOozQeLuBM937uTMNxbzQudORpsiBBylhn/j3k082rUNzdCwSTIXta2abjxPYxpHiXesY6i2KLY/doiHurYUppVMTHTBwAQOmj0sDiwocQp5DN4/RO9tAxYRX1TDOcOO5JcwIzp6xqS+Joi/xktUjBM0j66UkT6QwdRNRI+IKAqYDgNTARQsjQhTABP0rIGhmvTc0o/kHsI910X6jQx6WkdXDPS0jpGxMgW5RqZ5XYjAWf6KU0nF5yXfvG2vm8GAMsQjPEHncx20Ca2oG1TuCj8wqeZupfJUsTMq7j8IktX7EJ0i3VIfkaYYYW8Exa5QH61DFAT8aR8j/jDzlU7W112ItFYaI+eaP/bZgZmEsxEe7d7GRTNXTWcM05jGUeAd6RjKxxYvalvF6aGFhFz1DGVGgMO1dwxLHtI0EQ2BGamWitF0vmYu2AWc7Q6UHoVMl4KrzUnzhhC+RT4WNswnqyklzeOpljLcs1wW0V/WwHCK1qSTbDHDijYRySniW+QhsSuFYD9cohlQGHlkFEeLHblWRnk1BQi4O13oSb3AMTRRb6C8edvUHKLL0WOJ+3TM4dnoiyReSzLDUzkjKp9Aqra5XNx/kOtksj0KWkTjmeXP83DrEyh2BdGQyIkqcXcSf9pL0p8iWFPLsn9YQmdnR+GYs1lL/GgoPoLbdNLkCSEKAjVuPwOpIYbiI7Q4Gid9/qcxjZMZNpsNSZKOy2u94xxD+dji3ugBfrLrVoLOOmqdAS5qX4UsSIdL9qYlFSnAysDZ/PWXLqtoQMtr5rJPRulTaP1YE43rj0zqbGDq2grFqFtRS/3aOkYeGcVIWSWh0CVBZny8pWBgc8M54i8lsdfZEUQBySlaWYZLsvQLhMOOThKwtTknTRJYqXmbF/dJbEuQ+kMacaFInzhIqC5IsiZdyIjyGYAa0xBkgdD6etqvaa24uZw/l4IM6W7FmpLyDvFI2xOYhkl9tI64O4lgE9AkjXDNKJfMW8P7Ws9BkkwOHDgw5tg9hp1Pt1yJCUiCiG4aCAHwxO0cSI59/DSm8U5FTU0NTU1NCIIw8YPHwTvOMRRHvllNIaYk0Ewdu2Qjq2d54MCjqIaKXXJgGDq6aeCXffzvNdfjDlQ2nuU1cyNrYA/aK2oNFzdUj4b3aN435lB/QR3pAxncs1wFUZpiA1t8LHrWciBGRkcOyGBafRNswoS1/fJGcMkI6eGMpykSYs/v9tFiNnJJZA0PBjfTHeujxhXgbxZvKPxdjaiocQ1TNen5RR8CJm3XzKh4LgVJINujYBhWDyXqiZGVD5eOEPGnveTsOS5+YTWLls+lbUYLra0tuN3uqhd8TIkTVRIYpoEoiNQ4fAQcx673MY1pnAwwTZN0Os3Q0GHWhebmY3q9d5xjKI58RUQyehbTNOlLDSCLMhIiHrub+eZsMpEsoiqScCV5/fG9zN5Q2XhXWh6rWVlqdMoN7nhbuBPxEtWtqK2qUlapfl+/to7kyymUPgXJJyPaBLRRbdzafqXdhEojpNHnYoVsaU3kXGpUPwdz3Zx5yulc0rmG6HMx1JiGGtcQBBDcIkbaYPCBMMHVwYpN7tqVAXrvGAANEKBWC+BQHSR8KfwZL3FHEofq4BSzk/blrbS0NhMMBqt+54qm4JE8uF2W47AJMg7ZUfXx05jGOxEul3WvDQ0NEQqFjqms9I5zDMWR71BmBMM0kASxoNOgYhLQ/SQiKfw5LwlPEqfmwHjEJH1mpmrJpXh5LLI9xui2KLHnEjRfYRnWYoOrblC53V55C9ezzXXMDKjl9fvEzgTJl1MgCNj8EvXrgjgabICAb5EXmDz1RfkYbXG29Me5f+bB4GayksIL0ZfJ7s2yrmG1pbKmmghuETQT0SZgakbVElbjZSFGHhslN5RDcIm0Jpu44NVz2bL4KWIdCVyCk8sd61h1+XIGcv243e6q5yKSjRLORguZQtBZg9fpmdL5nMY03inI3yuqqk47hnLkI98nev/ET3fehmqoKLqCLMi4ZCer3OfwlPEcw75RXIaDdeHVNI00TKoWH3suUaIA13vbAGClctggF8nx6qY3iL8vQVugdAu360Av/tu8GKqBrd6GkTGOmpcoX78vboy7m1yooyoDG4eQ3aXypY5mx6R2Cyq9T/MVIf7yh51sqt0MQFughaQjbTm7lacRWl9Pzy/6MNIGok1A8kvYAraSLfHi7Mjd4aLp8ga6f9GPqZhgE1g9uILTdp+C53+5aZ/VyuzATKu5fICq5SNFUwhnowDYJRuaoRPORnHLrumMYRrvShxrbyGPd6RjAApR7x8OPE5WyxYyBqfs5MOzPsCZ9y9m1BEl5AwS6qtHcAtVxe7zvQJnj3OMcU3vTaNnDAzFwMyZYII760Y8VaAvUtqoFf8oVCTcS+xKjltaGq/0VN4YF53iYcUzAWebs5AZtH+udcLdgmpovDSEp92N8ZrBDE8zNo8Nh+koTCUtv+ZMBEwGHwhjaoa1HFi0JV4pO2q7ZgYmAkMPjGBqJraAzFkfWEzj6dWzJ0VTUE2tUCpSTQ3DNLBLNgBkUSKnq6imhoNpxzCNaRwtpkS7feONN7Js2TJ8Ph+hUIgPfvCDvP766yfq2I4Z+bKSU3ZatNuyk6vmX86CBfM46wOLWRQ5hfp9dRWJ2/LYuHcTX97+bb7xzPf45tD3eLLzz6ijqrVoNqpaXEuZw07hMJojIVbvWAm6SXesD1Mx+UhwPd4nPQgih7WnLcI9La3T++v+qpTXE1FiF5d6TMNEDVussrb6Uvpt0S7SfEUIQRCsPsk4n7kS2me14vN5GTSHCWcj9KUGSvY02q6ZwYIfzGPu12cz74Y5+Bb7SkpXeTry9MHMkde8pvXwc2Yx74Y5JSW1/bFDvDT8CtphfY1INkp3sp++5CDdyX4i2Sg2QUYURDRDB0AzdERBxCa8veOdJ554AkEQiEajb/WhHBe80z5POTo6OvjhD394wl7/m9/8Jqeffnrh50984hN88IMfPGHvNxlM6Q7atm0b1113HcuWLUPTNP75n/+Ziy66iN27d+PxvD3rutU4eVKrMkTb49QkA8ybMXvCpa98r2DLku3Mi86mqbfhMK9QLYP3DqPFdSjK4la+/B7mDs8iHkyw9LOL6ZBmsE8/iGOGJXBv6mDqJqZmlpSmiktLk+kLlDejRZuI/XCZynSVkv/VLAsctSra7MBM5tfMYdPBx9FMDVmQuaTjgpJ+RL68tT92iP0vHiIjVqbVKH7fSiOt+ca923Ty6ZYrCWciZAQFKCsZeV0EnTWEs1FyulroMUyXkd5+EASBe++9900zeB0dHXzxi1/ki1/84pvyfu80TMkxPPzwwyU/33rrrYRCIZ5//nnOO++843pgxxPlDdUxE0NNl7OBsfQJFRk7PX14/pebOWpHoQwT2R5Dz1rTT6iHnyxAUyREc6yRdqGlosC9oZnWFE2RuE6x8Zws51ClZnS1reNKhngyY7X7Y4d4PbqPGd7mQlnu9eg+9scOVTy31Wg1JipdFTvjJk8IE4jnEgg2EVmUMUyzpGRU66zBLbtKSkzTGItcLofdPnHZ8K3EyXCM7xYck4JbLBYDoK6urupjFEUhHo+X/HsrUZ4FjKeUVjz6aphGgeaifVYrNcsCBSPbenUTthoZ9CPPFWQBwSYgeSVEu1iI7AVBOCJwf3kDklMk250lF86R7c4iSEd6HWPKROMYV3eHq3BMvsU+mv+qkRmfbB5ToilHcansy9u/zca9myo+Lu8kW7xNBF211DgChLOjvDT8SsVz2143A7lW5pH2JzgQOzTp0lX+fYLOWkRBOLywpqOZGlldIatlUXSlpGTkkB14bZ63jVNQFIXPf/7zhEIhnE4nK1eu5LnnnhvzuKeeeorFixfjdDpZvnw5L7/8cuFvhw4dYv369dTW1uLxeFi4cCEPPvhg4e8vv/wy69atw+v10tjYyNVXX83IyEjh7+973/u4/vrr+eIXv0h9fT1r167lox/9KH/1V39VcgyqqlJfX8+vf/1rAAzD4MYbb2TWrFm4XC6WLFnC3XffXfKcBx98kHnz5uFyuTj//PM5ePDguOejo6MDgA996EMIglD4OV9C+cUvfsGsWbNwOp0ARKNRrr32WhoaGvD7/axevZodO3YUXm/fvn1cdtllNDY24vV6WbZsGY8//njJZz906BBf+tKXEAShpCG7fft2zj33XFwuF21tbXz+858nlUoV/j40NMT69etxuVzMmjWL22+/fdzPlsfNN9/MwoULcTgcNDc3c/311xf+NtHnmQh33303ixYtwuVyEQwGueCCC0qO+UTgqB2DYRh88YtfZMWKFZx22mlVH3fjjTcSCAQK/9ra2o72LY8LSg2PlQWk1HSBKqMY+R6FgMD+2CFSapqL2irz78g+GdkvI9gE5DoZzzwXjiY7jpC9YMgbLw0x74Y5zPlqB/NumIO93o6W1sn2KqReTZPtVdDSOomdlo5CsTOZbF8g35PouaWP/t8OFV6rGHn5y7ycaFbL4rV5yGrZSTnJ3uQAu8KvMZKJ8MvddxScSfm5bWoOYTQZFq3GBA6q0vsYpolu6JgmyIKEiICBiWbo+O3eKTmC9MEM0ediJT2OE4V//Md/ZOPGjfzqV7/ihRdeoLOzk7Vr1zI6OlryuK985St8//vf57nnnqOhoYH169ejqlbKed1116EoCk8++SS7du3i//7f/4vXa40eR6NRVq9ezdKlS/nLX/7Cww8/zODgIFdccUXJ6//qV7/Cbrfz1FNP8dOf/pSrrrqKBx54gGQyWXjMI488Qjqd5kMf+hBg3a+//vWv+elPf8orr7zCl770Jf7mb/6Gbdu2AdDd3c3ll1/O+vXreemll7j22mv53//7f497PvJO8ZZbbqG/v7/ESe7du5eNGzdyzz338NJLLwHwkY98hKGhIR566CGef/55zjjjDNasWVM4f8lkkve///1s3ryZF198kYsvvpj169fT1dUFwD333MOMGTP49re/TX9/P/39/YDlUC6++GI2bNjAzp07+e1vf8v27dtLjPgnPvEJuru72bp1K3fffTc/+clPCktj1XDTTTdx3XXX8elPf5pdu3Zx//3309nZWfj7RJ9nPPT393PllVdyzTXX8Oqrr/LEE09w+eWXW9WJE4ij7tJdd911vPzyy2zfvn3cx331q1/l7//+7ws/x+Pxt9Q5TJW3f0PnJQxnwtyz70FUXePR7m00uIMly2r5cVHvQg/p/Wm0iI4a1bEF5KqGPNubpf93Q0guEdEuWHQWsoDkFkv6COWU2AfquknFMhWdU15/2lANbEFrQ7u8J1FcRjMxGcmEMU0wMBARccnOioyweSd58+47ORTvQRBEOvxt2CVbYUdjPFqNalvl5SjeQxlIDWH6TURBwCk7MUwDA8tZuG2T749MtNh3PJFKpbjpppu49dZbWbduHQA///nPeeyxx/jlL3/JV77ylcJjv/GNb3DhhRcClhGfMWMG9957L1dccQVdXV1s2LCBRYsWATB79uzC83784x+zdOlSvvOd7xR+d/PNN9PW1saePXuYN28eAHPnzuXf//3fC4+ZM2cOHo+He++9l6uvvhqAO+64g0svvRSfz4eiKHznO9/h8ccf55xzzim87/bt2/nZz37GqlWruOmmm5gzZw7f//73AZg/f37BcVVDQ0MDcISuoRi5XI5f//rXhcds376dZ599lqGhIRwOy/F/73vf47777uPuu+/m05/+NEuWLGHJkiWF1/g//+f/cO+993L//fdz/fXXU1dXhyRJ+Hy+kve78cYbueqqqwp9h7lz5/Kf//mfhc/V1dXFQw89xLPPPsuyZcsA+OUvf8mpp55a9bMB3HDDDfzDP/wDX/jCFwq/yz9/Mp9nPPT396NpGpdffjkzZ1r3ZP6a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",
      "text/plain": [
       "<Figure size 400x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from umap import UMAP\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "source = out[\"y_pred\"][:,]\n",
    "target = Y_te_post[:, top_genes_idx]\n",
    "\n",
    "\n",
    "# Instantiate UMAP\n",
    "umap_model = UMAP(n_components=2, random_state=42)\n",
    "\n",
    "all_sample_umap = umap_model.fit_transform(np.vstack([target]))\n",
    "source_umap = umap_model.transform(source)\n",
    "target_umap = umap_model.transform(target)\n",
    "\n",
    "\n",
    "fig, ax = plt.subplots(figsize=(4, 4))\n",
    "# ax.scatter(source_umap[:, 0], source_umap[:, 1], s=10, alpha=0.7, label='train_post', color='C2')\n",
    "ax.scatter(target_umap[:, 0], target_umap[:, 1], s=10, alpha=0.7, label='observed treated cells', color=\"#C331C8\")\n",
    "ax.scatter(source_umap[:, 0], source_umap[:, 1], s=10, alpha=0.7, label='predicted cells', color=\"#24A139\")\n",
    "\n",
    "ax.set_title(f'{drug}')\n",
    "# ax.set_xlabel('UMAP 1')\n",
    "# ax.set_ylabel('UMAP 2')\n",
    "ax.set_aspect('equal', 'box')\n",
    "# Add a legend to distinguish the points\n",
    "ax.legend()\n",
    "# Adjust layout\n",
    "plt.tight_layout()\n",
    "# Display the plot\n",
    "plt.savefig(f\"./plots/scgen_on_4i_drug_{drug}.png\", dpi=300)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "59ab253e",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5e84a3b4",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "412e570e",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 52,
   "id": "a06d14a5",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "X_pre cells: (17565, 1000)\n",
      "X_post cells: (3541, 1000)\n",
      "(2832, 1000)\n",
      "(709, 1000)\n",
      "(2832, 1000)\n",
      "(709, 1000)\n",
      "Median heuristic gamma: 0.002211581349119358\n"
     ]
    }
   ],
   "source": [
    "drug = \"givinostat\"\n",
    "X_pre = adata_sc[adata_sc.obs[\"drug\"] == \"control\"].copy().to_df()\n",
    "X_post  = adata_sc[adata_sc.obs[\"drug\"] == drug].copy().to_df()\n",
    "\n",
    "print(\"X_pre cells:\", X_pre.shape)\n",
    "print(\"X_post cells:\", X_post.shape)\n",
    "\n",
    "top_genes_ids, top_genes_short, top_genes_idx = topk_markers(adata_sc, drug, k=100)\n",
    "\n",
    "X_tr_pre, X_te_pre, Y_tr_post, Y_te_post = split_train_test(X_pre.values, X_post.values, 0.8)\n",
    "\n",
    "print(X_tr_pre.shape)\n",
    "print(X_te_pre.shape)\n",
    "print(Y_tr_post.shape)\n",
    "print(Y_te_post.shape)\n",
    "\n",
    "\n",
    "# Compute median heuristic gamma on training data\n",
    "median_gamma = median_heuristic_gamma(X_tr_pre, Y_tr_post)\n",
    "print(\"Median heuristic gamma:\", median_gamma)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 53,
   "id": "b4d17c06",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "**************** Run: 0 ****************\n",
      "Epoch 0 | Train Loss 4142389.7500 | Train MMD 0.0110 | Test MMD 0.0132 | Test Wasserstein 8.0244\n",
      "Epoch 10 | Train Loss 4016080.0000 | Train MMD 0.0093 | Test MMD 0.0128 | Test Wasserstein 7.9222\n",
      "Epoch 20 | Train Loss 3908877.0000 | Train MMD 0.0087 | Test MMD 0.0120 | Test Wasserstein 7.8655\n",
      "Epoch 30 | Train Loss 3810793.0000 | Train MMD 0.0084 | Test MMD 0.0110 | Test Wasserstein 7.8190\n",
      "Epoch 40 | Train Loss 3720023.2500 | Train MMD 0.0081 | Test MMD 0.0101 | Test Wasserstein 7.7894\n",
      "Epoch 50 | Train Loss 3635463.2500 | Train MMD 0.0078 | Test MMD 0.0093 | Test Wasserstein 7.7704\n",
      "Epoch 60 | Train Loss 3556318.7500 | Train MMD 0.0075 | Test MMD 0.0086 | Test Wasserstein 7.7560\n",
      "Epoch 70 | Train Loss 3482020.7500 | Train MMD 0.0072 | Test MMD 0.0079 | Test Wasserstein 7.7445\n",
      "Epoch 80 | Train Loss 3412084.0000 | Train MMD 0.0070 | Test MMD 0.0073 | Test Wasserstein 7.7366\n",
      "Epoch 90 | Train Loss 3346111.2500 | Train MMD 0.0068 | Test MMD 0.0067 | Test Wasserstein 7.7322\n",
      "Epoch 100 | Train Loss 3283780.2500 | Train MMD 0.0067 | Test MMD 0.0062 | Test Wasserstein 7.7309\n",
      "Epoch 110 | Train Loss 3224795.7500 | Train MMD 0.0065 | Test MMD 0.0058 | Test Wasserstein 7.7316\n",
      "Epoch 120 | Train Loss 3168889.0000 | Train MMD 0.0064 | Test MMD 0.0054 | Test Wasserstein 7.7336\n",
      "Epoch 130 | Train Loss 3115818.7500 | Train MMD 0.0064 | Test MMD 0.0051 | Test Wasserstein 7.7371\n",
      "Epoch 140 | Train Loss 3065397.7500 | Train MMD 0.0063 | Test MMD 0.0047 | Test Wasserstein 7.7421\n",
      "Epoch 150 | Train Loss 3017445.0000 | Train MMD 0.0062 | Test MMD 0.0044 | Test Wasserstein 7.7483\n",
      "Epoch 160 | Train Loss 2971793.5000 | Train MMD 0.0062 | Test MMD 0.0042 | Test Wasserstein 7.7559\n",
      "Epoch 170 | Train Loss 2928289.7500 | Train MMD 0.0061 | Test MMD 0.0039 | Test Wasserstein 7.7648\n",
      "Epoch 180 | Train Loss 2886796.0000 | Train MMD 0.0061 | Test MMD 0.0037 | Test Wasserstein 7.7747\n",
      "Epoch 190 | Train Loss 2847184.2500 | Train MMD 0.0060 | Test MMD 0.0035 | Test Wasserstein 7.7857\n",
      "Epoch 200 | Train Loss 2809339.0000 | Train MMD 0.0060 | Test MMD 0.0033 | Test Wasserstein 7.7972\n",
      "Metrics: {'mmd2_gamma_median': 0.0033099483987888423, 'mmd2_gamma_0.5': 8.995337426692034e-05, 'mmd2_gamma_1.0': 4.204512256328778e-05, 'wasserstein_distance': 7.797176120112977, 'R2_feature_means': 0.8371524316037473}\n",
      "**************** Run: 1 ****************\n",
      "Epoch 0 | Train Loss 4142389.7500 | Train MMD 0.0110 | Test MMD 0.0132 | Test Wasserstein 8.0244\n",
      "Epoch 10 | Train Loss 4016080.0000 | Train MMD 0.0093 | Test MMD 0.0128 | Test Wasserstein 7.9222\n",
      "Epoch 20 | Train Loss 3908877.0000 | Train MMD 0.0087 | Test MMD 0.0120 | Test Wasserstein 7.8655\n",
      "Epoch 30 | Train Loss 3810793.0000 | Train MMD 0.0084 | Test MMD 0.0110 | Test Wasserstein 7.8190\n",
      "Epoch 40 | Train Loss 3720023.2500 | Train MMD 0.0081 | Test MMD 0.0101 | Test Wasserstein 7.7894\n",
      "Epoch 50 | Train Loss 3635463.2500 | Train MMD 0.0078 | Test MMD 0.0093 | Test Wasserstein 7.7704\n",
      "Epoch 60 | Train Loss 3556318.7500 | Train MMD 0.0075 | Test MMD 0.0086 | Test Wasserstein 7.7560\n",
      "Epoch 70 | Train Loss 3482020.7500 | Train MMD 0.0072 | Test MMD 0.0079 | Test Wasserstein 7.7445\n",
      "Epoch 80 | Train Loss 3412084.0000 | Train MMD 0.0070 | Test MMD 0.0073 | Test Wasserstein 7.7366\n",
      "Epoch 90 | Train Loss 3346111.2500 | Train MMD 0.0068 | Test MMD 0.0067 | Test Wasserstein 7.7322\n",
      "Epoch 100 | Train Loss 3283780.2500 | Train MMD 0.0067 | Test MMD 0.0062 | Test Wasserstein 7.7309\n",
      "Epoch 110 | Train Loss 3224795.7500 | Train MMD 0.0065 | Test MMD 0.0058 | Test Wasserstein 7.7316\n",
      "Epoch 120 | Train Loss 3168889.0000 | Train MMD 0.0064 | Test MMD 0.0054 | Test Wasserstein 7.7336\n",
      "Epoch 130 | Train Loss 3115818.7500 | Train MMD 0.0064 | Test MMD 0.0051 | Test Wasserstein 7.7371\n",
      "Epoch 140 | Train Loss 3065397.7500 | Train MMD 0.0063 | Test MMD 0.0047 | Test Wasserstein 7.7421\n",
      "Epoch 150 | Train Loss 3017445.0000 | Train MMD 0.0062 | Test MMD 0.0044 | Test Wasserstein 7.7483\n",
      "Epoch 160 | Train Loss 2971793.5000 | Train MMD 0.0062 | Test MMD 0.0042 | Test Wasserstein 7.7559\n",
      "Epoch 170 | Train Loss 2928289.7500 | Train MMD 0.0061 | Test MMD 0.0039 | Test Wasserstein 7.7648\n",
      "Epoch 180 | Train Loss 2886796.0000 | Train MMD 0.0061 | Test MMD 0.0037 | Test Wasserstein 7.7747\n",
      "Epoch 190 | Train Loss 2847184.5000 | Train MMD 0.0060 | Test MMD 0.0035 | Test Wasserstein 7.7857\n",
      "Epoch 200 | Train Loss 2809339.0000 | Train MMD 0.0060 | Test MMD 0.0033 | Test Wasserstein 7.7972\n",
      "Metrics: {'mmd2_gamma_median': 0.0033099483987888423, 'mmd2_gamma_0.5': 8.995338881883556e-05, 'mmd2_gamma_1.0': 4.204512301803513e-05, 'wasserstein_distance': 7.7971761414854415, 'R2_feature_means': 0.8371524028606401}\n",
      "**************** Run: 2 ****************\n",
      "Epoch 0 | Train Loss 4142389.7500 | Train MMD 0.0110 | Test MMD 0.0132 | Test Wasserstein 8.0244\n",
      "Epoch 10 | Train Loss 4016080.0000 | Train MMD 0.0093 | Test MMD 0.0128 | Test Wasserstein 7.9222\n",
      "Epoch 20 | Train Loss 3908877.0000 | Train MMD 0.0087 | Test MMD 0.0120 | Test Wasserstein 7.8655\n",
      "Epoch 30 | Train Loss 3810793.0000 | Train MMD 0.0084 | Test MMD 0.0110 | Test Wasserstein 7.8190\n",
      "Epoch 40 | Train Loss 3720023.2500 | Train MMD 0.0081 | Test MMD 0.0101 | Test Wasserstein 7.7894\n",
      "Epoch 50 | Train Loss 3635463.2500 | Train MMD 0.0078 | Test MMD 0.0093 | Test Wasserstein 7.7704\n",
      "Epoch 60 | Train Loss 3556318.7500 | Train MMD 0.0075 | Test MMD 0.0086 | Test Wasserstein 7.7560\n",
      "Epoch 70 | Train Loss 3482020.7500 | Train MMD 0.0072 | Test MMD 0.0079 | Test Wasserstein 7.7445\n",
      "Epoch 80 | Train Loss 3412084.0000 | Train MMD 0.0070 | Test MMD 0.0073 | Test Wasserstein 7.7366\n",
      "Epoch 90 | Train Loss 3346111.2500 | Train MMD 0.0068 | Test MMD 0.0067 | Test Wasserstein 7.7322\n",
      "Epoch 100 | Train Loss 3283780.2500 | Train MMD 0.0067 | Test MMD 0.0062 | Test Wasserstein 7.7309\n",
      "Epoch 110 | Train Loss 3224795.7500 | Train MMD 0.0065 | Test MMD 0.0058 | Test Wasserstein 7.7316\n",
      "Epoch 120 | Train Loss 3168889.2500 | Train MMD 0.0064 | Test MMD 0.0054 | Test Wasserstein 7.7336\n",
      "Epoch 130 | Train Loss 3115818.7500 | Train MMD 0.0064 | Test MMD 0.0051 | Test Wasserstein 7.7371\n",
      "Epoch 140 | Train Loss 3065397.7500 | Train MMD 0.0063 | Test MMD 0.0047 | Test Wasserstein 7.7421\n",
      "Epoch 150 | Train Loss 3017445.0000 | Train MMD 0.0062 | Test MMD 0.0044 | Test Wasserstein 7.7483\n",
      "Epoch 160 | Train Loss 2971793.5000 | Train MMD 0.0062 | Test MMD 0.0042 | Test Wasserstein 7.7559\n",
      "Epoch 170 | Train Loss 2928290.0000 | Train MMD 0.0061 | Test MMD 0.0039 | Test Wasserstein 7.7648\n",
      "Epoch 180 | Train Loss 2886796.0000 | Train MMD 0.0061 | Test MMD 0.0037 | Test Wasserstein 7.7747\n",
      "Epoch 190 | Train Loss 2847184.5000 | Train MMD 0.0060 | Test MMD 0.0035 | Test Wasserstein 7.7857\n",
      "Epoch 200 | Train Loss 2809339.0000 | Train MMD 0.0060 | Test MMD 0.0033 | Test Wasserstein 7.7972\n",
      "Metrics: {'mmd2_gamma_median': 0.0033099483987888423, 'mmd2_gamma_0.5': 8.995338154287795e-05, 'mmd2_gamma_1.0': 4.204512256328778e-05, 'wasserstein_distance': 7.797176132447379, 'R2_feature_means': 0.83715240892833}\n",
      "**************** Run: 3 ****************\n",
      "Epoch 0 | Train Loss 4142389.7500 | Train MMD 0.0110 | Test MMD 0.0132 | Test Wasserstein 8.0244\n",
      "Epoch 10 | Train Loss 4016080.0000 | Train MMD 0.0093 | Test MMD 0.0128 | Test Wasserstein 7.9222\n",
      "Epoch 20 | Train Loss 3908877.0000 | Train MMD 0.0087 | Test MMD 0.0120 | Test Wasserstein 7.8655\n",
      "Epoch 30 | Train Loss 3810793.0000 | Train MMD 0.0084 | Test MMD 0.0110 | Test Wasserstein 7.8190\n",
      "Epoch 40 | Train Loss 3720023.2500 | Train MMD 0.0081 | Test MMD 0.0101 | Test Wasserstein 7.7894\n",
      "Epoch 50 | Train Loss 3635463.2500 | Train MMD 0.0078 | Test MMD 0.0093 | Test Wasserstein 7.7704\n",
      "Epoch 60 | Train Loss 3556318.7500 | Train MMD 0.0075 | Test MMD 0.0086 | Test Wasserstein 7.7560\n",
      "Epoch 70 | Train Loss 3482020.7500 | Train MMD 0.0072 | Test MMD 0.0079 | Test Wasserstein 7.7445\n",
      "Epoch 80 | Train Loss 3412084.0000 | Train MMD 0.0070 | Test MMD 0.0073 | Test Wasserstein 7.7366\n",
      "Epoch 90 | Train Loss 3346111.2500 | Train MMD 0.0068 | Test MMD 0.0067 | Test Wasserstein 7.7322\n",
      "Epoch 100 | Train Loss 3283780.2500 | Train MMD 0.0067 | Test MMD 0.0062 | Test Wasserstein 7.7309\n",
      "Epoch 110 | Train Loss 3224795.7500 | Train MMD 0.0065 | Test MMD 0.0058 | Test Wasserstein 7.7316\n",
      "Epoch 120 | Train Loss 3168889.0000 | Train MMD 0.0064 | Test MMD 0.0054 | Test Wasserstein 7.7336\n",
      "Epoch 130 | Train Loss 3115818.7500 | Train MMD 0.0064 | Test MMD 0.0051 | Test Wasserstein 7.7371\n",
      "Epoch 140 | Train Loss 3065397.7500 | Train MMD 0.0063 | Test MMD 0.0047 | Test Wasserstein 7.7421\n",
      "Epoch 150 | Train Loss 3017445.0000 | Train MMD 0.0062 | Test MMD 0.0044 | Test Wasserstein 7.7483\n",
      "Epoch 160 | Train Loss 2971793.0000 | Train MMD 0.0062 | Test MMD 0.0042 | Test Wasserstein 7.7559\n",
      "Epoch 170 | Train Loss 2928289.7500 | Train MMD 0.0061 | Test MMD 0.0039 | Test Wasserstein 7.7648\n",
      "Epoch 180 | Train Loss 2886795.5000 | Train MMD 0.0061 | Test MMD 0.0037 | Test Wasserstein 7.7747\n",
      "Epoch 190 | Train Loss 2847184.5000 | Train MMD 0.0060 | Test MMD 0.0035 | Test Wasserstein 7.7857\n",
      "Epoch 200 | Train Loss 2809339.0000 | Train MMD 0.0060 | Test MMD 0.0033 | Test Wasserstein 7.7972\n",
      "Metrics: {'mmd2_gamma_median': 0.0033099483987888423, 'mmd2_gamma_0.5': 8.995338154287795e-05, 'mmd2_gamma_1.0': 4.2045120663432176e-05, 'wasserstein_distance': 7.797176143826389, 'R2_feature_means': 0.8371524028606401}\n",
      "**************** Run: 4 ****************\n",
      "Epoch 0 | Train Loss 4142389.7500 | Train MMD 0.0110 | Test MMD 0.0132 | Test Wasserstein 8.0244\n",
      "Epoch 10 | Train Loss 4016080.0000 | Train MMD 0.0093 | Test MMD 0.0128 | Test Wasserstein 7.9222\n",
      "Epoch 20 | Train Loss 3908877.0000 | Train MMD 0.0087 | Test MMD 0.0120 | Test Wasserstein 7.8655\n",
      "Epoch 30 | Train Loss 3810793.0000 | Train MMD 0.0084 | Test MMD 0.0110 | Test Wasserstein 7.8190\n",
      "Epoch 40 | Train Loss 3720023.2500 | Train MMD 0.0081 | Test MMD 0.0101 | Test Wasserstein 7.7894\n",
      "Epoch 50 | Train Loss 3635463.2500 | Train MMD 0.0078 | Test MMD 0.0093 | Test Wasserstein 7.7704\n",
      "Epoch 60 | Train Loss 3556318.7500 | Train MMD 0.0075 | Test MMD 0.0086 | Test Wasserstein 7.7560\n",
      "Epoch 70 | Train Loss 3482020.7500 | Train MMD 0.0072 | Test MMD 0.0079 | Test Wasserstein 7.7445\n",
      "Epoch 80 | Train Loss 3412084.0000 | Train MMD 0.0070 | Test MMD 0.0073 | Test Wasserstein 7.7366\n",
      "Epoch 90 | Train Loss 3346111.2500 | Train MMD 0.0068 | Test MMD 0.0067 | Test Wasserstein 7.7322\n",
      "Epoch 100 | Train Loss 3283780.2500 | Train MMD 0.0067 | Test MMD 0.0062 | Test Wasserstein 7.7309\n",
      "Epoch 110 | Train Loss 3224795.7500 | Train MMD 0.0065 | Test MMD 0.0058 | Test Wasserstein 7.7316\n",
      "Epoch 120 | Train Loss 3168889.2500 | Train MMD 0.0064 | Test MMD 0.0054 | Test Wasserstein 7.7336\n",
      "Epoch 130 | Train Loss 3115818.7500 | Train MMD 0.0064 | Test MMD 0.0051 | Test Wasserstein 7.7371\n",
      "Epoch 140 | Train Loss 3065397.7500 | Train MMD 0.0063 | Test MMD 0.0047 | Test Wasserstein 7.7421\n",
      "Epoch 150 | Train Loss 3017445.0000 | Train MMD 0.0062 | Test MMD 0.0044 | Test Wasserstein 7.7483\n",
      "Epoch 160 | Train Loss 2971793.5000 | Train MMD 0.0062 | Test MMD 0.0042 | Test Wasserstein 7.7559\n",
      "Epoch 170 | Train Loss 2928290.0000 | Train MMD 0.0061 | Test MMD 0.0039 | Test Wasserstein 7.7648\n",
      "Epoch 180 | Train Loss 2886796.0000 | Train MMD 0.0061 | Test MMD 0.0037 | Test Wasserstein 7.7747\n",
      "Epoch 190 | Train Loss 2847184.2500 | Train MMD 0.0060 | Test MMD 0.0035 | Test Wasserstein 7.7857\n",
      "Epoch 200 | Train Loss 2809339.0000 | Train MMD 0.0060 | Test MMD 0.0033 | Test Wasserstein 7.7972\n",
      "Metrics: {'mmd2_gamma_median': 0.0033099483987888423, 'mmd2_gamma_0.5': 8.995338154287795e-05, 'mmd2_gamma_1.0': 4.204512347278248e-05, 'wasserstein_distance': 7.797176123574345, 'R2_feature_means': 0.8371524490091455}\n",
      "**************** Run: 5 ****************\n",
      "Epoch 0 | Train Loss 4142389.7500 | Train MMD 0.0110 | Test MMD 0.0132 | Test Wasserstein 8.0244\n",
      "Epoch 10 | Train Loss 4016080.0000 | Train MMD 0.0093 | Test MMD 0.0128 | Test Wasserstein 7.9222\n",
      "Epoch 20 | Train Loss 3908877.0000 | Train MMD 0.0087 | Test MMD 0.0120 | Test Wasserstein 7.8655\n",
      "Epoch 30 | Train Loss 3810793.0000 | Train MMD 0.0084 | Test MMD 0.0110 | Test Wasserstein 7.8190\n",
      "Epoch 40 | Train Loss 3720023.7500 | Train MMD 0.0081 | Test MMD 0.0101 | Test Wasserstein 7.7894\n",
      "Epoch 50 | Train Loss 3635463.2500 | Train MMD 0.0078 | Test MMD 0.0093 | Test Wasserstein 7.7704\n",
      "Epoch 60 | Train Loss 3556318.7500 | Train MMD 0.0075 | Test MMD 0.0086 | Test Wasserstein 7.7560\n",
      "Epoch 70 | Train Loss 3482020.7500 | Train MMD 0.0072 | Test MMD 0.0079 | Test Wasserstein 7.7445\n",
      "Epoch 80 | Train Loss 3412084.0000 | Train MMD 0.0070 | Test MMD 0.0073 | Test Wasserstein 7.7366\n",
      "Epoch 90 | Train Loss 3346111.7500 | Train MMD 0.0068 | Test MMD 0.0067 | Test Wasserstein 7.7322\n",
      "Epoch 100 | Train Loss 3283780.2500 | Train MMD 0.0067 | Test MMD 0.0062 | Test Wasserstein 7.7309\n",
      "Epoch 110 | Train Loss 3224795.7500 | Train MMD 0.0065 | Test MMD 0.0058 | Test Wasserstein 7.7316\n",
      "Epoch 120 | Train Loss 3168889.2500 | Train MMD 0.0064 | Test MMD 0.0054 | Test Wasserstein 7.7336\n",
      "Epoch 130 | Train Loss 3115818.7500 | Train MMD 0.0064 | Test MMD 0.0051 | Test Wasserstein 7.7371\n",
      "Epoch 140 | Train Loss 3065397.7500 | Train MMD 0.0063 | Test MMD 0.0047 | Test Wasserstein 7.7421\n",
      "Epoch 150 | Train Loss 3017445.0000 | Train MMD 0.0062 | Test MMD 0.0044 | Test Wasserstein 7.7483\n",
      "Epoch 160 | Train Loss 2971793.5000 | Train MMD 0.0062 | Test MMD 0.0042 | Test Wasserstein 7.7559\n",
      "Epoch 170 | Train Loss 2928289.7500 | Train MMD 0.0061 | Test MMD 0.0039 | Test Wasserstein 7.7648\n",
      "Epoch 180 | Train Loss 2886795.5000 | Train MMD 0.0061 | Test MMD 0.0037 | Test Wasserstein 7.7747\n",
      "Epoch 190 | Train Loss 2847184.2500 | Train MMD 0.0060 | Test MMD 0.0035 | Test Wasserstein 7.7857\n",
      "Epoch 200 | Train Loss 2809339.0000 | Train MMD 0.0060 | Test MMD 0.0033 | Test Wasserstein 7.7972\n",
      "Metrics: {'mmd2_gamma_median': 0.0033099483987888423, 'mmd2_gamma_0.5': 8.995337426692034e-05, 'mmd2_gamma_1.0': 4.204512256328778e-05, 'wasserstein_distance': 7.79717613544893, 'R2_feature_means': 0.8371524202660394}\n",
      "**************** Run: 6 ****************\n",
      "Epoch 0 | Train Loss 4142389.7500 | Train MMD 0.0110 | Test MMD 0.0132 | Test Wasserstein 8.0244\n",
      "Epoch 10 | Train Loss 4016080.0000 | Train MMD 0.0093 | Test MMD 0.0128 | Test Wasserstein 7.9222\n",
      "Epoch 20 | Train Loss 3908877.0000 | Train MMD 0.0087 | Test MMD 0.0120 | Test Wasserstein 7.8655\n",
      "Epoch 30 | Train Loss 3810793.0000 | Train MMD 0.0084 | Test MMD 0.0110 | Test Wasserstein 7.8190\n",
      "Epoch 40 | Train Loss 3720023.2500 | Train MMD 0.0081 | Test MMD 0.0101 | Test Wasserstein 7.7894\n",
      "Epoch 50 | Train Loss 3635463.2500 | Train MMD 0.0078 | Test MMD 0.0093 | Test Wasserstein 7.7704\n",
      "Epoch 60 | Train Loss 3556318.7500 | Train MMD 0.0075 | Test MMD 0.0086 | Test Wasserstein 7.7560\n",
      "Epoch 70 | Train Loss 3482021.2500 | Train MMD 0.0072 | Test MMD 0.0079 | Test Wasserstein 7.7445\n",
      "Epoch 80 | Train Loss 3412084.0000 | Train MMD 0.0070 | Test MMD 0.0073 | Test Wasserstein 7.7366\n",
      "Epoch 90 | Train Loss 3346111.2500 | Train MMD 0.0068 | Test MMD 0.0067 | Test Wasserstein 7.7322\n",
      "Epoch 100 | Train Loss 3283780.2500 | Train MMD 0.0067 | Test MMD 0.0062 | Test Wasserstein 7.7309\n",
      "Epoch 110 | Train Loss 3224796.0000 | Train MMD 0.0065 | Test MMD 0.0058 | Test Wasserstein 7.7316\n",
      "Epoch 120 | Train Loss 3168889.2500 | Train MMD 0.0064 | Test MMD 0.0054 | Test Wasserstein 7.7336\n",
      "Epoch 130 | Train Loss 3115818.7500 | Train MMD 0.0064 | Test MMD 0.0051 | Test Wasserstein 7.7371\n",
      "Epoch 140 | Train Loss 3065397.7500 | Train MMD 0.0063 | Test MMD 0.0047 | Test Wasserstein 7.7421\n",
      "Epoch 150 | Train Loss 3017445.0000 | Train MMD 0.0062 | Test MMD 0.0044 | Test Wasserstein 7.7483\n",
      "Epoch 160 | Train Loss 2971793.0000 | Train MMD 0.0062 | Test MMD 0.0042 | Test Wasserstein 7.7559\n",
      "Epoch 170 | Train Loss 2928289.7500 | Train MMD 0.0061 | Test MMD 0.0039 | Test Wasserstein 7.7648\n",
      "Epoch 180 | Train Loss 2886796.0000 | Train MMD 0.0061 | Test MMD 0.0037 | Test Wasserstein 7.7747\n",
      "Epoch 190 | Train Loss 2847184.2500 | Train MMD 0.0060 | Test MMD 0.0035 | Test Wasserstein 7.7857\n",
      "Epoch 200 | Train Loss 2809339.0000 | Train MMD 0.0060 | Test MMD 0.0033 | Test Wasserstein 7.7972\n",
      "Metrics: {'mmd2_gamma_median': 0.0033099483987888423, 'mmd2_gamma_0.5': 8.995338154287795e-05, 'mmd2_gamma_1.0': 4.204511967307128e-05, 'wasserstein_distance': 7.797176125489458, 'R2_feature_means': 0.8371524202660394}\n",
      "**************** Run: 7 ****************\n",
      "Epoch 0 | Train Loss 4142389.7500 | Train MMD 0.0110 | Test MMD 0.0132 | Test Wasserstein 8.0244\n",
      "Epoch 10 | Train Loss 4016080.0000 | Train MMD 0.0093 | Test MMD 0.0128 | Test Wasserstein 7.9222\n",
      "Epoch 20 | Train Loss 3908877.0000 | Train MMD 0.0087 | Test MMD 0.0120 | Test Wasserstein 7.8655\n",
      "Epoch 30 | Train Loss 3810793.0000 | Train MMD 0.0084 | Test MMD 0.0110 | Test Wasserstein 7.8190\n",
      "Epoch 40 | Train Loss 3720023.2500 | Train MMD 0.0081 | Test MMD 0.0101 | Test Wasserstein 7.7894\n",
      "Epoch 50 | Train Loss 3635463.2500 | Train MMD 0.0078 | Test MMD 0.0093 | Test Wasserstein 7.7704\n",
      "Epoch 60 | Train Loss 3556318.7500 | Train MMD 0.0075 | Test MMD 0.0086 | Test Wasserstein 7.7560\n",
      "Epoch 70 | Train Loss 3482020.7500 | Train MMD 0.0072 | Test MMD 0.0079 | Test Wasserstein 7.7445\n",
      "Epoch 80 | Train Loss 3412084.0000 | Train MMD 0.0070 | Test MMD 0.0073 | Test Wasserstein 7.7366\n",
      "Epoch 90 | Train Loss 3346111.2500 | Train MMD 0.0068 | Test MMD 0.0067 | Test Wasserstein 7.7322\n",
      "Epoch 100 | Train Loss 3283780.2500 | Train MMD 0.0067 | Test MMD 0.0062 | Test Wasserstein 7.7309\n",
      "Epoch 110 | Train Loss 3224795.7500 | Train MMD 0.0065 | Test MMD 0.0058 | Test Wasserstein 7.7316\n",
      "Epoch 120 | Train Loss 3168889.2500 | Train MMD 0.0064 | Test MMD 0.0054 | Test Wasserstein 7.7336\n",
      "Epoch 130 | Train Loss 3115818.7500 | Train MMD 0.0064 | Test MMD 0.0051 | Test Wasserstein 7.7371\n",
      "Epoch 140 | Train Loss 3065397.7500 | Train MMD 0.0063 | Test MMD 0.0047 | Test Wasserstein 7.7421\n",
      "Epoch 150 | Train Loss 3017445.0000 | Train MMD 0.0062 | Test MMD 0.0044 | Test Wasserstein 7.7483\n",
      "Epoch 160 | Train Loss 2971793.5000 | Train MMD 0.0062 | Test MMD 0.0042 | Test Wasserstein 7.7559\n",
      "Epoch 170 | Train Loss 2928290.0000 | Train MMD 0.0061 | Test MMD 0.0039 | Test Wasserstein 7.7648\n",
      "Epoch 180 | Train Loss 2886795.5000 | Train MMD 0.0061 | Test MMD 0.0037 | Test Wasserstein 7.7747\n",
      "Epoch 190 | Train Loss 2847184.2500 | Train MMD 0.0060 | Test MMD 0.0035 | Test Wasserstein 7.7857\n",
      "Epoch 200 | Train Loss 2809339.0000 | Train MMD 0.0060 | Test MMD 0.0033 | Test Wasserstein 7.7972\n",
      "Metrics: {'mmd2_gamma_median': 0.0033099483987888423, 'mmd2_gamma_0.5': 8.995338154287795e-05, 'mmd2_gamma_1.0': 4.204512248242158e-05, 'wasserstein_distance': 7.797176115087254, 'R2_feature_means': 0.8371524490091455}\n",
      "**************** Run: 8 ****************\n",
      "Epoch 0 | Train Loss 4142389.7500 | Train MMD 0.0110 | Test MMD 0.0132 | Test Wasserstein 8.0244\n",
      "Epoch 10 | Train Loss 4016080.0000 | Train MMD 0.0093 | Test MMD 0.0128 | Test Wasserstein 7.9222\n",
      "Epoch 20 | Train Loss 3908877.0000 | Train MMD 0.0087 | Test MMD 0.0120 | Test Wasserstein 7.8655\n",
      "Epoch 30 | Train Loss 3810793.0000 | Train MMD 0.0084 | Test MMD 0.0110 | Test Wasserstein 7.8190\n",
      "Epoch 40 | Train Loss 3720023.2500 | Train MMD 0.0081 | Test MMD 0.0101 | Test Wasserstein 7.7894\n",
      "Epoch 50 | Train Loss 3635463.2500 | Train MMD 0.0078 | Test MMD 0.0093 | Test Wasserstein 7.7704\n",
      "Epoch 60 | Train Loss 3556318.7500 | Train MMD 0.0075 | Test MMD 0.0086 | Test Wasserstein 7.7560\n",
      "Epoch 70 | Train Loss 3482020.7500 | Train MMD 0.0072 | Test MMD 0.0079 | Test Wasserstein 7.7445\n",
      "Epoch 80 | Train Loss 3412084.0000 | Train MMD 0.0070 | Test MMD 0.0073 | Test Wasserstein 7.7366\n",
      "Epoch 90 | Train Loss 3346111.2500 | Train MMD 0.0068 | Test MMD 0.0067 | Test Wasserstein 7.7322\n",
      "Epoch 100 | Train Loss 3283780.2500 | Train MMD 0.0067 | Test MMD 0.0062 | Test Wasserstein 7.7309\n",
      "Epoch 110 | Train Loss 3224795.7500 | Train MMD 0.0065 | Test MMD 0.0058 | Test Wasserstein 7.7316\n",
      "Epoch 120 | Train Loss 3168889.0000 | Train MMD 0.0064 | Test MMD 0.0054 | Test Wasserstein 7.7336\n",
      "Epoch 130 | Train Loss 3115818.7500 | Train MMD 0.0064 | Test MMD 0.0051 | Test Wasserstein 7.7371\n",
      "Epoch 140 | Train Loss 3065397.7500 | Train MMD 0.0063 | Test MMD 0.0047 | Test Wasserstein 7.7421\n",
      "Epoch 150 | Train Loss 3017445.0000 | Train MMD 0.0062 | Test MMD 0.0044 | Test Wasserstein 7.7483\n",
      "Epoch 160 | Train Loss 2971793.0000 | Train MMD 0.0062 | Test MMD 0.0042 | Test Wasserstein 7.7559\n",
      "Epoch 170 | Train Loss 2928289.7500 | Train MMD 0.0061 | Test MMD 0.0039 | Test Wasserstein 7.7648\n",
      "Epoch 180 | Train Loss 2886796.0000 | Train MMD 0.0061 | Test MMD 0.0037 | Test Wasserstein 7.7747\n",
      "Epoch 190 | Train Loss 2847184.2500 | Train MMD 0.0060 | Test MMD 0.0035 | Test Wasserstein 7.7857\n",
      "Epoch 200 | Train Loss 2809339.0000 | Train MMD 0.0060 | Test MMD 0.0033 | Test Wasserstein 7.7972\n",
      "Metrics: {'mmd2_gamma_median': 0.0033099483987888423, 'mmd2_gamma_0.5': 8.995338914230034e-05, 'mmd2_gamma_1.0': 4.204512537263809e-05, 'wasserstein_distance': 7.797176126532874, 'R2_feature_means': 0.8371523915229294}\n",
      "**************** Run: 9 ****************\n",
      "Epoch 0 | Train Loss 4142389.7500 | Train MMD 0.0110 | Test MMD 0.0132 | Test Wasserstein 8.0244\n",
      "Epoch 10 | Train Loss 4016080.0000 | Train MMD 0.0093 | Test MMD 0.0128 | Test Wasserstein 7.9222\n",
      "Epoch 20 | Train Loss 3908877.0000 | Train MMD 0.0087 | Test MMD 0.0120 | Test Wasserstein 7.8655\n",
      "Epoch 30 | Train Loss 3810793.0000 | Train MMD 0.0084 | Test MMD 0.0110 | Test Wasserstein 7.8190\n",
      "Epoch 40 | Train Loss 3720023.7500 | Train MMD 0.0081 | Test MMD 0.0101 | Test Wasserstein 7.7894\n",
      "Epoch 50 | Train Loss 3635463.2500 | Train MMD 0.0078 | Test MMD 0.0093 | Test Wasserstein 7.7704\n",
      "Epoch 60 | Train Loss 3556318.7500 | Train MMD 0.0075 | Test MMD 0.0086 | Test Wasserstein 7.7560\n",
      "Epoch 70 | Train Loss 3482020.7500 | Train MMD 0.0072 | Test MMD 0.0079 | Test Wasserstein 7.7445\n",
      "Epoch 80 | Train Loss 3412084.0000 | Train MMD 0.0070 | Test MMD 0.0073 | Test Wasserstein 7.7366\n",
      "Epoch 90 | Train Loss 3346111.2500 | Train MMD 0.0068 | Test MMD 0.0067 | Test Wasserstein 7.7322\n",
      "Epoch 100 | Train Loss 3283780.2500 | Train MMD 0.0067 | Test MMD 0.0062 | Test Wasserstein 7.7309\n",
      "Epoch 110 | Train Loss 3224795.7500 | Train MMD 0.0065 | Test MMD 0.0058 | Test Wasserstein 7.7316\n",
      "Epoch 120 | Train Loss 3168889.2500 | Train MMD 0.0064 | Test MMD 0.0054 | Test Wasserstein 7.7336\n",
      "Epoch 130 | Train Loss 3115818.7500 | Train MMD 0.0064 | Test MMD 0.0051 | Test Wasserstein 7.7371\n",
      "Epoch 140 | Train Loss 3065397.7500 | Train MMD 0.0063 | Test MMD 0.0047 | Test Wasserstein 7.7421\n",
      "Epoch 150 | Train Loss 3017445.0000 | Train MMD 0.0062 | Test MMD 0.0044 | Test Wasserstein 7.7483\n",
      "Epoch 160 | Train Loss 2971793.5000 | Train MMD 0.0062 | Test MMD 0.0042 | Test Wasserstein 7.7559\n",
      "Epoch 170 | Train Loss 2928289.7500 | Train MMD 0.0061 | Test MMD 0.0039 | Test Wasserstein 7.7648\n",
      "Epoch 180 | Train Loss 2886796.0000 | Train MMD 0.0061 | Test MMD 0.0037 | Test Wasserstein 7.7747\n",
      "Epoch 190 | Train Loss 2847184.2500 | Train MMD 0.0060 | Test MMD 0.0035 | Test Wasserstein 7.7857\n",
      "Epoch 200 | Train Loss 2809339.0000 | Train MMD 0.0060 | Test MMD 0.0033 | Test Wasserstein 7.7972\n",
      "Metrics: {'mmd2_gamma_median': 0.0033099483987888423, 'mmd2_gamma_0.5': 8.995338154287795e-05, 'mmd2_gamma_1.0': 4.204512347278248e-05, 'wasserstein_distance': 7.797176137862719, 'R2_feature_means': 0.8371523915229294}\n",
      "=== Metrics Summary over Runs for top 100 genes ===\n",
      "                        mean  std\n",
      "mmd2_gamma_median     0.0033  0.0\n",
      "mmd2_gamma_0.5        0.0001  0.0\n",
      "mmd2_gamma_1.0        0.0000  0.0\n",
      "wasserstein_distance  7.7972  0.0\n",
      "R2_feature_means      0.8372  0.0\n"
     ]
    }
   ],
   "source": [
    "# Multiple runs for robustness\n",
    "all_metrics = []\n",
    "for run in range(10):\n",
    "    print(f\"**************** Run: {run} ****************\")\n",
    "    seed = 1234 + run\n",
    "    torch.manual_seed(seed)\n",
    "    torch.cuda.manual_seed_all(seed)\n",
    "    random.seed(seed)\n",
    "    np.random.seed(seed)\n",
    "    torch.backends.cudnn.deterministic = True\n",
    "    torch.backends.cudnn.benchmark = False\n",
    "\n",
    "    out = AOT(X_tr_pre, Y_tr_post, X_te_pre, Y_te_post, n_epochs=200, batch_size=4096, lr=1e-4, device=\"cuda\", verbose=True, top_feature_subset=top_genes_idx, seed=seed)\n",
    "    metrics = summarize_metrics(out[\"y_pred\"][:, top_genes_idx], Y_te_post[:, top_genes_idx], median_gamma)\n",
    "    print(\"Metrics:\", metrics)\n",
    "    all_metrics.append(metrics)\n",
    "\n",
    "# Results summary\n",
    "df = pd.DataFrame(all_metrics)\n",
    "print(\"=== Metrics Summary over Runs for top 100 genes ===\")\n",
    "print(df.describe().T[['mean', 'std']].round(4))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "822b9837",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8fe1d90d",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c367fda4",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 54,
   "id": "b8815e20",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "X_pre cells: (17565, 1000)\n",
      "X_post cells: (4505, 1000)\n",
      "(3604, 1000)\n",
      "(901, 1000)\n",
      "(3604, 1000)\n",
      "(901, 1000)\n",
      "Median heuristic gamma: 0.0021515925828798567\n"
     ]
    }
   ],
   "source": [
    "drug = \"abexinostat\"\n",
    "X_pre = adata_sc[adata_sc.obs[\"drug\"] == \"control\"].copy().to_df()\n",
    "X_post  = adata_sc[adata_sc.obs[\"drug\"] == drug].copy().to_df()\n",
    "\n",
    "print(\"X_pre cells:\", X_pre.shape)\n",
    "print(\"X_post cells:\", X_post.shape)\n",
    "\n",
    "top_genes_ids, top_genes_short, top_genes_idx = topk_markers(adata_sc, drug, k=100)\n",
    "\n",
    "X_tr_pre, X_te_pre, Y_tr_post, Y_te_post = split_train_test(X_pre.values, X_post.values, 0.8)\n",
    "\n",
    "print(X_tr_pre.shape)\n",
    "print(X_te_pre.shape)\n",
    "print(Y_tr_post.shape)\n",
    "print(Y_te_post.shape)\n",
    "\n",
    "\n",
    "# Compute median heuristic gamma on training data\n",
    "median_gamma = median_heuristic_gamma(X_tr_pre, Y_tr_post)\n",
    "print(\"Median heuristic gamma:\", median_gamma)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 55,
   "id": "39f9f54b",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "**************** Run: 0 ****************\n",
      "Epoch 0 | Train Loss 5246983.0000 | Train MMD 0.0147 | Test MMD 0.0203 | Test Wasserstein 7.9843\n",
      "Epoch 10 | Train Loss 5085231.0000 | Train MMD 0.0132 | Test MMD 0.0185 | Test Wasserstein 7.8207\n",
      "Epoch 20 | Train Loss 4955467.5000 | Train MMD 0.0122 | Test MMD 0.0167 | Test Wasserstein 7.7213\n",
      "Epoch 30 | Train Loss 4839393.0000 | Train MMD 0.0113 | Test MMD 0.0149 | Test Wasserstein 7.6603\n",
      "Epoch 40 | Train Loss 4732824.5000 | Train MMD 0.0104 | Test MMD 0.0133 | Test Wasserstein 7.6215\n",
      "Epoch 50 | Train Loss 4634250.5000 | Train MMD 0.0097 | Test MMD 0.0120 | Test Wasserstein 7.5952\n",
      "Epoch 60 | Train Loss 4542612.5000 | Train MMD 0.0092 | Test MMD 0.0108 | Test Wasserstein 7.5771\n",
      "Epoch 70 | Train Loss 4457099.5000 | Train MMD 0.0088 | Test MMD 0.0098 | Test Wasserstein 7.5632\n",
      "Epoch 80 | Train Loss 4377058.5000 | Train MMD 0.0086 | Test MMD 0.0090 | Test Wasserstein 7.5536\n",
      "Epoch 90 | Train Loss 4301951.0000 | Train MMD 0.0084 | Test MMD 0.0082 | Test Wasserstein 7.5481\n",
      "Epoch 100 | Train Loss 4231320.5000 | Train MMD 0.0082 | Test MMD 0.0076 | Test Wasserstein 7.5454\n",
      "Epoch 110 | Train Loss 4164773.7500 | Train MMD 0.0081 | Test MMD 0.0070 | Test Wasserstein 7.5457\n",
      "Epoch 120 | Train Loss 4101971.7500 | Train MMD 0.0080 | Test MMD 0.0065 | Test Wasserstein 7.5482\n",
      "Epoch 130 | Train Loss 4042599.7500 | Train MMD 0.0079 | Test MMD 0.0061 | Test Wasserstein 7.5521\n",
      "Epoch 140 | Train Loss 3986400.7500 | Train MMD 0.0078 | Test MMD 0.0056 | Test Wasserstein 7.5576\n",
      "Epoch 150 | Train Loss 3933146.0000 | Train MMD 0.0077 | Test MMD 0.0053 | Test Wasserstein 7.5646\n",
      "Epoch 160 | Train Loss 3882621.0000 | Train MMD 0.0077 | Test MMD 0.0049 | Test Wasserstein 7.5728\n",
      "Epoch 170 | Train Loss 3834633.5000 | Train MMD 0.0076 | Test MMD 0.0046 | Test Wasserstein 7.5817\n",
      "Epoch 180 | Train Loss 3789007.7500 | Train MMD 0.0076 | Test MMD 0.0043 | Test Wasserstein 7.5916\n",
      "Epoch 190 | Train Loss 3745583.7500 | Train MMD 0.0075 | Test MMD 0.0041 | Test Wasserstein 7.6023\n",
      "Epoch 200 | Train Loss 3704218.0000 | Train MMD 0.0075 | Test MMD 0.0038 | Test Wasserstein 7.6137\n",
      "Metrics: {'mmd2_gamma_median': 0.0038440887541963953, 'mmd2_gamma_0.5': 1.2652594502630195e-05, 'mmd2_gamma_1.0': 2.357274195136815e-06, 'wasserstein_distance': 7.613657350247185, 'R2_feature_means': 0.8040116325616486}\n",
      "**************** Run: 1 ****************\n",
      "Epoch 0 | Train Loss 5246982.0000 | Train MMD 0.0147 | Test MMD 0.0203 | Test Wasserstein 7.9843\n",
      "Epoch 10 | Train Loss 5085231.0000 | Train MMD 0.0132 | Test MMD 0.0185 | Test Wasserstein 7.8207\n",
      "Epoch 20 | Train Loss 4955467.5000 | Train MMD 0.0122 | Test MMD 0.0167 | Test Wasserstein 7.7213\n",
      "Epoch 30 | Train Loss 4839393.0000 | Train MMD 0.0113 | Test MMD 0.0149 | Test Wasserstein 7.6603\n",
      "Epoch 40 | Train Loss 4732824.5000 | Train MMD 0.0104 | Test MMD 0.0133 | Test Wasserstein 7.6215\n",
      "Epoch 50 | Train Loss 4634249.5000 | Train MMD 0.0097 | Test MMD 0.0120 | Test Wasserstein 7.5952\n",
      "Epoch 60 | Train Loss 4542611.5000 | Train MMD 0.0092 | Test MMD 0.0108 | Test Wasserstein 7.5771\n",
      "Epoch 70 | Train Loss 4457099.5000 | Train MMD 0.0088 | Test MMD 0.0098 | Test Wasserstein 7.5632\n",
      "Epoch 80 | Train Loss 4377058.5000 | Train MMD 0.0086 | Test MMD 0.0090 | Test Wasserstein 7.5536\n",
      "Epoch 90 | Train Loss 4301951.0000 | Train MMD 0.0084 | Test MMD 0.0082 | Test Wasserstein 7.5481\n",
      "Epoch 100 | Train Loss 4231319.5000 | Train MMD 0.0082 | Test MMD 0.0076 | Test Wasserstein 7.5454\n",
      "Epoch 110 | Train Loss 4164774.2500 | Train MMD 0.0081 | Test MMD 0.0070 | Test Wasserstein 7.5457\n",
      "Epoch 120 | Train Loss 4101971.7500 | Train MMD 0.0080 | Test MMD 0.0065 | Test Wasserstein 7.5482\n",
      "Epoch 130 | Train Loss 4042599.7500 | Train MMD 0.0079 | Test MMD 0.0061 | Test Wasserstein 7.5521\n",
      "Epoch 140 | Train Loss 3986400.7500 | Train MMD 0.0078 | Test MMD 0.0056 | Test Wasserstein 7.5576\n",
      "Epoch 150 | Train Loss 3933145.5000 | Train MMD 0.0077 | Test MMD 0.0053 | Test Wasserstein 7.5646\n",
      "Epoch 160 | Train Loss 3882621.0000 | Train MMD 0.0077 | Test MMD 0.0049 | Test Wasserstein 7.5728\n",
      "Epoch 170 | Train Loss 3834633.0000 | Train MMD 0.0076 | Test MMD 0.0046 | Test Wasserstein 7.5817\n",
      "Epoch 180 | Train Loss 3789007.2500 | Train MMD 0.0076 | Test MMD 0.0043 | Test Wasserstein 7.5916\n",
      "Epoch 190 | Train Loss 3745583.7500 | Train MMD 0.0075 | Test MMD 0.0041 | Test Wasserstein 7.6023\n",
      "Epoch 200 | Train Loss 3704217.5000 | Train MMD 0.0075 | Test MMD 0.0038 | Test Wasserstein 7.6137\n",
      "Metrics: {'mmd2_gamma_median': 0.0038440887541963953, 'mmd2_gamma_0.5': 1.2652595565012312e-05, 'mmd2_gamma_1.0': 2.3572744856013404e-06, 'wasserstein_distance': 7.613657355793111, 'R2_feature_means': 0.8040115981959218}\n",
      "**************** Run: 2 ****************\n",
      "Epoch 0 | Train Loss 5246982.0000 | Train MMD 0.0147 | Test MMD 0.0203 | Test Wasserstein 7.9843\n",
      "Epoch 10 | Train Loss 5085231.0000 | Train MMD 0.0132 | Test MMD 0.0185 | Test Wasserstein 7.8207\n",
      "Epoch 20 | Train Loss 4955468.5000 | Train MMD 0.0122 | Test MMD 0.0167 | Test Wasserstein 7.7213\n",
      "Epoch 30 | Train Loss 4839393.0000 | Train MMD 0.0113 | Test MMD 0.0149 | Test Wasserstein 7.6603\n",
      "Epoch 40 | Train Loss 4732824.5000 | Train MMD 0.0104 | Test MMD 0.0133 | Test Wasserstein 7.6215\n",
      "Epoch 50 | Train Loss 4634250.5000 | Train MMD 0.0097 | Test MMD 0.0120 | Test Wasserstein 7.5952\n",
      "Epoch 60 | Train Loss 4542612.5000 | Train MMD 0.0092 | Test MMD 0.0108 | Test Wasserstein 7.5771\n",
      "Epoch 70 | Train Loss 4457099.5000 | Train MMD 0.0088 | Test MMD 0.0098 | Test Wasserstein 7.5632\n",
      "Epoch 80 | Train Loss 4377058.5000 | Train MMD 0.0086 | Test MMD 0.0090 | Test Wasserstein 7.5536\n",
      "Epoch 90 | Train Loss 4301951.0000 | Train MMD 0.0084 | Test MMD 0.0082 | Test Wasserstein 7.5481\n",
      "Epoch 100 | Train Loss 4231319.5000 | Train MMD 0.0082 | Test MMD 0.0076 | Test Wasserstein 7.5454\n",
      "Epoch 110 | Train Loss 4164774.2500 | Train MMD 0.0081 | Test MMD 0.0070 | Test Wasserstein 7.5457\n",
      "Epoch 120 | Train Loss 4101971.7500 | Train MMD 0.0080 | Test MMD 0.0065 | Test Wasserstein 7.5482\n",
      "Epoch 130 | Train Loss 4042599.2500 | Train MMD 0.0079 | Test MMD 0.0061 | Test Wasserstein 7.5521\n",
      "Epoch 140 | Train Loss 3986400.7500 | Train MMD 0.0078 | Test MMD 0.0056 | Test Wasserstein 7.5576\n",
      "Epoch 150 | Train Loss 3933145.5000 | Train MMD 0.0077 | Test MMD 0.0053 | Test Wasserstein 7.5646\n",
      "Epoch 160 | Train Loss 3882621.0000 | Train MMD 0.0077 | Test MMD 0.0049 | Test Wasserstein 7.5728\n",
      "Epoch 170 | Train Loss 3834633.0000 | Train MMD 0.0076 | Test MMD 0.0046 | Test Wasserstein 7.5817\n",
      "Epoch 180 | Train Loss 3789007.7500 | Train MMD 0.0076 | Test MMD 0.0043 | Test Wasserstein 7.5916\n",
      "Epoch 190 | Train Loss 3745583.7500 | Train MMD 0.0075 | Test MMD 0.0041 | Test Wasserstein 7.6023\n",
      "Epoch 200 | Train Loss 3704218.0000 | Train MMD 0.0075 | Test MMD 0.0038 | Test Wasserstein 7.6137\n",
      "Metrics: {'mmd2_gamma_median': 0.0038440887541963953, 'mmd2_gamma_0.5': 1.2652595451325474e-05, 'mmd2_gamma_1.0': 2.3572743341518287e-06, 'wasserstein_distance': 7.613657365782627, 'R2_feature_means': 0.8040115679920627}\n",
      "**************** Run: 3 ****************\n",
      "Epoch 0 | Train Loss 5246982.0000 | Train MMD 0.0147 | Test MMD 0.0203 | Test Wasserstein 7.9843\n",
      "Epoch 10 | Train Loss 5085231.0000 | Train MMD 0.0132 | Test MMD 0.0185 | Test Wasserstein 7.8207\n",
      "Epoch 20 | Train Loss 4955468.5000 | Train MMD 0.0122 | Test MMD 0.0167 | Test Wasserstein 7.7213\n",
      "Epoch 30 | Train Loss 4839393.0000 | Train MMD 0.0113 | Test MMD 0.0149 | Test Wasserstein 7.6603\n",
      "Epoch 40 | Train Loss 4732824.5000 | Train MMD 0.0104 | Test MMD 0.0133 | Test Wasserstein 7.6215\n",
      "Epoch 50 | Train Loss 4634250.5000 | Train MMD 0.0097 | Test MMD 0.0120 | Test Wasserstein 7.5952\n",
      "Epoch 60 | Train Loss 4542611.5000 | Train MMD 0.0092 | Test MMD 0.0108 | Test Wasserstein 7.5771\n",
      "Epoch 70 | Train Loss 4457099.5000 | Train MMD 0.0088 | Test MMD 0.0098 | Test Wasserstein 7.5632\n",
      "Epoch 80 | Train Loss 4377058.5000 | Train MMD 0.0086 | Test MMD 0.0090 | Test Wasserstein 7.5536\n",
      "Epoch 90 | Train Loss 4301951.0000 | Train MMD 0.0084 | Test MMD 0.0082 | Test Wasserstein 7.5481\n",
      "Epoch 100 | Train Loss 4231319.5000 | Train MMD 0.0082 | Test MMD 0.0076 | Test Wasserstein 7.5454\n",
      "Epoch 110 | Train Loss 4164773.7500 | Train MMD 0.0081 | Test MMD 0.0070 | Test Wasserstein 7.5457\n",
      "Epoch 120 | Train Loss 4101971.7500 | Train MMD 0.0080 | Test MMD 0.0065 | Test Wasserstein 7.5482\n",
      "Epoch 130 | Train Loss 4042599.7500 | Train MMD 0.0079 | Test MMD 0.0061 | Test Wasserstein 7.5521\n",
      "Epoch 140 | Train Loss 3986400.7500 | Train MMD 0.0078 | Test MMD 0.0056 | Test Wasserstein 7.5576\n",
      "Epoch 150 | Train Loss 3933146.0000 | Train MMD 0.0077 | Test MMD 0.0053 | Test Wasserstein 7.5646\n",
      "Epoch 160 | Train Loss 3882621.0000 | Train MMD 0.0077 | Test MMD 0.0049 | Test Wasserstein 7.5728\n",
      "Epoch 170 | Train Loss 3834633.0000 | Train MMD 0.0076 | Test MMD 0.0046 | Test Wasserstein 7.5817\n",
      "Epoch 180 | Train Loss 3789007.7500 | Train MMD 0.0076 | Test MMD 0.0043 | Test Wasserstein 7.5916\n",
      "Epoch 190 | Train Loss 3745583.7500 | Train MMD 0.0075 | Test MMD 0.0041 | Test Wasserstein 7.6023\n",
      "Epoch 200 | Train Loss 3704217.5000 | Train MMD 0.0075 | Test MMD 0.0038 | Test Wasserstein 7.6137\n",
      "Metrics: {'mmd2_gamma_median': 0.0038440887541963953, 'mmd2_gamma_0.5': 1.2652596741081269e-05, 'mmd2_gamma_1.0': 2.357275077188532e-06, 'wasserstein_distance': 7.613657359860124, 'R2_feature_means': 0.8040115830939935}\n",
      "**************** Run: 4 ****************\n",
      "Epoch 0 | Train Loss 5246983.0000 | Train MMD 0.0147 | Test MMD 0.0203 | Test Wasserstein 7.9843\n",
      "Epoch 10 | Train Loss 5085231.0000 | Train MMD 0.0132 | Test MMD 0.0185 | Test Wasserstein 7.8207\n",
      "Epoch 20 | Train Loss 4955468.5000 | Train MMD 0.0122 | Test MMD 0.0167 | Test Wasserstein 7.7213\n",
      "Epoch 30 | Train Loss 4839393.0000 | Train MMD 0.0113 | Test MMD 0.0149 | Test Wasserstein 7.6603\n",
      "Epoch 40 | Train Loss 4732824.5000 | Train MMD 0.0104 | Test MMD 0.0133 | Test Wasserstein 7.6215\n",
      "Epoch 50 | Train Loss 4634250.5000 | Train MMD 0.0097 | Test MMD 0.0120 | Test Wasserstein 7.5952\n",
      "Epoch 60 | Train Loss 4542611.5000 | Train MMD 0.0092 | Test MMD 0.0108 | Test Wasserstein 7.5771\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[55], line 13\u001b[0m\n\u001b[1;32m     10\u001b[0m torch\u001b[38;5;241m.\u001b[39mbackends\u001b[38;5;241m.\u001b[39mcudnn\u001b[38;5;241m.\u001b[39mdeterministic \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mTrue\u001b[39;00m\n\u001b[1;32m     11\u001b[0m torch\u001b[38;5;241m.\u001b[39mbackends\u001b[38;5;241m.\u001b[39mcudnn\u001b[38;5;241m.\u001b[39mbenchmark \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mFalse\u001b[39;00m\n\u001b[0;32m---> 13\u001b[0m out \u001b[38;5;241m=\u001b[39m \u001b[43mAOT\u001b[49m\u001b[43m(\u001b[49m\u001b[43mX_tr_pre\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mY_tr_post\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mX_te_pre\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mY_te_post\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mn_epochs\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m200\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mbatch_size\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m4096\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mlr\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m1e-4\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mdevice\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;124;43m\"\u001b[39;49m\u001b[38;5;124;43mcuda\u001b[39;49m\u001b[38;5;124;43m\"\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mverbose\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43;01mTrue\u001b[39;49;00m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mtop_feature_subset\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mtop_genes_idx\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mseed\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mseed\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     14\u001b[0m metrics \u001b[38;5;241m=\u001b[39m summarize_metrics(out[\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124my_pred\u001b[39m\u001b[38;5;124m\"\u001b[39m][:, top_genes_idx], Y_te_post[:, top_genes_idx], median_gamma)\n\u001b[1;32m     15\u001b[0m \u001b[38;5;28mprint\u001b[39m(\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mMetrics:\u001b[39m\u001b[38;5;124m\"\u001b[39m, metrics)\n",
      "Cell \u001b[0;32mIn[44], line 77\u001b[0m, in \u001b[0;36mAOT\u001b[0;34m(X_tr_pre, Y_tr_post, X_te_pre, Y_te_post, top_feature_subset, n_epochs, batch_size, lr, device, seed, verbose)\u001b[0m\n\u001b[1;32m     75\u001b[0m Xmap \u001b[38;5;241m=\u001b[39m mapper(bx)\n\u001b[1;32m     76\u001b[0m \u001b[38;5;28;01mwith\u001b[39;00m torch\u001b[38;5;241m.\u001b[39mno_grad():\n\u001b[0;32m---> 77\u001b[0m     P,_ \u001b[38;5;241m=\u001b[39m \u001b[43mexact_ot_plan\u001b[49m\u001b[43m(\u001b[49m\u001b[43mXmap\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mby\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     78\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28misinstance\u001b[39m(P, np\u001b[38;5;241m.\u001b[39mndarray):\n\u001b[1;32m     79\u001b[0m         P \u001b[38;5;241m=\u001b[39m torch\u001b[38;5;241m.\u001b[39mfrom_numpy(P)\u001b[38;5;241m.\u001b[39mto(device\u001b[38;5;241m=\u001b[39mdevice, dtype\u001b[38;5;241m=\u001b[39mtorch\u001b[38;5;241m.\u001b[39mfloat32)\n",
      "Cell \u001b[0;32mIn[44], line 29\u001b[0m, in \u001b[0;36mAOT.<locals>.exact_ot_plan\u001b[0;34m(X, Y)\u001b[0m\n\u001b[1;32m     27\u001b[0m a, b \u001b[38;5;241m=\u001b[39m torch\u001b[38;5;241m.\u001b[39mones(\u001b[38;5;28mlen\u001b[39m(X)), torch\u001b[38;5;241m.\u001b[39mones(\u001b[38;5;28mlen\u001b[39m(Y))\n\u001b[1;32m     28\u001b[0m C \u001b[38;5;241m=\u001b[39m cost_matrix(X, Y)\n\u001b[0;32m---> 29\u001b[0m Pi \u001b[38;5;241m=\u001b[39m \u001b[43mot\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43memd\u001b[49m\u001b[43m(\u001b[49m\u001b[43ma\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mb\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mC\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mnumItermax\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m1e7\u001b[39;49m\u001b[43m)\u001b[49m\n\u001b[1;32m     30\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m Pi, C\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/site-packages/ot/lp/_network_simplex.py:324\u001b[0m, in \u001b[0;36memd\u001b[0;34m(a, b, M, numItermax, log, center_dual, numThreads, check_marginals)\u001b[0m\n\u001b[1;32m    320\u001b[0m bsel \u001b[38;5;241m=\u001b[39m b \u001b[38;5;241m!=\u001b[39m \u001b[38;5;241m0\u001b[39m\n\u001b[1;32m    322\u001b[0m numThreads \u001b[38;5;241m=\u001b[39m check_number_threads(numThreads)\n\u001b[0;32m--> 324\u001b[0m G, cost, u, v, result_code \u001b[38;5;241m=\u001b[39m \u001b[43memd_c\u001b[49m\u001b[43m(\u001b[49m\u001b[43ma\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mb\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mM\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mnumItermax\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mnumThreads\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    326\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m center_dual:\n\u001b[1;32m    327\u001b[0m     u, v \u001b[38;5;241m=\u001b[39m center_ot_dual(u, v, a, b)\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": [
    "# Multiple runs for robustness\n",
    "all_metrics = []\n",
    "for run in range(10):\n",
    "    print(f\"**************** Run: {run} ****************\")\n",
    "    seed = 1234 + run\n",
    "    torch.manual_seed(seed)\n",
    "    torch.cuda.manual_seed_all(seed)\n",
    "    random.seed(seed)\n",
    "    np.random.seed(seed)\n",
    "    torch.backends.cudnn.deterministic = True\n",
    "    torch.backends.cudnn.benchmark = False\n",
    "\n",
    "    out = AOT(X_tr_pre, Y_tr_post, X_te_pre, Y_te_post, n_epochs=200, batch_size=4096, lr=1e-4, device=\"cuda\", verbose=True, top_feature_subset=top_genes_idx, seed=seed)\n",
    "    metrics = summarize_metrics(out[\"y_pred\"][:, top_genes_idx], Y_te_post[:, top_genes_idx], median_gamma)\n",
    "    print(\"Metrics:\", metrics)\n",
    "    all_metrics.append(metrics)\n",
    "\n",
    "# Results summary\n",
    "df = pd.DataFrame(all_metrics)\n",
    "print(\"=== Metrics Summary over Runs for top 100 genes ===\")\n",
    "print(df.describe().T[['mean', 'std']].round(4))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 59,
   "id": "7540bbd9",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "**************** Run: 0 ****************\n",
      "Epoch 0 | Train Loss 426154.1250 | Train MMD 0.0189 | Test MMD 0.0196 | Test Wasserstein 7.9598\n",
      "Epoch 10 | Train Loss 391959.4375 | Train MMD 0.0118 | Test MMD 0.0125 | Test Wasserstein 7.6411\n",
      "Epoch 20 | Train Loss 374228.6562 | Train MMD 0.0073 | Test MMD 0.0079 | Test Wasserstein 7.4776\n",
      "Epoch 30 | Train Loss 362087.1875 | Train MMD 0.0046 | Test MMD 0.0051 | Test Wasserstein 7.3812\n",
      "Epoch 40 | Train Loss 353254.2812 | Train MMD 0.0031 | Test MMD 0.0036 | Test Wasserstein 7.3146\n",
      "Epoch 50 | Train Loss 346384.0625 | Train MMD 0.0024 | Test MMD 0.0028 | Test Wasserstein 7.2617\n",
      "Epoch 60 | Train Loss 340754.5938 | Train MMD 0.0020 | Test MMD 0.0024 | Test Wasserstein 7.2185\n",
      "Epoch 70 | Train Loss 335922.9688 | Train MMD 0.0019 | Test MMD 0.0022 | Test Wasserstein 7.1821\n",
      "Epoch 80 | Train Loss 331653.5312 | Train MMD 0.0018 | Test MMD 0.0021 | Test Wasserstein 7.1495\n",
      "Epoch 90 | Train Loss 327840.7500 | Train MMD 0.0017 | Test MMD 0.0020 | Test Wasserstein 7.1185\n",
      "Epoch 100 | Train Loss 324376.5625 | Train MMD 0.0017 | Test MMD 0.0020 | Test Wasserstein 7.0904\n",
      "Epoch 110 | Train Loss 321247.2812 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 7.0632\n",
      "Epoch 120 | Train Loss 318325.7188 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 7.0370\n",
      "Epoch 130 | Train Loss 315594.4688 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 7.0120\n",
      "Epoch 140 | Train Loss 313050.2188 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 6.9891\n",
      "Epoch 150 | Train Loss 310691.5312 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 6.9665\n",
      "Epoch 160 | Train Loss 308462.3438 | Train MMD 0.0018 | Test MMD 0.0019 | Test Wasserstein 6.9441\n",
      "Epoch 170 | Train Loss 306356.2500 | Train MMD 0.0018 | Test MMD 0.0019 | Test Wasserstein 6.9226\n",
      "Epoch 180 | Train Loss 304249.2812 | Train MMD 0.0018 | Test MMD 0.0019 | Test Wasserstein 6.9021\n",
      "Epoch 190 | Train Loss 302211.1562 | Train MMD 0.0018 | Test MMD 0.0019 | Test Wasserstein 6.8814\n",
      "Epoch 200 | Train Loss 300303.1250 | Train MMD 0.0019 | Test MMD 0.0020 | Test Wasserstein 6.8613\n",
      "Metrics: {'mmd2_gamma_median': 0.0019529186769307039, 'mmd2_gamma_0.5': 5.979974738608369e-05, 'mmd2_gamma_1.0': 2.8911502820656756e-05, 'wasserstein_distance': 6.861321397191977, 'R2_feature_means': 0.9599339275963331}\n",
      "**************** Run: 1 ****************\n",
      "Epoch 0 | Train Loss 426154.1250 | Train MMD 0.0189 | Test MMD 0.0196 | Test Wasserstein 7.9598\n",
      "Epoch 10 | Train Loss 391959.3750 | Train MMD 0.0118 | Test MMD 0.0125 | Test Wasserstein 7.6411\n",
      "Epoch 20 | Train Loss 374228.5938 | Train MMD 0.0073 | Test MMD 0.0079 | Test Wasserstein 7.4776\n",
      "Epoch 30 | Train Loss 362086.9688 | Train MMD 0.0046 | Test MMD 0.0051 | Test Wasserstein 7.3812\n",
      "Epoch 40 | Train Loss 353254.9375 | Train MMD 0.0031 | Test MMD 0.0036 | Test Wasserstein 7.3147\n",
      "Epoch 50 | Train Loss 346387.5625 | Train MMD 0.0024 | Test MMD 0.0028 | Test Wasserstein 7.2618\n",
      "Epoch 60 | Train Loss 340757.2188 | Train MMD 0.0020 | Test MMD 0.0024 | Test Wasserstein 7.2185\n",
      "Epoch 70 | Train Loss 335920.8438 | Train MMD 0.0019 | Test MMD 0.0022 | Test Wasserstein 7.1823\n",
      "Epoch 80 | Train Loss 331633.7812 | Train MMD 0.0018 | Test MMD 0.0021 | Test Wasserstein 7.1495\n",
      "Epoch 90 | Train Loss 327793.7812 | Train MMD 0.0017 | Test MMD 0.0020 | Test Wasserstein 7.1186\n",
      "Epoch 100 | Train Loss 324312.5938 | Train MMD 0.0017 | Test MMD 0.0020 | Test Wasserstein 7.0895\n",
      "Epoch 110 | Train Loss 321163.1562 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 7.0624\n",
      "Epoch 120 | Train Loss 318267.1250 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 7.0366\n",
      "Epoch 130 | Train Loss 315557.1562 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 7.0119\n",
      "Epoch 140 | Train Loss 313047.2188 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 6.9884\n",
      "Epoch 150 | Train Loss 310697.5625 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 6.9655\n",
      "Epoch 160 | Train Loss 308491.2812 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 6.9435\n",
      "Epoch 170 | Train Loss 306381.4375 | Train MMD 0.0018 | Test MMD 0.0019 | Test Wasserstein 6.9221\n",
      "Epoch 180 | Train Loss 304325.1562 | Train MMD 0.0018 | Test MMD 0.0019 | Test Wasserstein 6.9009\n",
      "Epoch 190 | Train Loss 302367.1562 | Train MMD 0.0018 | Test MMD 0.0019 | Test Wasserstein 6.8805\n",
      "Epoch 200 | Train Loss 300493.8125 | Train MMD 0.0019 | Test MMD 0.0020 | Test Wasserstein 6.8605\n",
      "Metrics: {'mmd2_gamma_median': 0.0019504893624338049, 'mmd2_gamma_0.5': 5.976645766888472e-05, 'mmd2_gamma_1.0': 2.8918505655532904e-05, 'wasserstein_distance': 6.860460610398516, 'R2_feature_means': 0.9599038810655918}\n",
      "**************** Run: 2 ****************\n",
      "Epoch 0 | Train Loss 426154.1250 | Train MMD 0.0189 | Test MMD 0.0196 | Test Wasserstein 7.9598\n",
      "Epoch 10 | Train Loss 391959.4375 | Train MMD 0.0118 | Test MMD 0.0125 | Test Wasserstein 7.6411\n",
      "Epoch 20 | Train Loss 374228.6562 | Train MMD 0.0073 | Test MMD 0.0079 | Test Wasserstein 7.4776\n",
      "Epoch 30 | Train Loss 362087.1875 | Train MMD 0.0046 | Test MMD 0.0051 | Test Wasserstein 7.3812\n",
      "Epoch 40 | Train Loss 353254.2188 | Train MMD 0.0031 | Test MMD 0.0036 | Test Wasserstein 7.3146\n",
      "Epoch 50 | Train Loss 346384.1250 | Train MMD 0.0024 | Test MMD 0.0028 | Test Wasserstein 7.2617\n",
      "Epoch 60 | Train Loss 340752.9375 | Train MMD 0.0020 | Test MMD 0.0024 | Test Wasserstein 7.2185\n",
      "Epoch 70 | Train Loss 335918.5938 | Train MMD 0.0019 | Test MMD 0.0022 | Test Wasserstein 7.1821\n",
      "Epoch 80 | Train Loss 331638.4062 | Train MMD 0.0018 | Test MMD 0.0021 | Test Wasserstein 7.1499\n",
      "Epoch 90 | Train Loss 327810.6875 | Train MMD 0.0017 | Test MMD 0.0020 | Test Wasserstein 7.1188\n",
      "Epoch 100 | Train Loss 324311.2812 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 7.0904\n",
      "Epoch 110 | Train Loss 321171.8750 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 7.0633\n",
      "Epoch 120 | Train Loss 318278.6250 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 7.0375\n",
      "Epoch 130 | Train Loss 315544.6250 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 7.0121\n",
      "Epoch 140 | Train Loss 312992.3125 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 6.9880\n",
      "Epoch 150 | Train Loss 310619.6875 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 6.9654\n",
      "Epoch 160 | Train Loss 308378.1562 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 6.9435\n",
      "Epoch 170 | Train Loss 306235.7812 | Train MMD 0.0018 | Test MMD 0.0019 | Test Wasserstein 6.9222\n",
      "Epoch 180 | Train Loss 304189.1562 | Train MMD 0.0018 | Test MMD 0.0019 | Test Wasserstein 6.9014\n",
      "Epoch 190 | Train Loss 302191.9062 | Train MMD 0.0018 | Test MMD 0.0019 | Test Wasserstein 6.8810\n",
      "Epoch 200 | Train Loss 300285.6562 | Train MMD 0.0019 | Test MMD 0.0019 | Test Wasserstein 6.8607\n",
      "Metrics: {'mmd2_gamma_median': 0.0019479098680472706, 'mmd2_gamma_0.5': 5.9949542838515124e-05, 'mmd2_gamma_1.0': 2.892457100878446e-05, 'wasserstein_distance': 6.860662509956777, 'R2_feature_means': 0.9600142342197169}\n",
      "**************** Run: 3 ****************\n",
      "Epoch 0 | Train Loss 426154.1250 | Train MMD 0.0189 | Test MMD 0.0196 | Test Wasserstein 7.9598\n",
      "Epoch 10 | Train Loss 391959.3750 | Train MMD 0.0118 | Test MMD 0.0125 | Test Wasserstein 7.6411\n",
      "Epoch 20 | Train Loss 374228.5938 | Train MMD 0.0073 | Test MMD 0.0079 | Test Wasserstein 7.4776\n",
      "Epoch 30 | Train Loss 362086.9688 | Train MMD 0.0046 | Test MMD 0.0051 | Test Wasserstein 7.3812\n",
      "Epoch 40 | Train Loss 353254.9375 | Train MMD 0.0031 | Test MMD 0.0036 | Test Wasserstein 7.3147\n",
      "Epoch 50 | Train Loss 346387.5625 | Train MMD 0.0024 | Test MMD 0.0028 | Test Wasserstein 7.2618\n",
      "Epoch 60 | Train Loss 340757.2188 | Train MMD 0.0020 | Test MMD 0.0024 | Test Wasserstein 7.2185\n",
      "Epoch 70 | Train Loss 335920.8438 | Train MMD 0.0019 | Test MMD 0.0022 | Test Wasserstein 7.1823\n",
      "Epoch 80 | Train Loss 331633.6875 | Train MMD 0.0018 | Test MMD 0.0021 | Test Wasserstein 7.1495\n",
      "Epoch 90 | Train Loss 327793.7812 | Train MMD 0.0017 | Test MMD 0.0020 | Test Wasserstein 7.1186\n",
      "Epoch 100 | Train Loss 324312.5312 | Train MMD 0.0017 | Test MMD 0.0020 | Test Wasserstein 7.0895\n",
      "Epoch 110 | Train Loss 321163.1562 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 7.0624\n",
      "Epoch 120 | Train Loss 318267.1250 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 7.0366\n",
      "Epoch 130 | Train Loss 315557.1562 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 7.0119\n",
      "Epoch 140 | Train Loss 313047.2188 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 6.9884\n",
      "Epoch 150 | Train Loss 310697.5625 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 6.9655\n",
      "Epoch 160 | Train Loss 308491.3438 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 6.9435\n",
      "Epoch 170 | Train Loss 306381.4375 | Train MMD 0.0018 | Test MMD 0.0019 | Test Wasserstein 6.9221\n",
      "Epoch 180 | Train Loss 304325.1562 | Train MMD 0.0018 | Test MMD 0.0019 | Test Wasserstein 6.9009\n",
      "Epoch 190 | Train Loss 302367.1562 | Train MMD 0.0018 | Test MMD 0.0019 | Test Wasserstein 6.8805\n",
      "Epoch 200 | Train Loss 300493.8125 | Train MMD 0.0019 | Test MMD 0.0020 | Test Wasserstein 6.8605\n",
      "Metrics: {'mmd2_gamma_median': 0.0019506085717233557, 'mmd2_gamma_0.5': 5.9766463282655237e-05, 'mmd2_gamma_1.0': 2.8918508007670815e-05, 'wasserstein_distance': 6.860460606954845, 'R2_feature_means': 0.9599038756171684}\n",
      "**************** Run: 4 ****************\n",
      "Epoch 0 | Train Loss 426154.1250 | Train MMD 0.0189 | Test MMD 0.0196 | Test Wasserstein 7.9598\n",
      "Epoch 10 | Train Loss 391959.3750 | Train MMD 0.0118 | Test MMD 0.0125 | Test Wasserstein 7.6411\n",
      "Epoch 20 | Train Loss 374228.5000 | Train MMD 0.0073 | Test MMD 0.0079 | Test Wasserstein 7.4776\n",
      "Epoch 30 | Train Loss 362086.9688 | Train MMD 0.0046 | Test MMD 0.0051 | Test Wasserstein 7.3812\n",
      "Epoch 40 | Train Loss 353254.9375 | Train MMD 0.0031 | Test MMD 0.0036 | Test Wasserstein 7.3147\n",
      "Epoch 50 | Train Loss 346387.5625 | Train MMD 0.0024 | Test MMD 0.0028 | Test Wasserstein 7.2618\n",
      "Epoch 60 | Train Loss 340757.2188 | Train MMD 0.0020 | Test MMD 0.0024 | Test Wasserstein 7.2185\n",
      "Epoch 70 | Train Loss 335920.8438 | Train MMD 0.0019 | Test MMD 0.0022 | Test Wasserstein 7.1823\n",
      "Epoch 80 | Train Loss 331633.7812 | Train MMD 0.0018 | Test MMD 0.0021 | Test Wasserstein 7.1495\n",
      "Epoch 90 | Train Loss 327793.7812 | Train MMD 0.0017 | Test MMD 0.0020 | Test Wasserstein 7.1186\n",
      "Epoch 100 | Train Loss 324312.5938 | Train MMD 0.0017 | Test MMD 0.0020 | Test Wasserstein 7.0895\n",
      "Epoch 110 | Train Loss 321163.1562 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 7.0624\n",
      "Epoch 120 | Train Loss 318267.1250 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 7.0366\n",
      "Epoch 130 | Train Loss 315557.1562 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 7.0119\n",
      "Epoch 140 | Train Loss 313047.2188 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 6.9884\n",
      "Epoch 150 | Train Loss 310697.5625 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 6.9655\n",
      "Epoch 160 | Train Loss 308491.3438 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 6.9435\n",
      "Epoch 170 | Train Loss 306381.4375 | Train MMD 0.0018 | Test MMD 0.0019 | Test Wasserstein 6.9221\n",
      "Epoch 180 | Train Loss 304325.1562 | Train MMD 0.0018 | Test MMD 0.0019 | Test Wasserstein 6.9009\n",
      "Epoch 190 | Train Loss 302367.1562 | Train MMD 0.0018 | Test MMD 0.0019 | Test Wasserstein 6.8805\n",
      "Epoch 200 | Train Loss 300493.8125 | Train MMD 0.0019 | Test MMD 0.0020 | Test Wasserstein 6.8605\n",
      "Metrics: {'mmd2_gamma_median': 0.0019504893624338049, 'mmd2_gamma_0.5': 5.976646146366583e-05, 'mmd2_gamma_1.0': 2.8918503189708156e-05, 'wasserstein_distance': 6.860460530251622, 'R2_feature_means': 0.9599038938047205}\n",
      "**************** Run: 5 ****************\n",
      "Epoch 0 | Train Loss 426154.1250 | Train MMD 0.0189 | Test MMD 0.0196 | Test Wasserstein 7.9598\n",
      "Epoch 10 | Train Loss 391959.3750 | Train MMD 0.0118 | Test MMD 0.0125 | Test Wasserstein 7.6411\n",
      "Epoch 20 | Train Loss 374228.5938 | Train MMD 0.0073 | Test MMD 0.0079 | Test Wasserstein 7.4776\n",
      "Epoch 30 | Train Loss 362086.9688 | Train MMD 0.0046 | Test MMD 0.0051 | Test Wasserstein 7.3812\n",
      "Epoch 40 | Train Loss 353254.9375 | Train MMD 0.0031 | Test MMD 0.0036 | Test Wasserstein 7.3147\n",
      "Epoch 50 | Train Loss 346387.5625 | Train MMD 0.0024 | Test MMD 0.0028 | Test Wasserstein 7.2618\n",
      "Epoch 60 | Train Loss 340757.2188 | Train MMD 0.0020 | Test MMD 0.0024 | Test Wasserstein 7.2185\n",
      "Epoch 70 | Train Loss 335920.8438 | Train MMD 0.0019 | Test MMD 0.0022 | Test Wasserstein 7.1823\n",
      "Epoch 80 | Train Loss 331633.7812 | Train MMD 0.0018 | Test MMD 0.0021 | Test Wasserstein 7.1495\n",
      "Epoch 90 | Train Loss 327793.8438 | Train MMD 0.0017 | Test MMD 0.0020 | Test Wasserstein 7.1186\n",
      "Epoch 100 | Train Loss 324312.5312 | Train MMD 0.0017 | Test MMD 0.0020 | Test Wasserstein 7.0895\n",
      "Epoch 110 | Train Loss 321163.1562 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 7.0624\n",
      "Epoch 120 | Train Loss 318267.1250 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 7.0366\n",
      "Epoch 130 | Train Loss 315557.1562 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 7.0119\n",
      "Epoch 140 | Train Loss 313047.2188 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 6.9884\n",
      "Epoch 150 | Train Loss 310697.5625 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 6.9655\n",
      "Epoch 160 | Train Loss 308491.3438 | Train MMD 0.0017 | Test MMD 0.0019 | Test Wasserstein 6.9435\n",
      "Epoch 170 | Train Loss 306381.4375 | Train MMD 0.0018 | Test MMD 0.0019 | Test Wasserstein 6.9221\n",
      "Epoch 180 | Train Loss 304325.1562 | Train MMD 0.0018 | Test MMD 0.0019 | Test Wasserstein 6.9009\n",
      "Epoch 190 | Train Loss 302367.1562 | Train MMD 0.0018 | Test MMD 0.0019 | Test Wasserstein 6.8805\n",
      "Epoch 200 | Train Loss 300493.8125 | Train MMD 0.0019 | Test MMD 0.0020 | Test Wasserstein 6.8605\n",
      "Metrics: {'mmd2_gamma_median': 0.0019504893624338049, 'mmd2_gamma_0.5': 5.976646146366583e-05, 'mmd2_gamma_1.0': 2.8918505485002647e-05, 'wasserstein_distance': 6.860460541219263, 'R2_feature_means': 0.9599038868341333}\n",
      "**************** Run: 6 ****************\n",
      "Epoch 0 | Train Loss 426154.1250 | Train MMD 0.0189 | Test MMD 0.0196 | Test Wasserstein 7.9598\n",
      "Epoch 10 | Train Loss 391959.3750 | Train MMD 0.0118 | Test MMD 0.0125 | Test Wasserstein 7.6411\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[59], line 13\u001b[0m\n\u001b[1;32m     10\u001b[0m torch\u001b[38;5;241m.\u001b[39mbackends\u001b[38;5;241m.\u001b[39mcudnn\u001b[38;5;241m.\u001b[39mdeterministic \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mTrue\u001b[39;00m\n\u001b[1;32m     11\u001b[0m torch\u001b[38;5;241m.\u001b[39mbackends\u001b[38;5;241m.\u001b[39mcudnn\u001b[38;5;241m.\u001b[39mbenchmark \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mFalse\u001b[39;00m\n\u001b[0;32m---> 13\u001b[0m out \u001b[38;5;241m=\u001b[39m \u001b[43mAOT\u001b[49m\u001b[43m(\u001b[49m\u001b[43mX_tr_pre\u001b[49m\u001b[43m[\u001b[49m\u001b[43m:\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mtop_genes_idx\u001b[49m\u001b[43m]\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mY_tr_post\u001b[49m\u001b[43m[\u001b[49m\u001b[43m:\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mtop_genes_idx\u001b[49m\u001b[43m]\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mX_te_pre\u001b[49m\u001b[43m[\u001b[49m\u001b[43m:\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mtop_genes_idx\u001b[49m\u001b[43m]\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mY_te_post\u001b[49m\u001b[43m[\u001b[49m\u001b[43m:\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mtop_genes_idx\u001b[49m\u001b[43m]\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mn_epochs\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m200\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mbatch_size\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m4096\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mlr\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m1e-3\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mdevice\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;124;43m\"\u001b[39;49m\u001b[38;5;124;43mcuda\u001b[39;49m\u001b[38;5;124;43m\"\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mverbose\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43;01mTrue\u001b[39;49;00m\u001b[43m,\u001b[49m\u001b[43m  \u001b[49m\u001b[43mseed\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mseed\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     14\u001b[0m metrics \u001b[38;5;241m=\u001b[39m summarize_metrics(out[\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124my_pred\u001b[39m\u001b[38;5;124m\"\u001b[39m][:, ], Y_te_post[:, top_genes_idx], median_gamma)\n\u001b[1;32m     15\u001b[0m \u001b[38;5;28mprint\u001b[39m(\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mMetrics:\u001b[39m\u001b[38;5;124m\"\u001b[39m, metrics)\n",
      "Cell \u001b[0;32mIn[57], line 77\u001b[0m, in \u001b[0;36mAOT\u001b[0;34m(X_tr_pre, Y_tr_post, X_te_pre, Y_te_post, top_feature_subset, n_epochs, batch_size, lr, device, seed, verbose)\u001b[0m\n\u001b[1;32m     75\u001b[0m Xmap \u001b[38;5;241m=\u001b[39m mapper(bx)\n\u001b[1;32m     76\u001b[0m \u001b[38;5;28;01mwith\u001b[39;00m torch\u001b[38;5;241m.\u001b[39mno_grad():\n\u001b[0;32m---> 77\u001b[0m     P,_ \u001b[38;5;241m=\u001b[39m \u001b[43mexact_ot_plan\u001b[49m\u001b[43m(\u001b[49m\u001b[43mXmap\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mby\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     78\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28misinstance\u001b[39m(P, np\u001b[38;5;241m.\u001b[39mndarray):\n\u001b[1;32m     79\u001b[0m         P \u001b[38;5;241m=\u001b[39m torch\u001b[38;5;241m.\u001b[39mfrom_numpy(P)\u001b[38;5;241m.\u001b[39mto(device\u001b[38;5;241m=\u001b[39mdevice, dtype\u001b[38;5;241m=\u001b[39mtorch\u001b[38;5;241m.\u001b[39mfloat32)\n",
      "Cell \u001b[0;32mIn[57], line 29\u001b[0m, in \u001b[0;36mAOT.<locals>.exact_ot_plan\u001b[0;34m(X, Y)\u001b[0m\n\u001b[1;32m     27\u001b[0m a, b \u001b[38;5;241m=\u001b[39m torch\u001b[38;5;241m.\u001b[39mones(\u001b[38;5;28mlen\u001b[39m(X)), torch\u001b[38;5;241m.\u001b[39mones(\u001b[38;5;28mlen\u001b[39m(Y))\n\u001b[1;32m     28\u001b[0m C \u001b[38;5;241m=\u001b[39m cost_matrix(X, Y)\n\u001b[0;32m---> 29\u001b[0m Pi \u001b[38;5;241m=\u001b[39m \u001b[43mot\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43memd\u001b[49m\u001b[43m(\u001b[49m\u001b[43ma\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mb\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mC\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mnumItermax\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m1e7\u001b[39;49m\u001b[43m)\u001b[49m\n\u001b[1;32m     30\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m Pi, C\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/site-packages/ot/lp/_network_simplex.py:324\u001b[0m, in \u001b[0;36memd\u001b[0;34m(a, b, M, numItermax, log, center_dual, numThreads, check_marginals)\u001b[0m\n\u001b[1;32m    320\u001b[0m bsel \u001b[38;5;241m=\u001b[39m b \u001b[38;5;241m!=\u001b[39m \u001b[38;5;241m0\u001b[39m\n\u001b[1;32m    322\u001b[0m numThreads \u001b[38;5;241m=\u001b[39m check_number_threads(numThreads)\n\u001b[0;32m--> 324\u001b[0m G, cost, u, v, result_code \u001b[38;5;241m=\u001b[39m \u001b[43memd_c\u001b[49m\u001b[43m(\u001b[49m\u001b[43ma\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mb\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mM\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mnumItermax\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mnumThreads\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    326\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m center_dual:\n\u001b[1;32m    327\u001b[0m     u, v \u001b[38;5;241m=\u001b[39m center_ot_dual(u, v, a, b)\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": [
    "# Multiple runs for robustness\n",
    "all_metrics = []\n",
    "for run in range(10):\n",
    "    print(f\"**************** Run: {run} ****************\")\n",
    "    seed = 1234 + run\n",
    "    torch.manual_seed(seed)\n",
    "    torch.cuda.manual_seed_all(seed)\n",
    "    random.seed(seed)\n",
    "    np.random.seed(seed)\n",
    "    torch.backends.cudnn.deterministic = True\n",
    "    torch.backends.cudnn.benchmark = False\n",
    "\n",
    "    out = AOT(X_tr_pre[:, top_genes_idx], Y_tr_post[:, top_genes_idx], X_te_pre[:, top_genes_idx], Y_te_post[:, top_genes_idx], n_epochs=200, batch_size=4096, lr=1e-3, device=\"cuda\", verbose=True,  seed=seed)\n",
    "    metrics = summarize_metrics(out[\"y_pred\"][:, ], Y_te_post[:, top_genes_idx], median_gamma)\n",
    "    print(\"Metrics:\", metrics)\n",
    "    all_metrics.append(metrics)\n",
    "\n",
    "# Results summary\n",
    "df = pd.DataFrame(all_metrics)\n",
    "print(\"=== Metrics Summary over Runs for top 100 genes ===\")\n",
    "print(df.describe().T[['mean', 'std']].round(4))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "40a869ac",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
