{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b9e2d729",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "True\n",
      "11.7\n",
      "NVIDIA RTX 6000 Ada Generation\n",
      "2\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "48"
      ]
     },
     "execution_count": 1,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import torch\n",
    "print(torch.cuda.is_available())   # should be True\n",
    "print(torch.version.cuda)          # CUDA version PyTorch is built with\n",
    "print(torch.cuda.get_device_name(0)) if torch.cuda.is_available() else None\n",
    "print(torch.cuda.device_count())\n",
    "import os\n",
    "os.cpu_count()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "fe4f5248",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "from typing import Dict, Tuple, List, Optional\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import logging\n",
    "logger = logging.getLogger(__name__)\n",
    "\n",
    "def load_ap1_data_from_csv(csv_filepath: str, replicate: Optional[int] = None) -> Dict[str, np.ndarray]:\n",
    "    \"\"\"\n",
    "    Loads AP1 single-cell data from CSV or Excel file.\n",
    "\n",
    "    Args:\n",
    "        csv_filepath: Path to the CSV or Excel file\n",
    "\n",
    "    Returns:\n",
    "        Dictionary with condition identifiers as keys and feature matrices as values\n",
    "    \"\"\"\n",
    "    logger.info(f\"Loading data from: {csv_filepath}\")\n",
    "\n",
    "    # Load the data based on file extension\n",
    "    if csv_filepath.endswith('.csv'):\n",
    "        df = pd.read_csv(csv_filepath)\n",
    "    elif csv_filepath.endswith('.xlsx'):\n",
    "        df = pd.read_excel(csv_filepath)\n",
    "    else:\n",
    "        raise ValueError(\"Unsupported file format. Please provide a .csv or .xlsx file.\")\n",
    "    \n",
    "    replacement_map = {\n",
    "                        '0.316 uM Vemurafenib': 'Vem',\n",
    "                        '0.316 uM Vem + 0.0316 uM Tram': 'Vem+Tram'\n",
    "                        }\n",
    "    df['condition'] = df['condition'].replace(replacement_map)\n",
    "\n",
    "    print(df['condition'].unique())\n",
    "\n",
    "    # Define AP1 protein features (these are in log space already)\n",
    "    ap1_features = [\n",
    "        'cFOS (log a.u.)', 'p-cFOS (log a.u.)', 'FRA1 (log a.u.)', 'p-FRA1 (log a.u.)', 'FRA2 (log a.u.)',\n",
    "        'cJUN (log a.u.)', 'p-cJUN (log a.u.)', 'JUNB (log a.u.)', 'JUND (log a.u.)', 'p-ATF1 (log a.u.)',\n",
    "        'ATF2 (log a.u.)',\t 'p-ATF2 (log a.u.)', 'ATF3 (log a.u.)', 'ATF4 (log a.u.)', 'p-ATF4 (log a.u.)',\n",
    "        'ATF5 (log a.u.)', 'ATF6 (log a.u.)', 'MITF (log a.u.)', 'NGFR (log a.u.)', 'p-ERK (log a.u.)',\n",
    "    ]\n",
    "\n",
    "    # Check if all features exist\n",
    "    missing_features = [f for f in ap1_features if f not in df.columns]\n",
    "    if missing_features:\n",
    "        logger.warning(f\"Missing features: {missing_features}\")\n",
    "        ap1_features = [f for f in ap1_features if f in df.columns]\n",
    "\n",
    "    logger.info(f\"Using {len(ap1_features)} AP1 features\")\n",
    "\n",
    "    # Create condition-based data dictionary\n",
    "    data_dict = {}\n",
    "\n",
    "    if replicate is not None:\n",
    "        # Group by condition, time, and cell line\n",
    "        for (condition, time, cell_line, replicate_id), group in df.groupby(['condition', 'time', 'cell_line', 'replicate_id']):\n",
    "            # Create condition identifier\n",
    "            condition_id = f\"{cell_line}_{condition}_{time.replace(' ', '')}_rep{replicate_id}\"\n",
    "\n",
    "            # Extract feature matrix\n",
    "            feature_matrix = group[ap1_features].values\n",
    "\n",
    "            # Remove rows with any NaN values\n",
    "            valid_rows = ~np.isnan(feature_matrix).any(axis=1)\n",
    "            feature_matrix = feature_matrix[valid_rows]\n",
    "\n",
    "            if len(feature_matrix) > 0:\n",
    "                data_dict[condition_id] = feature_matrix\n",
    "                logger.info(f\"Loaded {condition_id}: {feature_matrix.shape}\")\n",
    "            else:\n",
    "                logger.warning(f\"No valid data for {condition_id}\")\n",
    "    else:\n",
    "        # Group by condition, time, and cell line\n",
    "        for (condition, time, cell_line), group in df.groupby(['condition', 'time', 'cell_line']):\n",
    "            # Create condition identifier\n",
    "            condition_id = f\"{cell_line}_{condition}_{time.replace(' ', '')}\"\n",
    "\n",
    "            # Extract feature matrix\n",
    "            feature_matrix = group[ap1_features].values\n",
    "\n",
    "            # Remove rows with any NaN values\n",
    "            valid_rows = ~np.isnan(feature_matrix).any(axis=1)\n",
    "            feature_matrix = feature_matrix[valid_rows]\n",
    "\n",
    "            if len(feature_matrix) > 0:\n",
    "                data_dict[condition_id] = feature_matrix\n",
    "                logger.info(f\"Loaded {condition_id}: {feature_matrix.shape}\")\n",
    "            else:\n",
    "                logger.warning(f\"No valid data for {condition_id}\")\n",
    "\n",
    "    return data_dict\n",
    "\n",
    "def prepare_pair_from_mat(cell_line: str,\n",
    "                          baseline_condition: str, baseline_time: str,\n",
    "                          target_condition: str, target_time: str,\n",
    "                          replicate: Optional[int] = None) -> Tuple[np.ndarray, np.ndarray]:\n",
    "    print(\"Cell line: \", cell_line)\n",
    "    raw_data_dict = load_ap1_data_from_csv('mmc5.xlsx', replicate)\n",
    "\n",
    "    if replicate is not None:\n",
    "        pre_key = f\"{cell_line}_{baseline_condition}_{baseline_time}_rep{replicate}\"\n",
    "        post_key = f\"{cell_line}_{target_condition}_{target_time}_rep{replicate}\"\n",
    "    else:\n",
    "        pre_key = f\"{cell_line}_{baseline_condition}_{baseline_time}\"\n",
    "        post_key = f\"{cell_line}_{target_condition}_{target_time}\"\n",
    "\n",
    "    if pre_key not in raw_data_dict or post_key not in raw_data_dict:\n",
    "        raise ValueError(f\"Pair not found: {pre_key}, {post_key}\")\n",
    "\n",
    "    # Equalize N\n",
    "    n = min(len(raw_data_dict[pre_key]), len(raw_data_dict[post_key]))\n",
    "    X_pre_raw = raw_data_dict[pre_key][:n]\n",
    "    X_post_raw = raw_data_dict[post_key][:n]\n",
    "    return X_pre_raw, X_post_raw\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "75d3cecc",
   "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": null,
   "id": "c533daf9",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "eb96ecaa",
   "metadata": {},
   "outputs": [],
   "source": [
    "import sys\n",
    "import torch\n",
    "from sklearn.preprocessing import StandardScaler\n",
    "from sklearn.decomposition import PCA\n",
    "from cellot.models.cellot import load_networks, compute_loss_f, compute_loss_g\n",
    "\n",
    "from sklearn.metrics.pairwise import rbf_kernel\n",
    "\n",
    "\n",
    "def mmd_distance(x, y, gamma):\n",
    "    xx = rbf_kernel(x, x, gamma)\n",
    "    xy = rbf_kernel(x, y, gamma)\n",
    "    yy = rbf_kernel(y, y, gamma)\n",
    "\n",
    "    return xx.mean() + yy.mean() - 2 * xy.mean()\n",
    "\n",
    "def compute_mmd_loss(lhs, rhs, gammas):\n",
    "    return np.mean([mmd_distance(lhs, rhs, g) for g in gammas])\n",
    "\n",
    "from cellot.losses.mmd import mmd_distance\n",
    "\n",
    "def run_cellot_pair(train_pre: np.ndarray, train_post: np.ndarray,\n",
    "                    test_pre: np.ndarray, test_post: np.ndarray,\n",
    "                    layers: Optional[List[int]] = [64, 64 ,64 ,64],\n",
    "                    n_epochs: int = 5000,\n",
    "                    feature_subset: Optional[List[int]] = None,) -> Dict:\n",
    "    \n",
    "    device = 'cuda'\n",
    "    print(f\"VERS torch={torch.__version__} (CellOT), device={device}\", file=sys.stderr, flush=True)\n",
    "\n",
    "\n",
    "    # Apply feature subset if specified\n",
    "    if feature_subset is not None:\n",
    "        print(f\"Using feature subset of size {len(feature_subset)}\", file=sys.stderr, flush=True)\n",
    "        train_pre = train_pre[:, feature_subset]\n",
    "        train_post = train_post[:, feature_subset]\n",
    "        test_pre = test_pre[:, feature_subset]\n",
    "        test_post = test_post[:, feature_subset]\n",
    "\n",
    "    # Preprocess: standardize jointly and optionally apply PCA for stability\n",
    "    X_all = np.vstack([train_pre, train_post])\n",
    "    scaler = StandardScaler()\n",
    "    X_all_s = scaler.fit_transform(X_all)\n",
    "    d = X_all_s.shape[1]\n",
    "    pca_dims = min(50, d)\n",
    "    if pca_dims < d:\n",
    "        pca = PCA(n_components=pca_dims, svd_solver='full', random_state=42)\n",
    "        X_all_p = pca.fit_transform(X_all_s)\n",
    "        tr_pre_p = X_all_p[:len(train_pre)]\n",
    "        tr_post_p = X_all_p[len(train_pre):]\n",
    "        te_pre_p = pca.transform(scaler.transform(test_pre))\n",
    "        use_pca = True\n",
    "    else:\n",
    "        tr_pre_p = X_all_s[:len(train_pre)]\n",
    "        tr_post_p = X_all_s[len(train_pre):]\n",
    "        te_pre_p = scaler.transform(test_pre)\n",
    "        use_pca = False\n",
    "\n",
    "    # Networks - Using official CellOT configuration\n",
    "    input_dim = tr_pre_p.shape[1]\n",
    "    config = {\n",
    "        'model': {\n",
    "            'name': 'cellot',\n",
    "            'hidden_units': layers,\n",
    "            'kernel_init_fxn': {'name': 'uniform', 'a': -0.01, 'b': 0.01},\n",
    "            'activation': 'relu',\n",
    "            'softplus_W_kernels': True,\n",
    "            'f': {},\n",
    "            'g': {}\n",
    "        }\n",
    "    }\n",
    "    f, g = load_networks(config, input_dim=input_dim)\n",
    "    f = f.to(device).float()\n",
    "    g = g.to(device).float()\n",
    "\n",
    "    # Data tensors\n",
    "    src = torch.tensor(tr_pre_p, dtype=torch.float32, device=device)\n",
    "    tgt = torch.tensor(tr_post_p, dtype=torch.float32, device=device)\n",
    "    te_src = torch.tensor(te_pre_p, dtype=torch.float32, device=device)\n",
    "\n",
    "    # Optimizers matching official config\n",
    "    lr = 1e-4\n",
    "    optim_f = torch.optim.Adam(f.parameters(), lr=lr, betas=(0.5, 0.9), weight_decay=0)\n",
    "    optim_g = torch.optim.Adam(g.parameters(), lr=lr, betas=(0.5, 0.9), weight_decay=0)\n",
    "\n",
    "    # No schedulers in official config\n",
    "    # n_epochs = 1200  # More epochs for better convergence\n",
    "    n_epochs = n_epochs + 1  \n",
    "    # Training loop following official CellOT implementation\n",
    "    f.train(); g.train()\n",
    "    batch_size = 256  # Official config\n",
    "    n_inner_iters = 10  # Official config\n",
    "\n",
    "\n",
    "    for epoch in range(n_epochs):\n",
    "        f.train(); g.train()\n",
    "        perm_t = torch.randperm(len(tgt), device=device)[:batch_size]\n",
    "        yt = tgt[perm_t]\n",
    "        \n",
    "        # Multiple g updates per iteration (official implementation)\n",
    "        for _ in range(n_inner_iters):\n",
    "            perm_s = torch.randperm(len(src), device=device)[:batch_size]\n",
    "            xs = src[perm_s].detach().clone().requires_grad_(True)\n",
    "            \n",
    "            optim_g.zero_grad()\n",
    "            g_loss = compute_loss_g(f, g, xs).mean()\n",
    "            g_loss.backward()\n",
    "            torch.nn.utils.clip_grad_norm_(g.parameters(), max_norm=0.5)\n",
    "            optim_g.step()\n",
    "        \n",
    "        # Single f update (official implementation)\n",
    "        perm_s = torch.randperm(len(src), device=device)[:batch_size]\n",
    "        xs = src[perm_s].detach().clone().requires_grad_(True)\n",
    "        \n",
    "        optim_f.zero_grad()\n",
    "        f_loss = compute_loss_f(f, g, xs, yt).mean()\n",
    "        f_loss.backward()\n",
    "        optim_f.step()\n",
    "        \n",
    "        # Clamp weights for f (official implementation)\n",
    "        if hasattr(f, 'clamp_w'):\n",
    "            f.clamp_w()\n",
    "        \n",
    "        \n",
    "        # ---- Evaluate train MMD and early-stop ----\n",
    "        if epoch % 50 == 0: \n",
    "            f.eval()\n",
    "            g.eval()\n",
    "\n",
    "\n",
    "            # Transport a fixed subset of training PRE (in preprocessed space)\n",
    "            tr_src_eval = src.requires_grad_(True)\n",
    "            tr_pred_p = g.transport(tr_src_eval).detach().cpu().numpy()\n",
    "            # Invert preprocessing to original space (so MMD is comparable to your final eval)\n",
    "            if use_pca:\n",
    "                tr_pred = scaler.inverse_transform(pca.inverse_transform(tr_pred_p))\n",
    "            else:\n",
    "                tr_pred = scaler.inverse_transform(tr_pred_p)\n",
    "            train_mmd_min = mmd_distance(train_post, tr_pred, gamma=1.0)\n",
    "\n",
    "\n",
    "            te_src_full = te_src.detach().clone().requires_grad_(True)\n",
    "            te_pred_full = g.transport(te_src_full).detach().cpu().numpy()\n",
    "            if use_pca:\n",
    "                te_pred_inv_full = scaler.inverse_transform(pca.inverse_transform(te_pred_full))\n",
    "            else:\n",
    "                te_pred_inv_full = scaler.inverse_transform(te_pred_full)\n",
    "            test_metrics = mmd_distance(test_post, te_pred_inv_full, gamma=median_gamma)\n",
    "\n",
    "            print(\n",
    "                f\"[CellOT] epoch={epoch} f_loss={f_loss.item():.4f} g_loss={g_loss.item():.4f} | \"\n",
    "                f\"train mmd={train_mmd_min:.4f} | \"\n",
    "                f\"test_mmd={test_metrics:.4f}\",\n",
    "                file=sys.stderr,\n",
    "                flush=True,\n",
    "            )\n",
    "\n",
    "                \n",
    "            \n",
    "\n",
    "    # Inference (CellOT transport requires gradients for autodiff)\n",
    "    f.eval(); g.eval()\n",
    "    # CellOT needs gradients even in eval mode for transport computation\n",
    "    te_src_for_transport = te_src.detach().clone().requires_grad_(True)\n",
    "    te_tx = g.transport(te_src_for_transport).detach().cpu().numpy()\n",
    "\n",
    "    # Inverse preprocess\n",
    "    if use_pca:\n",
    "        te_tx_inv = scaler.inverse_transform(pca.inverse_transform(te_tx))\n",
    "    else:\n",
    "        te_tx_inv = scaler.inverse_transform(te_tx)\n",
    "    # Final evaluation\n",
    "    metrics = summarize_metrics(test_post[:len(te_tx_inv)], te_tx_inv, median_gamma)\n",
    "\n",
    "    gammas = np.logspace(1, -3, num=50)\n",
    "    mmd = compute_mmd_loss(test_post[:len(te_tx_inv)], te_tx_inv, gammas=gammas)\n",
    "    print(f\"[CellOT] Final CellOT MMD: {mmd:.4f}\", file=sys.stderr, flush=True)\n",
    "    \n",
    "    return {'y_pred': te_tx_inv, 'metrics': metrics}\n",
    "    \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "b2a6a182",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Cell line:  COLO858\n",
      "['DMSO' 'Vem' 'Vem+Tram']\n",
      "X_pre cells: (3026, 20)\n",
      "X_post cells: (3026, 20)\n",
      "(2420, 20)\n",
      "(606, 20)\n",
      "(2420, 20)\n",
      "(606, 20)\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Median heuristic gamma: 0.05162262759745905\n",
      "**************** Run: 0 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=0 f_loss=-3711248.7500 g_loss=4308849.0000 | train mmd=0.1194 | test_mmd=0.8035\n",
      "[CellOT] epoch=50 f_loss=-833.0084 g_loss=188.3626 | train mmd=0.7705 | test_mmd=1.3087\n",
      "[CellOT] epoch=100 f_loss=-310.2260 g_loss=410.9702 | train mmd=1.0892 | test_mmd=0.4525\n",
      "[CellOT] epoch=150 f_loss=-539.5929 g_loss=601.9593 | train mmd=1.0342 | test_mmd=0.4038\n",
      "[CellOT] epoch=200 f_loss=-632.7882 g_loss=750.5660 | train mmd=0.8232 | test_mmd=0.3115\n",
      "[CellOT] epoch=250 f_loss=-905.5563 g_loss=960.3988 | train mmd=0.9845 | test_mmd=0.3211\n",
      "[CellOT] epoch=300 f_loss=-774.8613 g_loss=1053.6223 | train mmd=0.4470 | test_mmd=0.2048\n",
      "[CellOT] epoch=350 f_loss=-1104.7125 g_loss=1341.5306 | train mmd=0.8712 | test_mmd=0.2429\n",
      "[CellOT] epoch=400 f_loss=-1273.2632 g_loss=1473.2087 | train mmd=0.9551 | test_mmd=0.2471\n",
      "[CellOT] epoch=450 f_loss=-1374.7203 g_loss=1634.8203 | train mmd=0.9478 | test_mmd=0.2247\n",
      "[CellOT] epoch=500 f_loss=-1500.1665 g_loss=1812.8491 | train mmd=0.9645 | test_mmd=0.2121\n",
      "[CellOT] epoch=550 f_loss=-1567.6089 g_loss=1840.0544 | train mmd=0.8323 | test_mmd=0.1713\n",
      "[CellOT] epoch=600 f_loss=-2240.4524 g_loss=1976.2119 | train mmd=0.3491 | test_mmd=0.1646\n",
      "[CellOT] epoch=650 f_loss=-1488.0588 g_loss=2021.1210 | train mmd=0.6982 | test_mmd=0.1231\n",
      "[CellOT] epoch=700 f_loss=-1730.2584 g_loss=2349.3823 | train mmd=0.8873 | test_mmd=0.1431\n",
      "[CellOT] epoch=750 f_loss=-1454.6235 g_loss=1977.5425 | train mmd=0.5879 | test_mmd=0.0805\n",
      "[CellOT] epoch=800 f_loss=-1710.4099 g_loss=2527.2412 | train mmd=0.8273 | test_mmd=0.1108\n",
      "[CellOT] epoch=850 f_loss=-1710.0969 g_loss=2499.9160 | train mmd=0.7465 | test_mmd=0.0961\n",
      "[CellOT] epoch=900 f_loss=-1483.9204 g_loss=3755.5569 | train mmd=0.5000 | test_mmd=0.0434\n",
      "[CellOT] epoch=950 f_loss=-1617.9441 g_loss=2584.2573 | train mmd=0.7890 | test_mmd=0.0790\n",
      "[CellOT] epoch=1000 f_loss=-1560.5076 g_loss=3173.5400 | train mmd=0.5347 | test_mmd=0.0559\n",
      "[CellOT] epoch=1050 f_loss=-1365.3591 g_loss=2529.1970 | train mmd=0.6277 | test_mmd=0.0507\n",
      "[CellOT] epoch=1100 f_loss=-1379.6204 g_loss=2537.1768 | train mmd=0.6224 | test_mmd=0.0429\n",
      "[CellOT] epoch=1150 f_loss=-737.4210 g_loss=2643.4988 | train mmd=0.3623 | test_mmd=0.0452\n",
      "[CellOT] epoch=1200 f_loss=-925.8218 g_loss=2688.8452 | train mmd=0.5529 | test_mmd=0.0284\n",
      "[CellOT] epoch=1250 f_loss=-472.5839 g_loss=1869.7354 | train mmd=0.2917 | test_mmd=0.0401\n",
      "[CellOT] epoch=1300 f_loss=-684.0331 g_loss=2535.2087 | train mmd=0.5868 | test_mmd=0.0252\n",
      "[CellOT] epoch=1350 f_loss=58.1754 g_loss=2340.7578 | train mmd=0.2599 | test_mmd=0.0921\n",
      "[CellOT] epoch=1400 f_loss=-235.5298 g_loss=2097.1553 | train mmd=0.4384 | test_mmd=0.0195\n",
      "[CellOT] epoch=1450 f_loss=532.5986 g_loss=1421.7861 | train mmd=0.2240 | test_mmd=0.0769\n",
      "[CellOT] epoch=1500 f_loss=391.5562 g_loss=1380.5820 | train mmd=0.3953 | test_mmd=0.0155\n",
      "[CellOT] epoch=1550 f_loss=347.0425 g_loss=1381.4673 | train mmd=0.2492 | test_mmd=0.0178\n",
      "[CellOT] epoch=1600 f_loss=506.7656 g_loss=1055.3850 | train mmd=0.2235 | test_mmd=0.0514\n",
      "[CellOT] epoch=1650 f_loss=380.7615 g_loss=2399.2466 | train mmd=0.2142 | test_mmd=0.1904\n",
      "[CellOT] epoch=1700 f_loss=346.1938 g_loss=2910.3340 | train mmd=0.3354 | test_mmd=0.0259\n",
      "[CellOT] epoch=1750 f_loss=276.7375 g_loss=4583.7803 | train mmd=0.4008 | test_mmd=0.0236\n",
      "[CellOT] epoch=1800 f_loss=286.2015 g_loss=5379.2031 | train mmd=0.5415 | test_mmd=0.1064\n",
      "[CellOT] epoch=1850 f_loss=217.7732 g_loss=5525.0063 | train mmd=0.2237 | test_mmd=0.1206\n",
      "[CellOT] epoch=1900 f_loss=218.8994 g_loss=5634.7432 | train mmd=0.4410 | test_mmd=0.0206\n",
      "[CellOT] epoch=1950 f_loss=110.9526 g_loss=6207.3369 | train mmd=0.3337 | test_mmd=0.0288\n",
      "[CellOT] epoch=2000 f_loss=-257.9941 g_loss=6316.1289 | train mmd=0.5131 | test_mmd=0.0505\n",
      "[CellOT] Final CellOT MMD: 0.2067\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n",
      "[CellOT] epoch=0 f_loss=-2394172.0000 g_loss=2706679.5000 | train mmd=0.1194 | test_mmd=0.8035\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 0 metrics: {'mmd2_gamma_median': 0.05048895384609753, 'mmd2_gamma_0.5': 0.40183343459039045, 'mmd2_gamma_1.0': 0.5267122524272445, 'wasserstein_distance': 1.5104203871379596, 'R2_feature_means': 0.9334617192053316}\n",
      "**************** Run: 1 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=50 f_loss=-321.7100 g_loss=1471.6335 | train mmd=0.5025 | test_mmd=0.7120\n",
      "[CellOT] epoch=100 f_loss=-264.5189 g_loss=368.6028 | train mmd=1.1039 | test_mmd=0.4705\n",
      "[CellOT] epoch=150 f_loss=-466.9117 g_loss=544.0601 | train mmd=0.9753 | test_mmd=0.3924\n",
      "[CellOT] epoch=200 f_loss=-671.4962 g_loss=702.1522 | train mmd=0.9752 | test_mmd=0.3749\n",
      "[CellOT] epoch=250 f_loss=-807.8167 g_loss=915.1145 | train mmd=0.9810 | test_mmd=0.3454\n",
      "[CellOT] epoch=300 f_loss=-882.2280 g_loss=951.0939 | train mmd=0.8936 | test_mmd=0.2864\n",
      "[CellOT] epoch=350 f_loss=-990.1663 g_loss=1087.4072 | train mmd=0.8887 | test_mmd=0.2623\n",
      "[CellOT] epoch=400 f_loss=-1091.5792 g_loss=1274.7302 | train mmd=0.8268 | test_mmd=0.2260\n",
      "[CellOT] epoch=450 f_loss=-1218.9976 g_loss=1457.7679 | train mmd=0.9830 | test_mmd=0.2363\n",
      "[CellOT] epoch=500 f_loss=-1374.8107 g_loss=1628.7124 | train mmd=0.9841 | test_mmd=0.2187\n",
      "[CellOT] epoch=550 f_loss=-1295.7802 g_loss=1601.2111 | train mmd=0.9081 | test_mmd=0.1802\n",
      "[CellOT] epoch=600 f_loss=-1434.3641 g_loss=1788.6482 | train mmd=0.9508 | test_mmd=0.1730\n",
      "[CellOT] epoch=650 f_loss=-1367.2296 g_loss=1799.8857 | train mmd=0.8784 | test_mmd=0.1400\n",
      "[CellOT] epoch=700 f_loss=-1197.8254 g_loss=1567.4995 | train mmd=0.6826 | test_mmd=0.0885\n",
      "[CellOT] epoch=750 f_loss=-1330.5688 g_loss=1855.2108 | train mmd=0.8040 | test_mmd=0.0987\n",
      "[CellOT] epoch=800 f_loss=-1227.5488 g_loss=1867.1595 | train mmd=0.7363 | test_mmd=0.0730\n",
      "[CellOT] epoch=850 f_loss=-1234.8894 g_loss=1905.7220 | train mmd=0.7791 | test_mmd=0.0719\n",
      "[CellOT] epoch=900 f_loss=-941.5344 g_loss=1730.5951 | train mmd=0.6365 | test_mmd=0.0486\n",
      "[CellOT] epoch=950 f_loss=-830.6105 g_loss=1588.7976 | train mmd=0.5827 | test_mmd=0.0451\n",
      "[CellOT] epoch=1000 f_loss=-434.6685 g_loss=1377.0477 | train mmd=0.4427 | test_mmd=0.0158\n",
      "[CellOT] epoch=1050 f_loss=322.0960 g_loss=720.4773 | train mmd=0.2465 | test_mmd=0.0947\n",
      "[CellOT] epoch=1100 f_loss=-240.1601 g_loss=1451.8851 | train mmd=0.4816 | test_mmd=0.0169\n",
      "[CellOT] epoch=1150 f_loss=188.6239 g_loss=951.6958 | train mmd=0.2210 | test_mmd=0.0173\n",
      "[CellOT] epoch=1200 f_loss=-219.9604 g_loss=1158.4064 | train mmd=0.5565 | test_mmd=0.0229\n",
      "[CellOT] epoch=1250 f_loss=258.0971 g_loss=905.7253 | train mmd=0.2701 | test_mmd=0.0210\n",
      "[CellOT] epoch=1300 f_loss=518.4462 g_loss=417.9764 | train mmd=0.2311 | test_mmd=0.1086\n",
      "[CellOT] epoch=1350 f_loss=413.0073 g_loss=669.7982 | train mmd=0.4366 | test_mmd=0.2511\n",
      "[CellOT] epoch=1400 f_loss=261.3876 g_loss=1125.7285 | train mmd=0.2518 | test_mmd=0.0168\n",
      "[CellOT] epoch=1450 f_loss=184.3472 g_loss=1392.4840 | train mmd=0.2792 | test_mmd=0.0201\n",
      "[CellOT] epoch=1500 f_loss=135.8281 g_loss=1532.7563 | train mmd=0.3199 | test_mmd=0.0812\n",
      "[CellOT] epoch=1550 f_loss=48.0525 g_loss=1685.6057 | train mmd=0.5680 | test_mmd=0.1002\n",
      "[CellOT] epoch=1600 f_loss=37.0253 g_loss=1819.1173 | train mmd=0.5829 | test_mmd=0.1219\n",
      "[CellOT] epoch=1650 f_loss=48.0134 g_loss=1912.6067 | train mmd=0.2761 | test_mmd=0.0268\n",
      "[CellOT] epoch=1700 f_loss=38.0039 g_loss=2013.1731 | train mmd=0.2980 | test_mmd=0.0298\n",
      "[CellOT] epoch=1750 f_loss=35.0981 g_loss=2074.8682 | train mmd=0.4671 | test_mmd=0.2678\n",
      "[CellOT] epoch=1800 f_loss=28.3233 g_loss=2130.7896 | train mmd=0.4499 | test_mmd=0.0274\n",
      "[CellOT] epoch=1850 f_loss=12.6035 g_loss=2222.0662 | train mmd=0.2864 | test_mmd=0.0340\n",
      "[CellOT] epoch=1900 f_loss=7.7913 g_loss=2303.8157 | train mmd=0.3752 | test_mmd=0.0301\n",
      "[CellOT] epoch=1950 f_loss=32.4381 g_loss=2375.4507 | train mmd=0.3451 | test_mmd=0.1189\n",
      "[CellOT] epoch=2000 f_loss=13.4440 g_loss=2379.3237 | train mmd=0.4516 | test_mmd=0.0379\n",
      "[CellOT] Final CellOT MMD: 0.2352\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n",
      "[CellOT] epoch=0 f_loss=-3228914.5000 g_loss=3646237.2500 | train mmd=0.1194 | test_mmd=0.8035\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 1 metrics: {'mmd2_gamma_median': 0.03786830422157106, 'mmd2_gamma_0.5': 0.32268922994091676, 'mmd2_gamma_1.0': 0.48240962165214973, 'wasserstein_distance': 1.5642868795778642, 'R2_feature_means': 0.9474534715974264}\n",
      "**************** Run: 2 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=50 f_loss=333.2966 g_loss=608.5569 | train mmd=0.8554 | test_mmd=0.4500\n",
      "[CellOT] epoch=100 f_loss=-259.7007 g_loss=346.0290 | train mmd=1.1006 | test_mmd=0.4563\n",
      "[CellOT] epoch=150 f_loss=-489.6950 g_loss=534.0507 | train mmd=1.0873 | test_mmd=0.4226\n",
      "[CellOT] epoch=200 f_loss=-592.2945 g_loss=658.7969 | train mmd=0.9379 | test_mmd=0.3332\n",
      "[CellOT] epoch=250 f_loss=-702.2308 g_loss=811.6910 | train mmd=0.8385 | test_mmd=0.2770\n",
      "[CellOT] epoch=300 f_loss=-838.1639 g_loss=979.9009 | train mmd=0.8593 | test_mmd=0.2920\n",
      "[CellOT] epoch=350 f_loss=-927.4857 g_loss=1152.3304 | train mmd=0.8818 | test_mmd=0.2646\n",
      "[CellOT] epoch=400 f_loss=-1010.9990 g_loss=1204.1149 | train mmd=0.8062 | test_mmd=0.1964\n",
      "[CellOT] epoch=450 f_loss=-1144.4548 g_loss=1264.0502 | train mmd=0.8381 | test_mmd=0.1943\n",
      "[CellOT] epoch=500 f_loss=-1224.7454 g_loss=1455.1797 | train mmd=0.9006 | test_mmd=0.1853\n",
      "[CellOT] epoch=550 f_loss=-1166.1349 g_loss=1393.3724 | train mmd=0.8494 | test_mmd=0.1561\n",
      "[CellOT] epoch=600 f_loss=-1273.4744 g_loss=1729.7529 | train mmd=0.8813 | test_mmd=0.1568\n",
      "[CellOT] epoch=650 f_loss=-1181.2096 g_loss=1701.5164 | train mmd=0.8298 | test_mmd=0.1274\n",
      "[CellOT] epoch=700 f_loss=-1288.6292 g_loss=1729.4991 | train mmd=0.8241 | test_mmd=0.1172\n",
      "[CellOT] epoch=750 f_loss=-1242.9274 g_loss=1861.6727 | train mmd=0.7897 | test_mmd=0.0991\n",
      "[CellOT] epoch=800 f_loss=-871.9249 g_loss=613.2690 | train mmd=0.5382 | test_mmd=0.0638\n",
      "[CellOT] epoch=850 f_loss=-1113.3560 g_loss=1806.0695 | train mmd=0.7115 | test_mmd=0.0665\n",
      "[CellOT] epoch=900 f_loss=-1041.6133 g_loss=1943.6772 | train mmd=0.7271 | test_mmd=0.0599\n",
      "[CellOT] epoch=950 f_loss=-1048.3729 g_loss=1892.5699 | train mmd=0.7099 | test_mmd=0.0527\n",
      "[CellOT] epoch=1000 f_loss=-434.4273 g_loss=1469.5502 | train mmd=0.3658 | test_mmd=0.0130\n",
      "[CellOT] epoch=1050 f_loss=-641.5339 g_loss=1799.2941 | train mmd=0.5929 | test_mmd=0.0325\n",
      "[CellOT] epoch=1100 f_loss=-453.8997 g_loss=1678.8646 | train mmd=0.5531 | test_mmd=0.0269\n",
      "[CellOT] epoch=1150 f_loss=-368.4128 g_loss=1648.2224 | train mmd=0.5475 | test_mmd=0.0233\n",
      "[CellOT] epoch=1200 f_loss=-50.9999 g_loss=1271.2717 | train mmd=0.4392 | test_mmd=0.0204\n",
      "[CellOT] epoch=1250 f_loss=-58.7960 g_loss=1297.2300 | train mmd=0.5118 | test_mmd=0.0195\n",
      "[CellOT] epoch=1300 f_loss=336.1712 g_loss=1096.6611 | train mmd=0.3464 | test_mmd=0.0114\n",
      "[CellOT] epoch=1350 f_loss=639.5896 g_loss=1125.9388 | train mmd=0.2931 | test_mmd=0.0211\n",
      "[CellOT] epoch=1400 f_loss=570.6132 g_loss=541.2994 | train mmd=0.8145 | test_mmd=0.3384\n",
      "[CellOT] epoch=1450 f_loss=414.8147 g_loss=1448.9266 | train mmd=0.3224 | test_mmd=0.0293\n",
      "[CellOT] epoch=1500 f_loss=236.5335 g_loss=2729.2500 | train mmd=0.3080 | test_mmd=0.0406\n",
      "[CellOT] epoch=1550 f_loss=186.7422 g_loss=3110.3594 | train mmd=0.2799 | test_mmd=0.1010\n",
      "[CellOT] epoch=1600 f_loss=175.1766 g_loss=3834.0117 | train mmd=0.4766 | test_mmd=0.1012\n",
      "[CellOT] epoch=1650 f_loss=125.8577 g_loss=4031.8506 | train mmd=0.3292 | test_mmd=0.0197\n",
      "[CellOT] epoch=1700 f_loss=113.2542 g_loss=4524.0098 | train mmd=0.3136 | test_mmd=0.1070\n",
      "[CellOT] epoch=1750 f_loss=-63.5202 g_loss=4942.9976 | train mmd=0.3025 | test_mmd=0.0976\n",
      "[CellOT] epoch=1800 f_loss=33.9680 g_loss=5145.1221 | train mmd=0.5145 | test_mmd=0.1244\n",
      "[CellOT] epoch=1850 f_loss=150.4073 g_loss=5398.0527 | train mmd=0.6625 | test_mmd=0.4919\n",
      "[CellOT] epoch=1900 f_loss=48.7215 g_loss=5540.4502 | train mmd=0.8004 | test_mmd=0.1289\n",
      "[CellOT] epoch=1950 f_loss=82.3656 g_loss=5787.1753 | train mmd=0.5351 | test_mmd=0.1505\n",
      "[CellOT] epoch=2000 f_loss=84.1736 g_loss=5769.4277 | train mmd=0.2793 | test_mmd=0.0467\n",
      "[CellOT] Final CellOT MMD: 0.1470\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n",
      "[CellOT] epoch=0 f_loss=-3716457.2500 g_loss=4422647.0000 | train mmd=0.1194 | test_mmd=0.8035\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 2 metrics: {'mmd2_gamma_median': 0.04669145024276067, 'mmd2_gamma_0.5': 0.2190976554031423, 'mmd2_gamma_1.0': 0.29485659854956026, 'wasserstein_distance': 1.5051835959774167, 'R2_feature_means': 0.9194399404930336}\n",
      "**************** Run: 3 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=50 f_loss=291.2912 g_loss=290.3577 | train mmd=1.0950 | test_mmd=0.4806\n",
      "[CellOT] epoch=100 f_loss=-404.4247 g_loss=534.0639 | train mmd=1.1022 | test_mmd=0.4497\n",
      "[CellOT] epoch=150 f_loss=-617.6700 g_loss=719.9370 | train mmd=0.8297 | test_mmd=0.3488\n",
      "[CellOT] epoch=200 f_loss=-829.7662 g_loss=901.8757 | train mmd=0.8224 | test_mmd=0.3089\n",
      "[CellOT] epoch=250 f_loss=-412.4865 g_loss=413.6842 | train mmd=0.3198 | test_mmd=0.1992\n",
      "[CellOT] epoch=300 f_loss=-1260.0210 g_loss=1427.8896 | train mmd=0.9695 | test_mmd=0.2818\n",
      "[CellOT] epoch=350 f_loss=-1342.4579 g_loss=1520.9692 | train mmd=0.9110 | test_mmd=0.2414\n",
      "[CellOT] epoch=400 f_loss=-974.6050 g_loss=1541.9993 | train mmd=0.5298 | test_mmd=0.1775\n",
      "[CellOT] epoch=450 f_loss=-1545.4827 g_loss=1769.6133 | train mmd=0.9009 | test_mmd=0.1870\n",
      "[CellOT] epoch=500 f_loss=-1694.9407 g_loss=2030.6742 | train mmd=0.9311 | test_mmd=0.1767\n",
      "[CellOT] epoch=550 f_loss=-1361.7074 g_loss=1952.9875 | train mmd=0.7693 | test_mmd=0.1327\n",
      "[CellOT] epoch=600 f_loss=-1494.6077 g_loss=2171.8071 | train mmd=0.8018 | test_mmd=0.1193\n",
      "[CellOT] epoch=650 f_loss=-1711.5527 g_loss=2330.5894 | train mmd=0.8907 | test_mmd=0.1310\n",
      "[CellOT] epoch=700 f_loss=-1557.1721 g_loss=2269.8123 | train mmd=0.7185 | test_mmd=0.0911\n",
      "[CellOT] epoch=750 f_loss=-941.2489 g_loss=1882.5043 | train mmd=0.4258 | test_mmd=0.1028\n",
      "[CellOT] epoch=800 f_loss=-1392.0566 g_loss=1890.7411 | train mmd=0.5856 | test_mmd=0.0741\n",
      "[CellOT] epoch=850 f_loss=-760.1407 g_loss=1607.9077 | train mmd=0.3308 | test_mmd=0.0373\n",
      "[CellOT] epoch=900 f_loss=-1300.7683 g_loss=2275.8164 | train mmd=0.6664 | test_mmd=0.0577\n",
      "[CellOT] epoch=950 f_loss=-860.9681 g_loss=2058.1301 | train mmd=0.4884 | test_mmd=0.0297\n",
      "[CellOT] epoch=1000 f_loss=-719.1471 g_loss=1779.8767 | train mmd=0.3950 | test_mmd=0.0125\n",
      "[CellOT] epoch=1050 f_loss=-469.3294 g_loss=1754.4053 | train mmd=0.3705 | test_mmd=0.0246\n",
      "[CellOT] epoch=1100 f_loss=-411.0247 g_loss=1590.8525 | train mmd=0.3422 | test_mmd=0.0118\n",
      "[CellOT] epoch=1150 f_loss=36.8093 g_loss=1308.9680 | train mmd=0.2316 | test_mmd=0.0134\n",
      "[CellOT] epoch=1200 f_loss=204.6785 g_loss=1270.0255 | train mmd=0.2310 | test_mmd=0.0507\n",
      "[CellOT] epoch=1250 f_loss=522.3653 g_loss=920.3507 | train mmd=0.1461 | test_mmd=0.0647\n",
      "[CellOT] epoch=1300 f_loss=647.4952 g_loss=231.1236 | train mmd=0.4921 | test_mmd=0.1181\n",
      "[CellOT] epoch=1350 f_loss=592.8937 g_loss=729.7634 | train mmd=0.3289 | test_mmd=0.0307\n",
      "[CellOT] epoch=1400 f_loss=823.5034 g_loss=550.5197 | train mmd=0.2157 | test_mmd=0.0377\n",
      "[CellOT] epoch=1450 f_loss=326.2511 g_loss=581.7768 | train mmd=0.1601 | test_mmd=0.0419\n",
      "[CellOT] epoch=1500 f_loss=285.0130 g_loss=703.3934 | train mmd=0.2549 | test_mmd=0.1134\n",
      "[CellOT] epoch=1550 f_loss=182.5706 g_loss=884.2104 | train mmd=0.3398 | test_mmd=0.0299\n",
      "[CellOT] epoch=1600 f_loss=132.3535 g_loss=999.4786 | train mmd=0.2954 | test_mmd=0.1252\n",
      "[CellOT] epoch=1650 f_loss=150.6293 g_loss=1130.5403 | train mmd=0.3378 | test_mmd=0.1682\n",
      "[CellOT] epoch=1700 f_loss=112.1689 g_loss=1211.8094 | train mmd=0.3177 | test_mmd=0.1441\n",
      "[CellOT] epoch=1750 f_loss=55.8189 g_loss=1309.0834 | train mmd=0.3919 | test_mmd=0.1578\n",
      "[CellOT] epoch=1800 f_loss=108.0921 g_loss=1384.7568 | train mmd=0.3154 | test_mmd=0.0243\n",
      "[CellOT] epoch=1850 f_loss=28.0178 g_loss=1439.3604 | train mmd=0.2369 | test_mmd=0.0458\n",
      "[CellOT] epoch=1900 f_loss=28.6261 g_loss=1496.3849 | train mmd=0.5238 | test_mmd=0.2352\n",
      "[CellOT] epoch=1950 f_loss=20.9681 g_loss=1547.8958 | train mmd=0.9251 | test_mmd=0.8733\n",
      "[CellOT] epoch=2000 f_loss=37.3120 g_loss=1589.0955 | train mmd=0.1815 | test_mmd=0.0419\n",
      "[CellOT] Final CellOT MMD: 0.1005\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n",
      "[CellOT] epoch=0 f_loss=-3264690.0000 g_loss=3713765.5000 | train mmd=0.1194 | test_mmd=0.8035\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 3 metrics: {'mmd2_gamma_median': 0.04191160228126467, 'mmd2_gamma_0.5': 0.1312207036709674, 'mmd2_gamma_1.0': 0.20153978514346998, 'wasserstein_distance': 1.5048635644856905, 'R2_feature_means': 0.8645338277590623}\n",
      "**************** Run: 4 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=50 f_loss=371.1553 g_loss=220.1760 | train mmd=1.0970 | test_mmd=0.4802\n",
      "[CellOT] epoch=100 f_loss=-340.8432 g_loss=485.8857 | train mmd=1.1017 | test_mmd=0.4632\n",
      "[CellOT] epoch=150 f_loss=-610.1500 g_loss=721.6631 | train mmd=1.1003 | test_mmd=0.4324\n",
      "[CellOT] epoch=200 f_loss=-763.4123 g_loss=865.2776 | train mmd=0.8806 | test_mmd=0.3500\n",
      "[CellOT] epoch=250 f_loss=-980.2932 g_loss=1089.5227 | train mmd=0.9692 | test_mmd=0.3319\n",
      "[CellOT] epoch=300 f_loss=-1161.0197 g_loss=1343.0693 | train mmd=0.9954 | test_mmd=0.3068\n",
      "[CellOT] epoch=350 f_loss=-1286.6556 g_loss=1375.4475 | train mmd=0.9397 | test_mmd=0.2600\n",
      "[CellOT] epoch=400 f_loss=-1408.2736 g_loss=1552.1418 | train mmd=0.8610 | test_mmd=0.2195\n",
      "[CellOT] epoch=450 f_loss=-1401.1232 g_loss=1724.2766 | train mmd=0.9095 | test_mmd=0.2113\n",
      "[CellOT] epoch=500 f_loss=-1507.4128 g_loss=1877.3314 | train mmd=0.9092 | test_mmd=0.2038\n",
      "[CellOT] epoch=550 f_loss=-1596.6880 g_loss=1963.6460 | train mmd=0.8575 | test_mmd=0.1773\n",
      "[CellOT] epoch=600 f_loss=-1766.8804 g_loss=2155.4917 | train mmd=0.9296 | test_mmd=0.1704\n",
      "[CellOT] epoch=650 f_loss=-1537.0154 g_loss=1968.9615 | train mmd=0.7913 | test_mmd=0.1212\n",
      "[CellOT] epoch=700 f_loss=-1657.2654 g_loss=2224.8972 | train mmd=0.8391 | test_mmd=0.1168\n",
      "[CellOT] epoch=750 f_loss=-1351.1302 g_loss=2001.7581 | train mmd=0.6933 | test_mmd=0.0900\n",
      "[CellOT] epoch=800 f_loss=-1441.9521 g_loss=2344.0625 | train mmd=0.7808 | test_mmd=0.0797\n",
      "[CellOT] epoch=850 f_loss=-1360.8873 g_loss=2126.1179 | train mmd=0.6755 | test_mmd=0.0557\n",
      "[CellOT] epoch=900 f_loss=-460.5046 g_loss=1906.6260 | train mmd=0.3997 | test_mmd=0.0216\n",
      "[CellOT] epoch=950 f_loss=-903.0786 g_loss=1746.8470 | train mmd=0.4770 | test_mmd=0.0205\n",
      "[CellOT] epoch=1000 f_loss=-812.5750 g_loss=1748.8298 | train mmd=0.4997 | test_mmd=0.0305\n",
      "[CellOT] epoch=1050 f_loss=13.3225 g_loss=822.7970 | train mmd=0.2379 | test_mmd=0.0727\n",
      "[CellOT] epoch=1100 f_loss=-81.9688 g_loss=1240.1814 | train mmd=0.3546 | test_mmd=0.0249\n",
      "[CellOT] epoch=1150 f_loss=14.3452 g_loss=1230.4865 | train mmd=0.3121 | test_mmd=0.0312\n",
      "[CellOT] epoch=1200 f_loss=751.8273 g_loss=329.4335 | train mmd=0.2705 | test_mmd=0.0662\n",
      "[CellOT] epoch=1250 f_loss=546.3494 g_loss=676.6817 | train mmd=0.2130 | test_mmd=0.0417\n",
      "[CellOT] epoch=1300 f_loss=503.9720 g_loss=491.7362 | train mmd=0.2896 | test_mmd=0.0364\n",
      "[CellOT] epoch=1350 f_loss=245.6192 g_loss=840.0536 | train mmd=0.2595 | test_mmd=0.0681\n",
      "[CellOT] epoch=1400 f_loss=151.3449 g_loss=1025.0216 | train mmd=0.4160 | test_mmd=0.0201\n",
      "[CellOT] epoch=1450 f_loss=94.3656 g_loss=1209.7524 | train mmd=0.7102 | test_mmd=0.1815\n",
      "[CellOT] epoch=1500 f_loss=110.6782 g_loss=1322.6215 | train mmd=0.3666 | test_mmd=0.0210\n",
      "[CellOT] epoch=1550 f_loss=84.3970 g_loss=1439.5654 | train mmd=0.6101 | test_mmd=0.2036\n",
      "[CellOT] epoch=1600 f_loss=39.4391 g_loss=1523.8029 | train mmd=0.3112 | test_mmd=0.0903\n",
      "[CellOT] epoch=1650 f_loss=17.6261 g_loss=1606.4128 | train mmd=0.3084 | test_mmd=0.0252\n",
      "[CellOT] epoch=1700 f_loss=31.4818 g_loss=1682.6108 | train mmd=0.4159 | test_mmd=0.5260\n",
      "[CellOT] epoch=1750 f_loss=34.2109 g_loss=1753.7510 | train mmd=0.3029 | test_mmd=0.1247\n",
      "[CellOT] epoch=1800 f_loss=53.6163 g_loss=1798.4326 | train mmd=0.7597 | test_mmd=0.1850\n",
      "[CellOT] epoch=1850 f_loss=17.2371 g_loss=1833.4338 | train mmd=0.2593 | test_mmd=0.0700\n",
      "[CellOT] epoch=1900 f_loss=17.7451 g_loss=1881.4243 | train mmd=0.2922 | test_mmd=0.0657\n",
      "[CellOT] epoch=1950 f_loss=8.9966 g_loss=1915.2062 | train mmd=0.2522 | test_mmd=0.0519\n",
      "[CellOT] epoch=2000 f_loss=22.0655 g_loss=1928.9226 | train mmd=0.2700 | test_mmd=0.0529\n",
      "[CellOT] Final CellOT MMD: 0.1458\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n",
      "[CellOT] epoch=0 f_loss=-3371216.7500 g_loss=3750507.0000 | train mmd=0.1194 | test_mmd=0.8035\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 4 metrics: {'mmd2_gamma_median': 0.05290110297442174, 'mmd2_gamma_0.5': 0.22091388092033937, 'mmd2_gamma_1.0': 0.28659624122444904, 'wasserstein_distance': 1.4928620492401263, 'R2_feature_means': 0.9109169228088972}\n",
      "**************** Run: 5 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=50 f_loss=537.1843 g_loss=208.9524 | train mmd=1.0932 | test_mmd=0.4863\n",
      "[CellOT] epoch=100 f_loss=-284.5012 g_loss=354.9893 | train mmd=1.1031 | test_mmd=0.4548\n",
      "[CellOT] epoch=150 f_loss=-402.3971 g_loss=462.8392 | train mmd=0.8293 | test_mmd=0.3416\n",
      "[CellOT] epoch=200 f_loss=-621.2473 g_loss=637.9877 | train mmd=0.9043 | test_mmd=0.3398\n",
      "[CellOT] epoch=250 f_loss=-686.2466 g_loss=771.1151 | train mmd=0.8315 | test_mmd=0.2757\n",
      "[CellOT] epoch=300 f_loss=-854.4163 g_loss=990.0989 | train mmd=0.9376 | test_mmd=0.2735\n",
      "[CellOT] epoch=350 f_loss=-779.4391 g_loss=957.7034 | train mmd=0.7346 | test_mmd=0.1825\n",
      "[CellOT] epoch=400 f_loss=-983.4127 g_loss=1147.0083 | train mmd=0.8352 | test_mmd=0.1985\n",
      "[CellOT] epoch=450 f_loss=-976.3540 g_loss=1156.4822 | train mmd=0.7817 | test_mmd=0.1709\n",
      "[CellOT] epoch=500 f_loss=-1044.4026 g_loss=1329.3972 | train mmd=0.7699 | test_mmd=0.1721\n",
      "[CellOT] epoch=550 f_loss=-1012.5821 g_loss=1327.0564 | train mmd=0.7042 | test_mmd=0.1144\n",
      "[CellOT] epoch=600 f_loss=-1260.5537 g_loss=1464.4340 | train mmd=0.8852 | test_mmd=0.1455\n",
      "[CellOT] epoch=650 f_loss=-1238.8257 g_loss=1632.4968 | train mmd=0.8802 | test_mmd=0.1257\n",
      "[CellOT] epoch=700 f_loss=-1182.3276 g_loss=1645.7855 | train mmd=0.8224 | test_mmd=0.1029\n",
      "[CellOT] epoch=750 f_loss=-506.6436 g_loss=1171.3300 | train mmd=0.4262 | test_mmd=0.0654\n",
      "[CellOT] epoch=800 f_loss=-886.6049 g_loss=1505.3660 | train mmd=0.6551 | test_mmd=0.0806\n",
      "[CellOT] epoch=850 f_loss=-564.2461 g_loss=1305.5625 | train mmd=0.4170 | test_mmd=0.0295\n",
      "[CellOT] epoch=900 f_loss=-865.8182 g_loss=1547.7109 | train mmd=0.6838 | test_mmd=0.0558\n",
      "[CellOT] epoch=950 f_loss=-292.8719 g_loss=1188.3496 | train mmd=0.3271 | test_mmd=0.0210\n",
      "[CellOT] epoch=1000 f_loss=-571.6363 g_loss=1378.2273 | train mmd=0.4666 | test_mmd=0.0459\n",
      "[CellOT] epoch=1050 f_loss=-345.6153 g_loss=1247.4625 | train mmd=0.5037 | test_mmd=0.0210\n",
      "[CellOT] epoch=1100 f_loss=-212.5431 g_loss=1321.2709 | train mmd=0.4392 | test_mmd=0.0226\n",
      "[CellOT] epoch=1150 f_loss=264.8593 g_loss=675.5964 | train mmd=0.2170 | test_mmd=0.0741\n",
      "[CellOT] epoch=1200 f_loss=-24.9392 g_loss=1043.9641 | train mmd=0.3984 | test_mmd=0.0182\n",
      "[CellOT] epoch=1250 f_loss=118.9307 g_loss=995.9104 | train mmd=0.3701 | test_mmd=0.0158\n",
      "[CellOT] epoch=1300 f_loss=299.1110 g_loss=623.8364 | train mmd=0.3251 | test_mmd=0.0184\n",
      "[CellOT] epoch=1350 f_loss=577.0228 g_loss=537.4897 | train mmd=0.2835 | test_mmd=0.1367\n",
      "[CellOT] epoch=1400 f_loss=353.8289 g_loss=788.4496 | train mmd=0.3722 | test_mmd=0.0153\n",
      "[CellOT] epoch=1450 f_loss=493.6592 g_loss=746.3757 | train mmd=0.2995 | test_mmd=0.0585\n",
      "[CellOT] epoch=1500 f_loss=421.0737 g_loss=971.2675 | train mmd=0.2650 | test_mmd=0.0766\n",
      "[CellOT] epoch=1550 f_loss=176.8998 g_loss=1668.7700 | train mmd=0.2995 | test_mmd=0.0225\n",
      "[CellOT] epoch=1600 f_loss=117.1001 g_loss=2137.0352 | train mmd=0.2811 | test_mmd=0.1074\n",
      "[CellOT] epoch=1650 f_loss=97.8166 g_loss=2458.9727 | train mmd=0.3473 | test_mmd=0.0654\n",
      "[CellOT] epoch=1700 f_loss=75.9906 g_loss=2784.6455 | train mmd=0.2938 | test_mmd=0.0525\n",
      "[CellOT] epoch=1750 f_loss=28.4535 g_loss=3004.5063 | train mmd=0.4535 | test_mmd=0.0239\n",
      "[CellOT] epoch=1800 f_loss=43.9423 g_loss=3143.7864 | train mmd=0.3653 | test_mmd=0.1094\n",
      "[CellOT] epoch=1850 f_loss=37.9563 g_loss=3232.4209 | train mmd=0.2084 | test_mmd=0.0304\n",
      "[CellOT] epoch=1900 f_loss=8.3077 g_loss=3354.8518 | train mmd=0.4608 | test_mmd=0.0247\n",
      "[CellOT] epoch=1950 f_loss=18.3228 g_loss=3475.0645 | train mmd=0.6104 | test_mmd=0.1979\n",
      "[CellOT] epoch=2000 f_loss=15.1981 g_loss=3555.9629 | train mmd=0.9457 | test_mmd=0.2585\n",
      "[CellOT] Final CellOT MMD: 0.4570\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n",
      "[CellOT] epoch=0 f_loss=-2759912.7500 g_loss=2823950.2500 | train mmd=0.1194 | test_mmd=0.8035\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 5 metrics: {'mmd2_gamma_median': 0.25848034189101265, 'mmd2_gamma_0.5': 0.9557154129384127, 'mmd2_gamma_1.0': 0.9562474687534365, 'wasserstein_distance': 2.2224580725430165, 'R2_feature_means': 0.5367073987061493}\n",
      "**************** Run: 6 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=50 f_loss=401.5027 g_loss=343.5271 | train mmd=1.0955 | test_mmd=0.4833\n",
      "[CellOT] epoch=100 f_loss=-299.4300 g_loss=369.9283 | train mmd=1.1005 | test_mmd=0.4492\n",
      "[CellOT] epoch=150 f_loss=-459.7321 g_loss=533.3893 | train mmd=0.9536 | test_mmd=0.3782\n",
      "[CellOT] epoch=200 f_loss=-640.8138 g_loss=698.8550 | train mmd=0.9205 | test_mmd=0.3362\n",
      "[CellOT] epoch=250 f_loss=-790.4200 g_loss=883.5696 | train mmd=0.9778 | test_mmd=0.3071\n",
      "[CellOT] epoch=300 f_loss=-927.0762 g_loss=997.4968 | train mmd=0.8999 | test_mmd=0.2575\n",
      "[CellOT] epoch=350 f_loss=-993.5437 g_loss=1126.6053 | train mmd=0.9689 | test_mmd=0.2660\n",
      "[CellOT] epoch=400 f_loss=-1039.5995 g_loss=1239.8086 | train mmd=0.9030 | test_mmd=0.2199\n",
      "[CellOT] epoch=450 f_loss=-736.6775 g_loss=896.5120 | train mmd=0.5049 | test_mmd=0.1089\n",
      "[CellOT] epoch=500 f_loss=-1127.2477 g_loss=1447.3062 | train mmd=0.9039 | test_mmd=0.1790\n",
      "[CellOT] epoch=550 f_loss=-1135.1716 g_loss=1443.1674 | train mmd=0.8598 | test_mmd=0.1482\n",
      "[CellOT] epoch=600 f_loss=-896.2444 g_loss=1370.2419 | train mmd=0.6251 | test_mmd=0.0832\n",
      "[CellOT] epoch=650 f_loss=-1195.8090 g_loss=1651.5441 | train mmd=0.8547 | test_mmd=0.1192\n",
      "[CellOT] epoch=700 f_loss=-776.8064 g_loss=1386.7964 | train mmd=0.5513 | test_mmd=0.0545\n",
      "[CellOT] epoch=750 f_loss=-665.8385 g_loss=1102.5422 | train mmd=0.4338 | test_mmd=0.0705\n",
      "[CellOT] epoch=800 f_loss=-351.3026 g_loss=1061.0670 | train mmd=0.3378 | test_mmd=0.0419\n",
      "[CellOT] epoch=850 f_loss=-788.8989 g_loss=1570.2877 | train mmd=0.6411 | test_mmd=0.0457\n",
      "[CellOT] epoch=900 f_loss=-650.5813 g_loss=1510.0930 | train mmd=0.5699 | test_mmd=0.0407\n",
      "[CellOT] epoch=950 f_loss=-519.9928 g_loss=1504.8545 | train mmd=0.6311 | test_mmd=0.0349\n",
      "[CellOT] epoch=1000 f_loss=-14.1903 g_loss=1032.6202 | train mmd=0.2708 | test_mmd=0.0439\n",
      "[CellOT] epoch=1050 f_loss=-62.5670 g_loss=845.3260 | train mmd=0.4211 | test_mmd=0.0120\n",
      "[CellOT] epoch=1100 f_loss=78.8747 g_loss=702.9385 | train mmd=0.2861 | test_mmd=0.0134\n",
      "[CellOT] epoch=1150 f_loss=261.3566 g_loss=392.7169 | train mmd=0.2548 | test_mmd=0.0553\n",
      "[CellOT] epoch=1200 f_loss=39.0707 g_loss=539.5664 | train mmd=0.4110 | test_mmd=0.1936\n",
      "[CellOT] epoch=1250 f_loss=119.5190 g_loss=616.9396 | train mmd=0.2664 | test_mmd=0.0347\n",
      "[CellOT] epoch=1300 f_loss=80.6303 g_loss=697.6652 | train mmd=0.2703 | test_mmd=0.1003\n",
      "[CellOT] epoch=1350 f_loss=67.8602 g_loss=775.3306 | train mmd=0.3552 | test_mmd=0.2810\n",
      "[CellOT] epoch=1400 f_loss=15.5616 g_loss=829.2929 | train mmd=0.2547 | test_mmd=0.0240\n",
      "[CellOT] epoch=1450 f_loss=20.8528 g_loss=885.6523 | train mmd=0.3289 | test_mmd=0.1575\n",
      "[CellOT] epoch=1500 f_loss=53.4044 g_loss=916.6874 | train mmd=0.2971 | test_mmd=0.0282\n",
      "[CellOT] epoch=1550 f_loss=19.1017 g_loss=971.6868 | train mmd=0.4340 | test_mmd=0.1803\n",
      "[CellOT] epoch=1600 f_loss=12.4987 g_loss=998.2148 | train mmd=0.2322 | test_mmd=0.0311\n",
      "[CellOT] epoch=1650 f_loss=17.7271 g_loss=1039.1277 | train mmd=0.2124 | test_mmd=0.0985\n",
      "[CellOT] epoch=1700 f_loss=37.6165 g_loss=1064.6921 | train mmd=0.1522 | test_mmd=0.0595\n",
      "[CellOT] epoch=1750 f_loss=12.2927 g_loss=1100.2910 | train mmd=0.1797 | test_mmd=0.0415\n",
      "[CellOT] epoch=1800 f_loss=16.4562 g_loss=1123.0483 | train mmd=0.2926 | test_mmd=0.0356\n",
      "[CellOT] epoch=1850 f_loss=9.0070 g_loss=1132.2051 | train mmd=0.2338 | test_mmd=0.1328\n",
      "[CellOT] epoch=1900 f_loss=4.2468 g_loss=1144.0605 | train mmd=0.2364 | test_mmd=0.0770\n",
      "[CellOT] epoch=1950 f_loss=7.1816 g_loss=1149.1735 | train mmd=0.3172 | test_mmd=0.2287\n",
      "[CellOT] epoch=2000 f_loss=5.0599 g_loss=1160.9525 | train mmd=0.6624 | test_mmd=0.2933\n",
      "[CellOT] Final CellOT MMD: 0.3296\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n",
      "[CellOT] epoch=0 f_loss=-4022479.0000 g_loss=5019536.0000 | train mmd=0.1194 | test_mmd=0.8035\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 6 metrics: {'mmd2_gamma_median': 0.2933400267663404, 'mmd2_gamma_0.5': 0.7994999183804564, 'mmd2_gamma_1.0': 0.6565100073752622, 'wasserstein_distance': 2.2205631960640786, 'R2_feature_means': 0.4671581950199334}\n",
      "**************** Run: 7 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=50 f_loss=292.7917 g_loss=214.2929 | train mmd=1.0902 | test_mmd=0.4783\n",
      "[CellOT] epoch=100 f_loss=-377.2357 g_loss=455.2849 | train mmd=1.1026 | test_mmd=0.4541\n",
      "[CellOT] epoch=150 f_loss=-568.6565 g_loss=667.0499 | train mmd=0.9637 | test_mmd=0.3747\n",
      "[CellOT] epoch=200 f_loss=-753.6572 g_loss=820.9336 | train mmd=0.9040 | test_mmd=0.3296\n",
      "[CellOT] epoch=250 f_loss=-879.0084 g_loss=1091.0347 | train mmd=0.8652 | test_mmd=0.2763\n",
      "[CellOT] epoch=300 f_loss=-1187.5549 g_loss=1347.8577 | train mmd=1.0384 | test_mmd=0.3213\n",
      "[CellOT] epoch=350 f_loss=-1256.6246 g_loss=1501.8817 | train mmd=0.9038 | test_mmd=0.2666\n",
      "[CellOT] epoch=400 f_loss=-1381.3774 g_loss=1519.9766 | train mmd=0.9251 | test_mmd=0.2337\n",
      "[CellOT] epoch=450 f_loss=-1603.9512 g_loss=1883.5952 | train mmd=0.9904 | test_mmd=0.2428\n",
      "[CellOT] epoch=500 f_loss=-1731.8359 g_loss=2076.9692 | train mmd=1.0136 | test_mmd=0.2262\n",
      "[CellOT] epoch=550 f_loss=-1804.9048 g_loss=2114.5750 | train mmd=0.9289 | test_mmd=0.1743\n",
      "[CellOT] epoch=600 f_loss=-1817.3363 g_loss=2287.7915 | train mmd=0.9342 | test_mmd=0.1703\n",
      "[CellOT] epoch=650 f_loss=-1822.0203 g_loss=2298.6548 | train mmd=0.9506 | test_mmd=0.1604\n",
      "[CellOT] epoch=700 f_loss=-1803.9766 g_loss=2536.4414 | train mmd=0.9065 | test_mmd=0.1392\n",
      "[CellOT] epoch=750 f_loss=-1739.1036 g_loss=2548.3103 | train mmd=0.8742 | test_mmd=0.1095\n",
      "[CellOT] epoch=800 f_loss=-1738.0310 g_loss=2605.2271 | train mmd=0.7971 | test_mmd=0.0910\n",
      "[CellOT] epoch=850 f_loss=-1195.5959 g_loss=2209.2063 | train mmd=0.5163 | test_mmd=0.0390\n",
      "[CellOT] epoch=900 f_loss=-1686.2812 g_loss=2640.5273 | train mmd=0.7822 | test_mmd=0.0729\n",
      "[CellOT] epoch=950 f_loss=-1431.3032 g_loss=2302.0974 | train mmd=0.6686 | test_mmd=0.0491\n",
      "[CellOT] epoch=1000 f_loss=-1471.1887 g_loss=2725.2759 | train mmd=0.7599 | test_mmd=0.0576\n",
      "[CellOT] epoch=1050 f_loss=-1186.9818 g_loss=2572.9456 | train mmd=0.6848 | test_mmd=0.0369\n",
      "[CellOT] epoch=1100 f_loss=-865.2893 g_loss=2608.7563 | train mmd=0.4936 | test_mmd=0.0182\n",
      "[CellOT] epoch=1150 f_loss=-580.9549 g_loss=1709.7153 | train mmd=0.4703 | test_mmd=0.0119\n",
      "[CellOT] epoch=1200 f_loss=-384.4820 g_loss=2286.3384 | train mmd=0.4301 | test_mmd=0.0134\n",
      "[CellOT] epoch=1250 f_loss=-267.1249 g_loss=1923.6309 | train mmd=0.4889 | test_mmd=0.0187\n",
      "[CellOT] epoch=1300 f_loss=-215.0529 g_loss=1758.6602 | train mmd=0.5013 | test_mmd=0.0162\n",
      "[CellOT] epoch=1350 f_loss=394.4164 g_loss=1299.0876 | train mmd=0.3402 | test_mmd=0.0117\n",
      "[CellOT] epoch=1400 f_loss=561.3389 g_loss=952.8909 | train mmd=0.3379 | test_mmd=0.0200\n",
      "[CellOT] epoch=1450 f_loss=772.1964 g_loss=743.0863 | train mmd=0.2800 | test_mmd=0.0362\n",
      "[CellOT] epoch=1500 f_loss=436.7231 g_loss=670.8518 | train mmd=0.2525 | test_mmd=0.0284\n",
      "[CellOT] epoch=1550 f_loss=483.0302 g_loss=899.3469 | train mmd=0.2116 | test_mmd=0.0230\n",
      "[CellOT] epoch=1600 f_loss=234.0347 g_loss=871.1824 | train mmd=0.2809 | test_mmd=0.1425\n",
      "[CellOT] epoch=1650 f_loss=102.8019 g_loss=1163.9672 | train mmd=0.4168 | test_mmd=0.1390\n",
      "[CellOT] epoch=1700 f_loss=181.6054 g_loss=1127.7153 | train mmd=0.2559 | test_mmd=0.0642\n",
      "[CellOT] epoch=1750 f_loss=171.8699 g_loss=1210.4010 | train mmd=0.3324 | test_mmd=0.0183\n",
      "[CellOT] epoch=1800 f_loss=119.6511 g_loss=1249.2574 | train mmd=0.3152 | test_mmd=0.0518\n",
      "[CellOT] epoch=1850 f_loss=55.4952 g_loss=1304.6436 | train mmd=0.4711 | test_mmd=0.0904\n",
      "[CellOT] epoch=1900 f_loss=55.9029 g_loss=1555.3475 | train mmd=0.5068 | test_mmd=0.3504\n",
      "[CellOT] epoch=1950 f_loss=105.1423 g_loss=1396.5403 | train mmd=0.4350 | test_mmd=0.1535\n",
      "[CellOT] epoch=2000 f_loss=82.8260 g_loss=1391.5500 | train mmd=0.2037 | test_mmd=0.0314\n",
      "[CellOT] Final CellOT MMD: 0.1039\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n",
      "[CellOT] epoch=0 f_loss=-2975184.7500 g_loss=3337473.5000 | train mmd=0.1194 | test_mmd=0.8035\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 7 metrics: {'mmd2_gamma_median': 0.03143173711348313, 'mmd2_gamma_0.5': 0.14231245219023858, 'mmd2_gamma_1.0': 0.21344213636028736, 'wasserstein_distance': 1.3724629202130083, 'R2_feature_means': 0.9198471817308377}\n",
      "**************** Run: 8 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=50 f_loss=267.3444 g_loss=239.0335 | train mmd=1.0984 | test_mmd=0.4770\n",
      "[CellOT] epoch=100 f_loss=-408.4763 g_loss=498.8193 | train mmd=1.1003 | test_mmd=0.4439\n",
      "[CellOT] epoch=150 f_loss=-527.1819 g_loss=632.3821 | train mmd=0.7694 | test_mmd=0.3385\n",
      "[CellOT] epoch=200 f_loss=-867.6202 g_loss=973.2981 | train mmd=0.9836 | test_mmd=0.3370\n",
      "[CellOT] epoch=250 f_loss=-1084.7754 g_loss=1124.6013 | train mmd=1.0089 | test_mmd=0.3114\n",
      "[CellOT] epoch=300 f_loss=-1064.8107 g_loss=1054.4296 | train mmd=0.8010 | test_mmd=0.2282\n",
      "[CellOT] epoch=350 f_loss=-1314.7800 g_loss=1532.6914 | train mmd=0.9754 | test_mmd=0.2576\n",
      "[CellOT] epoch=400 f_loss=-1214.6750 g_loss=1654.1409 | train mmd=0.6401 | test_mmd=0.1879\n",
      "[CellOT] epoch=450 f_loss=-1599.7610 g_loss=1731.5620 | train mmd=0.9562 | test_mmd=0.2167\n",
      "[CellOT] epoch=500 f_loss=-379.4892 g_loss=1737.3503 | train mmd=0.2323 | test_mmd=0.4748\n",
      "[CellOT] epoch=550 f_loss=-1489.8427 g_loss=2002.0403 | train mmd=0.8026 | test_mmd=0.1566\n",
      "[CellOT] epoch=600 f_loss=-1667.4036 g_loss=1905.7499 | train mmd=0.8194 | test_mmd=0.1490\n",
      "[CellOT] epoch=650 f_loss=-1804.4392 g_loss=2299.6509 | train mmd=0.8450 | test_mmd=0.1389\n",
      "[CellOT] epoch=700 f_loss=-1632.9286 g_loss=2422.9639 | train mmd=0.7483 | test_mmd=0.1097\n",
      "[CellOT] epoch=750 f_loss=-1719.2478 g_loss=2260.7253 | train mmd=0.7526 | test_mmd=0.1059\n",
      "[CellOT] epoch=800 f_loss=-1645.2245 g_loss=2602.8477 | train mmd=0.6847 | test_mmd=0.0802\n",
      "[CellOT] epoch=850 f_loss=-1756.4897 g_loss=2109.2439 | train mmd=0.6881 | test_mmd=0.0803\n",
      "[CellOT] epoch=900 f_loss=-1672.4958 g_loss=2354.3066 | train mmd=0.5417 | test_mmd=0.0450\n",
      "[CellOT] epoch=950 f_loss=-1372.0105 g_loss=2352.0925 | train mmd=0.5713 | test_mmd=0.0503\n",
      "[CellOT] epoch=1000 f_loss=-1407.0371 g_loss=2478.2261 | train mmd=0.6004 | test_mmd=0.0452\n",
      "[CellOT] epoch=1050 f_loss=-1354.1062 g_loss=2565.2300 | train mmd=0.6707 | test_mmd=0.0533\n",
      "[CellOT] epoch=1100 f_loss=-1044.6313 g_loss=2331.2786 | train mmd=0.5814 | test_mmd=0.0366\n",
      "[CellOT] epoch=1150 f_loss=-1303.0425 g_loss=2477.9412 | train mmd=0.6285 | test_mmd=0.0347\n",
      "[CellOT] epoch=1200 f_loss=-628.4044 g_loss=2363.5234 | train mmd=0.5087 | test_mmd=0.0274\n",
      "[CellOT] epoch=1250 f_loss=-558.6964 g_loss=2434.9600 | train mmd=0.5358 | test_mmd=0.0217\n",
      "[CellOT] epoch=1300 f_loss=-390.6727 g_loss=2194.4199 | train mmd=0.4869 | test_mmd=0.0194\n",
      "[CellOT] epoch=1350 f_loss=107.5596 g_loss=1790.4471 | train mmd=0.2933 | test_mmd=0.0109\n",
      "[CellOT] epoch=1400 f_loss=-43.6850 g_loss=1687.4454 | train mmd=0.3860 | test_mmd=0.0097\n",
      "[CellOT] epoch=1450 f_loss=-1651.9744 g_loss=1239.0653 | train mmd=0.1612 | test_mmd=0.1697\n",
      "[CellOT] epoch=1500 f_loss=442.8248 g_loss=1262.6519 | train mmd=0.3742 | test_mmd=0.0095\n",
      "[CellOT] epoch=1550 f_loss=442.0745 g_loss=1499.3263 | train mmd=0.4543 | test_mmd=0.0149\n",
      "[CellOT] epoch=1600 f_loss=615.2669 g_loss=1277.0728 | train mmd=0.2797 | test_mmd=0.0247\n",
      "[CellOT] epoch=1650 f_loss=707.1146 g_loss=1637.9081 | train mmd=0.6270 | test_mmd=0.2831\n",
      "[CellOT] epoch=1700 f_loss=806.9130 g_loss=1559.6434 | train mmd=0.1689 | test_mmd=0.0338\n",
      "[CellOT] epoch=1750 f_loss=504.0480 g_loss=1908.8406 | train mmd=0.1971 | test_mmd=0.2112\n",
      "[CellOT] epoch=1800 f_loss=365.7894 g_loss=2407.7729 | train mmd=0.2495 | test_mmd=0.0682\n",
      "[CellOT] epoch=1850 f_loss=289.6008 g_loss=3213.1255 | train mmd=0.2413 | test_mmd=0.1346\n",
      "[CellOT] epoch=1900 f_loss=-161.0562 g_loss=3538.5420 | train mmd=0.7208 | test_mmd=0.2321\n",
      "[CellOT] epoch=1950 f_loss=303.8571 g_loss=3733.9304 | train mmd=0.2283 | test_mmd=0.1792\n",
      "[CellOT] epoch=2000 f_loss=210.0379 g_loss=3786.2612 | train mmd=0.3215 | test_mmd=0.0237\n",
      "[CellOT] Final CellOT MMD: 0.1674\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n",
      "[CellOT] epoch=0 f_loss=-2348696.0000 g_loss=2440465.2500 | train mmd=0.1194 | test_mmd=0.8035\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 8 metrics: {'mmd2_gamma_median': 0.023672708505664275, 'mmd2_gamma_0.5': 0.2183586559247176, 'mmd2_gamma_1.0': 0.3423260496624505, 'wasserstein_distance': 1.5564991837948456, 'R2_feature_means': 0.9608458246834212}\n",
      "**************** Run: 9 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=50 f_loss=206.3362 g_loss=224.3822 | train mmd=1.0996 | test_mmd=0.4696\n",
      "[CellOT] epoch=100 f_loss=-374.6874 g_loss=453.0627 | train mmd=1.0990 | test_mmd=0.4300\n",
      "[CellOT] epoch=150 f_loss=-545.1318 g_loss=593.6744 | train mmd=0.8501 | test_mmd=0.3390\n",
      "[CellOT] epoch=200 f_loss=-767.4663 g_loss=877.6129 | train mmd=0.9503 | test_mmd=0.3311\n",
      "[CellOT] epoch=250 f_loss=-1005.4695 g_loss=1121.4001 | train mmd=1.0348 | test_mmd=0.3259\n",
      "[CellOT] epoch=300 f_loss=-1169.2090 g_loss=1316.5654 | train mmd=1.0376 | test_mmd=0.2967\n",
      "[CellOT] epoch=350 f_loss=-1232.0183 g_loss=1438.5402 | train mmd=1.0044 | test_mmd=0.2597\n",
      "[CellOT] epoch=400 f_loss=-1412.2026 g_loss=1650.6383 | train mmd=1.0155 | test_mmd=0.2495\n",
      "[CellOT] epoch=450 f_loss=-1407.8982 g_loss=1779.7078 | train mmd=0.8563 | test_mmd=0.1877\n",
      "[CellOT] epoch=500 f_loss=-1070.4270 g_loss=1867.9447 | train mmd=0.5398 | test_mmd=0.1142\n",
      "[CellOT] epoch=550 f_loss=-1505.2593 g_loss=1988.9412 | train mmd=0.8993 | test_mmd=0.1594\n",
      "[CellOT] epoch=600 f_loss=-1610.5153 g_loss=2204.0239 | train mmd=0.8726 | test_mmd=0.1429\n",
      "[CellOT] epoch=650 f_loss=-1669.8557 g_loss=2254.2141 | train mmd=0.8660 | test_mmd=0.1246\n",
      "[CellOT] epoch=700 f_loss=-1659.9985 g_loss=2389.8882 | train mmd=0.9039 | test_mmd=0.1164\n",
      "[CellOT] epoch=750 f_loss=-1393.3451 g_loss=2330.8281 | train mmd=0.7804 | test_mmd=0.0854\n",
      "[CellOT] epoch=800 f_loss=-1483.9126 g_loss=2417.4775 | train mmd=0.7944 | test_mmd=0.0765\n",
      "[CellOT] epoch=850 f_loss=-1280.7773 g_loss=2411.5830 | train mmd=0.7814 | test_mmd=0.0652\n",
      "[CellOT] epoch=900 f_loss=-1050.3999 g_loss=2316.7959 | train mmd=0.6835 | test_mmd=0.0460\n",
      "[CellOT] epoch=950 f_loss=-939.5017 g_loss=2269.1147 | train mmd=0.6164 | test_mmd=0.0359\n",
      "[CellOT] epoch=1000 f_loss=-932.8046 g_loss=2234.3755 | train mmd=0.6305 | test_mmd=0.0330\n",
      "[CellOT] epoch=1050 f_loss=-912.6646 g_loss=2334.0762 | train mmd=0.6305 | test_mmd=0.0300\n",
      "[CellOT] epoch=1100 f_loss=-219.2904 g_loss=2049.2764 | train mmd=0.4920 | test_mmd=0.0165\n",
      "[CellOT] epoch=1150 f_loss=-246.7243 g_loss=1448.7251 | train mmd=0.5058 | test_mmd=0.0151\n",
      "[CellOT] epoch=1200 f_loss=-205.8655 g_loss=2006.7512 | train mmd=0.4963 | test_mmd=0.0119\n",
      "[CellOT] epoch=1250 f_loss=142.6194 g_loss=1709.0718 | train mmd=0.4649 | test_mmd=0.0147\n",
      "[CellOT] epoch=1300 f_loss=877.5183 g_loss=1086.0734 | train mmd=0.2383 | test_mmd=0.0216\n",
      "[CellOT] epoch=1350 f_loss=645.3527 g_loss=1142.0322 | train mmd=0.4100 | test_mmd=0.0195\n",
      "[CellOT] epoch=1400 f_loss=337.8253 g_loss=1275.0632 | train mmd=0.2474 | test_mmd=0.0324\n",
      "[CellOT] epoch=1450 f_loss=266.0600 g_loss=3268.3169 | train mmd=0.3637 | test_mmd=0.3054\n",
      "[CellOT] epoch=1500 f_loss=260.1302 g_loss=3202.2441 | train mmd=0.4000 | test_mmd=0.0233\n",
      "[CellOT] epoch=1550 f_loss=159.2489 g_loss=3020.8369 | train mmd=0.2824 | test_mmd=0.0333\n",
      "[CellOT] epoch=1600 f_loss=307.5273 g_loss=4183.6885 | train mmd=0.3409 | test_mmd=0.1434\n",
      "[CellOT] epoch=1650 f_loss=142.1304 g_loss=4056.5718 | train mmd=0.5276 | test_mmd=0.1431\n",
      "[CellOT] epoch=1700 f_loss=-14.0563 g_loss=4744.6982 | train mmd=0.2318 | test_mmd=0.0884\n",
      "[CellOT] epoch=1750 f_loss=112.0592 g_loss=4970.7344 | train mmd=0.3861 | test_mmd=0.0531\n",
      "[CellOT] epoch=1800 f_loss=-111.4049 g_loss=5829.7080 | train mmd=0.2263 | test_mmd=0.0796\n",
      "[CellOT] epoch=1850 f_loss=255.7552 g_loss=5323.2998 | train mmd=0.2265 | test_mmd=0.0670\n",
      "[CellOT] epoch=1900 f_loss=96.7364 g_loss=5772.7729 | train mmd=0.3011 | test_mmd=0.0330\n",
      "[CellOT] epoch=1950 f_loss=-737.7509 g_loss=5759.3398 | train mmd=0.5344 | test_mmd=0.1151\n",
      "[CellOT] epoch=2000 f_loss=-610.1172 g_loss=6288.3271 | train mmd=0.5128 | test_mmd=0.1587\n",
      "[CellOT] Final CellOT MMD: 0.2372\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/umap/umap_.py:1952: UserWarning: n_jobs value 1 overridden to 1 by setting random_state. Use no seed for parallelism.\n",
      "  warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 9 metrics: {'mmd2_gamma_median': 0.15868687203566734, 'mmd2_gamma_0.5': 0.5510471192753064, 'mmd2_gamma_1.0': 0.5198928655391812, 'wasserstein_distance': 1.8520064378430123, 'R2_feature_means': 0.5120801398631696}\n",
      "                        mean     std\n",
      "mmd2_gamma_median     0.0995  0.1007\n",
      "mmd2_gamma_0.5        0.3963  0.2855\n",
      "mmd2_gamma_1.0        0.4481  0.2335\n",
      "wasserstein_distance  1.6802  0.3099\n",
      "R2_feature_means      0.7972  0.2037\n"
     ]
    },
    {
     "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/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": "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BoFAIBCkIQxDCdm0aRMrV67E4/HQ0NDAhg0b2LdvX6nFKjo/+MEPkCSJW2+9tdSiFIXOzk6++tWvUltbi8vlYtmyZbz55pulFmvSMQyDu+66i5aWFlwuFwsXLuTuu+/Oq6nbdOOVV17h8ssvp6mpCUmSeOqpp9KetyyL7373u8yePRuXy8VFF13EgQMHSiNsFoRhKCE7duzgxhtv5LXXXuOFF15A0zQ+//nPEwwGSy1a0WhtbeVXv/oVZ511VqlFKQr9/f2sWbMGm83Gtm3b+OCDD/jxj39MdXV1qUWbdO6//342b97Mz3/+cz788EPuv/9+fvjDH/Kzn/2s1KIVnGAwyNlnn82//uu/Zn3+hz/8IT/96U/55S9/yeuvv05ZWRkXX3wxkUikyJLmwBJMGXp6eizA2rFjR6lFKQqDg4PWokWLrBdeeMG68MILrVtuuaXUIk06d9xxh7V27dpSi1ESLr30Uuv6669Pe+zLX/6ytXHjxhJJVBwA68knn0z+2zRNq7Gx0frRj36UfGxgYMByOBzWY489VgIJhyM8himEz+cDoKampsSSFIcbb7yRSy+9lIsuuqjUohSNrVu3smLFCq666ioaGhpYvnw5v/71r0stVlFYvXo1L730Evv37wfgnXfeYefOnVxyySUllqy4tLW10dXVlXbfV1ZWct555/Hqq6+WULKTTLnuqjMV0zS59dZbWbNmDUuXLi21OJPO448/zttvv01ra2upRSkqhw8fZvPmzdx+++18+9vfprW1lZtvvhm73c51111XavEmlTvvvBO/38/ixYtRFAXDMLj33nvZuHFjqUUrKl1dXQDMmjUr7fFZs2Ylnys1wjBMEW688Ubee+89du7cWWpRJp2jR49yyy238MILL+B0OkstTlExTZMVK1Zw3333AbB8+XLee+89fvnLX37iDcNvf/tbHnnkER599FHOPPNM9uzZw6233kpTU9Mn/tynGyKUNAW46aab+P3vf8/LL788I2ZRvPXWW/T09HDuueeiqiqqqrJjxw5++tOfoqoqhmGUWsRJY/bs2SxZsiTtsTPOOIMjR46USKLi8fd///fceeedfOUrX2HZsmVce+213HbbbWzatKnUohWVxsZGALq7u9Me7+7uTj5XaoRhKCGWZXHTTTfx5JNPsn37dlpaWkotUlFYv349e/fuZc+ePcm/FStWsHHjRvbs2YOiKKUWcdJYs2bNsJTk/fv3M3/+/BJJVDxCodCwyWGKomCaZokkKg0tLS00Njby0ksvJR/z+/28/vrrrFq1qoSSnUSEkkrIjTfeyKOPPsrTTz+Nx+NJxhcrKytxuVwllm7y8Hg8w/ZRysrKqK2t/cTvr9x2222sXr2a++67j6uvvpo33niDLVu2sGXLllKLNulcfvnl3HvvvcybN48zzzyT3bt388ADD3D99deXWrSCEwgEOHjwYPLfbW1t7Nmzh5qaGubNm8ett97KPffcw6JFi2hpaeGuu+6iqamJDRs2lE7oVEqdFjWTAbL+Pfjgg6UWrejMlHRVy7KsZ555xlq6dKnlcDisxYsXW1u2bCm1SEXB7/dbt9xyizVv3jzL6XRap5xyivWP//iPVjQaLbVoBefll1/O+tu+7rrrLMuKp6zedddd1qxZsyyHw2GtX7/e2rdvX2mFTkHMYxAIBAJBGmKPQSAQCARpCMMgEAgEgjSEYRAIBAJBGsIwCAQCgSANYRgEAoFAkIYwDAKBQCBIQxgGgUAgEKQhDINAIBAI0hCGQSAQCARpCMMgEAgEgjSEYRAIBAJBGv8fqLg/zeq5Se4AAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 400x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "drug = \"Vem\"\n",
    "X_pre, X_post = prepare_pair_from_mat('COLO858', 'DMSO','24h', drug, '72h')\n",
    "jfe_indices = [1, 6, 0, 5, 4, 7, 8, 2, 3, 19]  \n",
    "\n",
    "print(\"X_pre cells:\", X_pre.shape)\n",
    "print(\"X_post cells:\", X_post.shape)\n",
    "\n",
    "X_tr_pre, X_te_pre, Y_tr_post, Y_te_post = split_train_test(X_pre, X_post, 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",
    "# 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",
    "\n",
    "\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 = run_cellot_pair(X_tr_pre[:, jfe_indices], Y_tr_post[:, jfe_indices], X_te_pre[:, jfe_indices], Y_te_post[:, jfe_indices], n_epochs=2000)\n",
    "    metrics = summarize_metrics(out[\"y_pred\"], Y_te_post[:, jfe_indices], median_gamma)\n",
    "    print(f\"Run {run} metrics: {metrics}\")\n",
    "    all_metrics.append(metrics)\n",
    "\n",
    "# Results summary\n",
    "df = pd.DataFrame(all_metrics)\n",
    "print(df.describe().T[['mean', 'std']].round(4))\n",
    "\n",
    "\n",
    "from umap import UMAP\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "source = Y_tr_post[:, jfe_indices]\n",
    "target = Y_te_post[:, jfe_indices]\n",
    "predicted = out.get('y_pred') \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([source, target]))\n",
    "source_umap = umap_model.transform(source)\n",
    "target_umap = umap_model.transform(target)\n",
    "y_pred_umap = umap_model.transform(predicted)\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=\"#C88131\")\n",
    "ax.scatter(y_pred_umap[:, 0], y_pred_umap[:, 1], s=10, alpha=0.7, label='predicted cells', color=\"#1F4D8D\")\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/cellot_on_4i_drug_{drug}.png\", dpi=300)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c338b528",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "31734785",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Cell line:  WM902B\n",
      "['DMSO' 'Vem' 'Vem+Tram']\n",
      "X_pre cells: (5690, 20)\n",
      "X_post cells: (5690, 20)\n",
      "(4552, 20)\n",
      "(1138, 20)\n",
      "(4552, 20)\n",
      "(1138, 20)\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Median heuristic gamma: 0.06456030933401641\n",
      "**************** Run: 0 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=0 f_loss=-3355624.0000 g_loss=3746488.5000 | train mmd=0.1601 | test_mmd=0.7867\n",
      "[CellOT] epoch=50 f_loss=333.7708 g_loss=222.1668 | train mmd=1.0116 | test_mmd=0.3859\n",
      "[CellOT] epoch=100 f_loss=-353.9976 g_loss=380.7114 | train mmd=1.0782 | test_mmd=0.3928\n",
      "[CellOT] epoch=150 f_loss=-553.4442 g_loss=602.5024 | train mmd=0.9788 | test_mmd=0.3278\n",
      "[CellOT] epoch=200 f_loss=-747.8425 g_loss=769.8306 | train mmd=0.9830 | test_mmd=0.3216\n",
      "[CellOT] epoch=250 f_loss=-1222.3127 g_loss=875.1893 | train mmd=0.4355 | test_mmd=0.3517\n",
      "[CellOT] epoch=300 f_loss=-1021.9438 g_loss=987.0919 | train mmd=0.9042 | test_mmd=0.2553\n",
      "[CellOT] epoch=350 f_loss=-1007.6368 g_loss=1563.5920 | train mmd=0.7365 | test_mmd=0.2038\n",
      "[CellOT] epoch=400 f_loss=-1224.8462 g_loss=1320.4329 | train mmd=0.7786 | test_mmd=0.1890\n",
      "[CellOT] epoch=450 f_loss=-1439.9064 g_loss=1574.4694 | train mmd=0.8283 | test_mmd=0.1974\n",
      "[CellOT] epoch=500 f_loss=-1539.9265 g_loss=1761.7930 | train mmd=0.8322 | test_mmd=0.1846\n",
      "[CellOT] epoch=550 f_loss=-1576.5948 g_loss=2100.2749 | train mmd=0.6951 | test_mmd=0.1336\n",
      "[CellOT] epoch=600 f_loss=-1761.8350 g_loss=1947.7341 | train mmd=0.7364 | test_mmd=0.1336\n",
      "[CellOT] epoch=650 f_loss=-1946.8402 g_loss=1716.1530 | train mmd=0.4402 | test_mmd=0.1273\n",
      "[CellOT] epoch=700 f_loss=-1689.0154 g_loss=2221.2329 | train mmd=0.7368 | test_mmd=0.1206\n",
      "[CellOT] epoch=750 f_loss=-1578.3660 g_loss=3004.2725 | train mmd=0.4871 | test_mmd=0.1248\n",
      "[CellOT] epoch=800 f_loss=-1870.1047 g_loss=2276.5366 | train mmd=0.5597 | test_mmd=0.0807\n",
      "[CellOT] epoch=850 f_loss=-1842.5024 g_loss=2168.1514 | train mmd=0.6506 | test_mmd=0.0780\n",
      "[CellOT] epoch=900 f_loss=-1694.5193 g_loss=2418.6704 | train mmd=0.5444 | test_mmd=0.0569\n",
      "[CellOT] epoch=950 f_loss=-1872.4347 g_loss=2606.6465 | train mmd=0.6142 | test_mmd=0.0650\n",
      "[CellOT] epoch=1000 f_loss=-1454.0654 g_loss=2345.0378 | train mmd=0.4263 | test_mmd=0.0332\n",
      "[CellOT] epoch=1050 f_loss=-1303.3510 g_loss=1432.5454 | train mmd=0.2364 | test_mmd=0.1063\n",
      "[CellOT] epoch=1100 f_loss=-2038.8220 g_loss=1555.1451 | train mmd=0.2026 | test_mmd=0.0856\n",
      "[CellOT] epoch=1150 f_loss=-1106.8733 g_loss=2238.9883 | train mmd=0.4557 | test_mmd=0.0261\n",
      "[CellOT] epoch=1200 f_loss=-1073.7209 g_loss=1651.9576 | train mmd=0.1531 | test_mmd=0.0305\n",
      "[CellOT] epoch=1250 f_loss=-442.3487 g_loss=1734.8228 | train mmd=0.3251 | test_mmd=0.0391\n",
      "[CellOT] epoch=1300 f_loss=231.4848 g_loss=1812.1368 | train mmd=0.2474 | test_mmd=0.0227\n",
      "[CellOT] epoch=1350 f_loss=-309.5928 g_loss=1132.1133 | train mmd=0.3209 | test_mmd=0.0352\n",
      "[CellOT] epoch=1400 f_loss=-1148.3132 g_loss=516.0500 | train mmd=0.2766 | test_mmd=0.0906\n",
      "[CellOT] epoch=1450 f_loss=248.4359 g_loss=738.0060 | train mmd=0.2397 | test_mmd=0.0258\n",
      "[CellOT] epoch=1500 f_loss=377.7561 g_loss=1475.5526 | train mmd=0.1956 | test_mmd=0.0472\n",
      "[CellOT] epoch=1550 f_loss=677.1317 g_loss=1416.0686 | train mmd=0.1497 | test_mmd=0.0270\n",
      "[CellOT] epoch=1600 f_loss=402.1601 g_loss=2546.8970 | train mmd=0.2733 | test_mmd=0.0540\n",
      "[CellOT] epoch=1650 f_loss=410.8057 g_loss=2310.3374 | train mmd=0.2099 | test_mmd=0.0375\n",
      "[CellOT] epoch=1700 f_loss=195.3062 g_loss=2478.3877 | train mmd=0.2477 | test_mmd=0.0261\n",
      "[CellOT] epoch=1750 f_loss=-231.9660 g_loss=3104.4604 | train mmd=0.2705 | test_mmd=0.0625\n",
      "[CellOT] epoch=1800 f_loss=-988.5690 g_loss=3206.4509 | train mmd=0.4098 | test_mmd=0.0923\n",
      "[CellOT] epoch=1850 f_loss=197.4088 g_loss=3787.4097 | train mmd=0.5045 | test_mmd=0.2138\n",
      "[CellOT] epoch=1900 f_loss=224.5607 g_loss=3631.2820 | train mmd=0.4758 | test_mmd=0.1422\n",
      "[CellOT] epoch=1950 f_loss=156.2908 g_loss=3624.8682 | train mmd=0.2562 | test_mmd=0.0687\n",
      "[CellOT] epoch=2000 f_loss=158.8407 g_loss=3891.7102 | train mmd=0.2765 | test_mmd=0.0228\n",
      "[CellOT] Final CellOT MMD: 0.1378\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 0 metrics: {'mmd2_gamma_median': 0.0228316173429175, 'mmd2_gamma_0.5': 0.1774883102596979, 'mmd2_gamma_1.0': 0.2762745201154747, 'wasserstein_distance': 1.2717783514075056, 'R2_feature_means': 0.9828502127731991}\n",
      "**************** Run: 1 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=0 f_loss=-2720677.0000 g_loss=3092151.0000 | train mmd=0.1601 | test_mmd=0.7867\n",
      "[CellOT] epoch=50 f_loss=-5.2138 g_loss=282.3934 | train mmd=0.5492 | test_mmd=0.2800\n",
      "[CellOT] epoch=100 f_loss=-310.0056 g_loss=442.9417 | train mmd=1.0258 | test_mmd=0.3759\n",
      "[CellOT] epoch=150 f_loss=-532.9669 g_loss=544.6382 | train mmd=0.9971 | test_mmd=0.3510\n",
      "[CellOT] epoch=200 f_loss=-804.6294 g_loss=742.5805 | train mmd=0.5925 | test_mmd=0.2491\n",
      "[CellOT] epoch=250 f_loss=-863.2524 g_loss=884.1175 | train mmd=0.8449 | test_mmd=0.2562\n",
      "[CellOT] epoch=300 f_loss=-1173.9114 g_loss=1076.5608 | train mmd=0.7368 | test_mmd=0.2385\n",
      "[CellOT] epoch=350 f_loss=-1279.1575 g_loss=1363.8320 | train mmd=0.8343 | test_mmd=0.2252\n",
      "[CellOT] epoch=400 f_loss=-1726.0999 g_loss=4739.1592 | train mmd=0.3783 | test_mmd=0.3016\n",
      "[CellOT] epoch=450 f_loss=-1652.4561 g_loss=1596.6509 | train mmd=0.7504 | test_mmd=0.1710\n",
      "[CellOT] epoch=500 f_loss=-1573.1339 g_loss=1801.2411 | train mmd=0.7371 | test_mmd=0.1568\n",
      "[CellOT] epoch=550 f_loss=-1897.0985 g_loss=1991.4930 | train mmd=0.8103 | test_mmd=0.1617\n",
      "[CellOT] epoch=600 f_loss=-2093.0967 g_loss=2377.7070 | train mmd=0.7865 | test_mmd=0.1481\n",
      "[CellOT] epoch=650 f_loss=-1988.8484 g_loss=2711.5737 | train mmd=0.6844 | test_mmd=0.1276\n",
      "[CellOT] epoch=700 f_loss=-2370.1133 g_loss=3107.0964 | train mmd=0.7049 | test_mmd=0.1085\n",
      "[CellOT] epoch=750 f_loss=-2503.3672 g_loss=2700.1201 | train mmd=0.7333 | test_mmd=0.1071\n",
      "[CellOT] epoch=800 f_loss=-2189.3345 g_loss=2795.8804 | train mmd=0.5870 | test_mmd=0.0733\n",
      "[CellOT] epoch=850 f_loss=-2449.3037 g_loss=3198.2358 | train mmd=0.6195 | test_mmd=0.0704\n",
      "[CellOT] epoch=900 f_loss=-2479.8191 g_loss=3436.0444 | train mmd=0.6490 | test_mmd=0.0755\n",
      "[CellOT] epoch=950 f_loss=-1862.1106 g_loss=3203.1992 | train mmd=0.3848 | test_mmd=0.0385\n",
      "[CellOT] epoch=1000 f_loss=-1725.1248 g_loss=2882.7656 | train mmd=0.3618 | test_mmd=0.0226\n",
      "[CellOT] epoch=1050 f_loss=-2162.0911 g_loss=3622.5762 | train mmd=0.4741 | test_mmd=0.0309\n",
      "[CellOT] epoch=1100 f_loss=-2170.7593 g_loss=3371.6177 | train mmd=0.4508 | test_mmd=0.0264\n",
      "[CellOT] epoch=1150 f_loss=-1807.2253 g_loss=3673.2593 | train mmd=0.4042 | test_mmd=0.0219\n",
      "[CellOT] epoch=1200 f_loss=-1528.1602 g_loss=2895.8389 | train mmd=0.3236 | test_mmd=0.0417\n",
      "[CellOT] epoch=1250 f_loss=-949.4923 g_loss=1606.1248 | train mmd=0.2179 | test_mmd=0.0173\n",
      "[CellOT] epoch=1300 f_loss=-439.9180 g_loss=2124.7749 | train mmd=0.2482 | test_mmd=0.0416\n",
      "[CellOT] epoch=1350 f_loss=312.7729 g_loss=1097.6289 | train mmd=0.1823 | test_mmd=0.0705\n",
      "[CellOT] epoch=1400 f_loss=241.3018 g_loss=943.4615 | train mmd=0.1926 | test_mmd=0.0419\n",
      "[CellOT] epoch=1450 f_loss=-632.5642 g_loss=1037.2065 | train mmd=0.2932 | test_mmd=0.1172\n",
      "[CellOT] epoch=1500 f_loss=246.6732 g_loss=1073.3455 | train mmd=0.1986 | test_mmd=0.0351\n",
      "[CellOT] epoch=1550 f_loss=475.1475 g_loss=1470.0906 | train mmd=0.2519 | test_mmd=0.0788\n",
      "[CellOT] epoch=1600 f_loss=-1050.9546 g_loss=1341.9016 | train mmd=0.2467 | test_mmd=0.0496\n",
      "[CellOT] epoch=1650 f_loss=583.7865 g_loss=1601.7097 | train mmd=0.1823 | test_mmd=0.0726\n",
      "[CellOT] epoch=1700 f_loss=482.9683 g_loss=1990.7627 | train mmd=0.2787 | test_mmd=0.0629\n",
      "[CellOT] epoch=1750 f_loss=476.2921 g_loss=1952.9911 | train mmd=0.2404 | test_mmd=0.0260\n",
      "[CellOT] epoch=1800 f_loss=559.5601 g_loss=2288.0688 | train mmd=0.3002 | test_mmd=0.1232\n",
      "[CellOT] epoch=1850 f_loss=190.6223 g_loss=2189.2815 | train mmd=0.2801 | test_mmd=0.0388\n",
      "[CellOT] epoch=1900 f_loss=-412.8759 g_loss=2334.4951 | train mmd=0.2991 | test_mmd=0.1141\n",
      "[CellOT] epoch=1950 f_loss=269.6523 g_loss=2402.2375 | train mmd=0.2078 | test_mmd=0.0485\n",
      "[CellOT] epoch=2000 f_loss=303.3450 g_loss=2436.2085 | train mmd=0.3267 | test_mmd=0.0250\n",
      "[CellOT] Final CellOT MMD: 0.1695\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 1 metrics: {'mmd2_gamma_median': 0.025000217173692274, 'mmd2_gamma_0.5': 0.21331412659367222, 'mmd2_gamma_1.0': 0.33009929741306765, 'wasserstein_distance': 1.376725662459481, 'R2_feature_means': 0.9844541720146133}\n",
      "**************** Run: 2 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=0 f_loss=-2905245.5000 g_loss=3447477.2500 | train mmd=0.1601 | test_mmd=0.7867\n",
      "[CellOT] epoch=50 f_loss=519.1649 g_loss=214.1427 | train mmd=0.8588 | test_mmd=0.3663\n",
      "[CellOT] epoch=100 f_loss=-309.9214 g_loss=320.7916 | train mmd=0.9900 | test_mmd=0.3539\n",
      "[CellOT] epoch=150 f_loss=-493.4025 g_loss=504.0600 | train mmd=1.0492 | test_mmd=0.3698\n",
      "[CellOT] epoch=200 f_loss=-649.4603 g_loss=711.0432 | train mmd=0.9240 | test_mmd=0.3140\n",
      "[CellOT] epoch=250 f_loss=-817.1006 g_loss=801.4869 | train mmd=0.9067 | test_mmd=0.2692\n",
      "[CellOT] epoch=300 f_loss=-1032.6857 g_loss=1079.4613 | train mmd=0.9385 | test_mmd=0.2762\n",
      "[CellOT] epoch=350 f_loss=-1164.3910 g_loss=1202.6924 | train mmd=0.8658 | test_mmd=0.2391\n",
      "[CellOT] epoch=400 f_loss=-1300.3292 g_loss=1507.4213 | train mmd=0.8123 | test_mmd=0.2136\n",
      "[CellOT] epoch=450 f_loss=-1485.3475 g_loss=1657.9482 | train mmd=0.7943 | test_mmd=0.1770\n",
      "[CellOT] epoch=500 f_loss=-1762.0170 g_loss=2054.2466 | train mmd=0.7439 | test_mmd=0.1648\n",
      "[CellOT] epoch=550 f_loss=-1893.6194 g_loss=2275.0576 | train mmd=0.7415 | test_mmd=0.1474\n",
      "[CellOT] epoch=600 f_loss=-2050.7134 g_loss=2692.6194 | train mmd=0.5859 | test_mmd=0.1484\n",
      "[CellOT] epoch=650 f_loss=-2528.5330 g_loss=2797.6851 | train mmd=0.6029 | test_mmd=0.0926\n",
      "[CellOT] epoch=700 f_loss=-2662.9521 g_loss=3204.5989 | train mmd=0.6115 | test_mmd=0.1002\n",
      "[CellOT] epoch=750 f_loss=-2419.7632 g_loss=3344.6992 | train mmd=0.5331 | test_mmd=0.0727\n",
      "[CellOT] epoch=800 f_loss=-2477.3765 g_loss=3015.0886 | train mmd=0.4221 | test_mmd=0.0884\n",
      "[CellOT] epoch=850 f_loss=-1649.3250 g_loss=2018.4347 | train mmd=0.2642 | test_mmd=0.1927\n",
      "[CellOT] epoch=900 f_loss=-930.5891 g_loss=323.1766 | train mmd=0.1794 | test_mmd=0.4054\n",
      "[CellOT] epoch=950 f_loss=-933.4590 g_loss=1868.7249 | train mmd=0.1927 | test_mmd=0.3574\n",
      "[CellOT] epoch=1000 f_loss=-723.6962 g_loss=2552.1475 | train mmd=0.1692 | test_mmd=0.2219\n",
      "[CellOT] epoch=1050 f_loss=-294.9123 g_loss=888.2798 | train mmd=0.1446 | test_mmd=0.3950\n",
      "[CellOT] epoch=1100 f_loss=-210.6796 g_loss=860.7264 | train mmd=0.1371 | test_mmd=0.3393\n",
      "[CellOT] epoch=1150 f_loss=-656.8375 g_loss=1240.7773 | train mmd=0.1745 | test_mmd=0.2008\n",
      "[CellOT] epoch=1200 f_loss=-407.8255 g_loss=1397.1138 | train mmd=0.1493 | test_mmd=0.1272\n",
      "[CellOT] epoch=1250 f_loss=381.7934 g_loss=-4.9051 | train mmd=0.1760 | test_mmd=0.4985\n",
      "[CellOT] epoch=1300 f_loss=47.1134 g_loss=-3.4233 | train mmd=0.1718 | test_mmd=0.2726\n",
      "[CellOT] epoch=1350 f_loss=118.5603 g_loss=69.5443 | train mmd=0.2024 | test_mmd=0.1376\n",
      "[CellOT] epoch=1400 f_loss=45.8049 g_loss=0.9441 | train mmd=0.1629 | test_mmd=0.0426\n",
      "[CellOT] epoch=1450 f_loss=27.7802 g_loss=1785.1843 | train mmd=0.1802 | test_mmd=0.0954\n",
      "[CellOT] epoch=1500 f_loss=-70.8536 g_loss=1.6680 | train mmd=0.1217 | test_mmd=0.0454\n",
      "[CellOT] epoch=1550 f_loss=30.1345 g_loss=20.6401 | train mmd=0.1102 | test_mmd=0.0295\n",
      "[CellOT] epoch=1600 f_loss=5.4261 g_loss=2.7798 | train mmd=0.0827 | test_mmd=0.0125\n",
      "[CellOT] epoch=1650 f_loss=-3.7534 g_loss=4.1089 | train mmd=0.1301 | test_mmd=0.0546\n",
      "[CellOT] epoch=1700 f_loss=11.0826 g_loss=5.4588 | train mmd=0.0349 | test_mmd=0.0059\n",
      "[CellOT] epoch=1750 f_loss=11.4282 g_loss=4.7862 | train mmd=0.0508 | test_mmd=0.0068\n",
      "[CellOT] epoch=1800 f_loss=8.4472 g_loss=5.1344 | train mmd=0.0593 | test_mmd=0.0054\n",
      "[CellOT] epoch=1850 f_loss=3.1602 g_loss=5.1192 | train mmd=0.0497 | test_mmd=0.0095\n",
      "[CellOT] epoch=1900 f_loss=0.5294 g_loss=5.8641 | train mmd=0.0428 | test_mmd=0.0036\n",
      "[CellOT] epoch=1950 f_loss=-0.2250 g_loss=5.5811 | train mmd=0.0396 | test_mmd=0.0123\n",
      "[CellOT] epoch=2000 f_loss=0.4635 g_loss=5.9473 | train mmd=0.0513 | test_mmd=0.0119\n",
      "[CellOT] Final CellOT MMD: 0.0192\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 2 metrics: {'mmd2_gamma_median': 0.011935214930424154, 'mmd2_gamma_0.5': 0.048177248590129174, 'mmd2_gamma_1.0': 0.05449169813541521, 'wasserstein_distance': 0.8684969970706492, 'R2_feature_means': 0.9836968802286432}\n",
      "**************** Run: 3 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=0 f_loss=-3364332.2500 g_loss=3905918.5000 | train mmd=0.1601 | test_mmd=0.7867\n",
      "[CellOT] epoch=50 f_loss=-132.5198 g_loss=310.5785 | train mmd=0.8561 | test_mmd=0.3551\n",
      "[CellOT] epoch=100 f_loss=-426.5147 g_loss=552.5078 | train mmd=1.0384 | test_mmd=0.3645\n",
      "[CellOT] epoch=150 f_loss=-749.8436 g_loss=785.6952 | train mmd=0.7522 | test_mmd=0.2540\n",
      "[CellOT] epoch=200 f_loss=-879.5096 g_loss=1017.0789 | train mmd=0.8422 | test_mmd=0.2438\n",
      "[CellOT] epoch=250 f_loss=-1141.9054 g_loss=1261.3506 | train mmd=0.8529 | test_mmd=0.2394\n",
      "[CellOT] epoch=300 f_loss=-1377.6338 g_loss=1471.4250 | train mmd=0.4182 | test_mmd=0.2145\n",
      "[CellOT] epoch=350 f_loss=-1591.0481 g_loss=1269.5283 | train mmd=0.8073 | test_mmd=0.1918\n",
      "[CellOT] epoch=400 f_loss=-1818.0625 g_loss=2014.2578 | train mmd=0.8753 | test_mmd=0.1970\n",
      "[CellOT] epoch=450 f_loss=-1900.0978 g_loss=2109.5947 | train mmd=0.8137 | test_mmd=0.1666\n",
      "[CellOT] epoch=500 f_loss=-2141.6509 g_loss=2232.5940 | train mmd=0.7976 | test_mmd=0.1536\n",
      "[CellOT] epoch=550 f_loss=-2195.5198 g_loss=2683.6929 | train mmd=0.8371 | test_mmd=0.1567\n",
      "[CellOT] epoch=600 f_loss=-2187.3030 g_loss=2363.2603 | train mmd=0.7237 | test_mmd=0.1130\n",
      "[CellOT] epoch=650 f_loss=-1808.0789 g_loss=2573.2651 | train mmd=0.5212 | test_mmd=0.0732\n",
      "[CellOT] epoch=700 f_loss=-2097.2932 g_loss=2938.1995 | train mmd=0.5935 | test_mmd=0.0761\n",
      "[CellOT] epoch=750 f_loss=-2104.4268 g_loss=3494.2583 | train mmd=0.5681 | test_mmd=0.0693\n",
      "[CellOT] epoch=800 f_loss=-2079.5098 g_loss=3131.9961 | train mmd=0.5407 | test_mmd=0.0487\n",
      "[CellOT] epoch=850 f_loss=-3458.3767 g_loss=2027.4879 | train mmd=0.3616 | test_mmd=0.4329\n",
      "[CellOT] epoch=900 f_loss=-1475.2551 g_loss=2772.3848 | train mmd=0.4433 | test_mmd=0.0659\n",
      "[CellOT] epoch=950 f_loss=-1857.3589 g_loss=3238.2764 | train mmd=0.4525 | test_mmd=0.0335\n",
      "[CellOT] epoch=1000 f_loss=-1723.6460 g_loss=2994.4480 | train mmd=0.4392 | test_mmd=0.0237\n",
      "[CellOT] epoch=1050 f_loss=-1349.4835 g_loss=2942.7119 | train mmd=0.3328 | test_mmd=0.0188\n",
      "[CellOT] epoch=1100 f_loss=251.9518 g_loss=2583.7559 | train mmd=0.1545 | test_mmd=0.0872\n",
      "[CellOT] epoch=1150 f_loss=-997.0500 g_loss=2155.1167 | train mmd=0.2864 | test_mmd=0.0301\n",
      "[CellOT] epoch=1200 f_loss=473.1789 g_loss=999.7903 | train mmd=0.2218 | test_mmd=0.1348\n",
      "[CellOT] epoch=1250 f_loss=-77.0730 g_loss=2613.0762 | train mmd=0.1689 | test_mmd=0.0424\n",
      "[CellOT] epoch=1300 f_loss=759.1928 g_loss=809.2794 | train mmd=0.1884 | test_mmd=0.0701\n",
      "[CellOT] epoch=1350 f_loss=-4588.9551 g_loss=762.6298 | train mmd=0.3020 | test_mmd=0.1131\n",
      "[CellOT] epoch=1400 f_loss=-55.0886 g_loss=855.2638 | train mmd=0.1894 | test_mmd=0.0307\n",
      "[CellOT] epoch=1450 f_loss=317.8698 g_loss=1051.3844 | train mmd=0.2302 | test_mmd=0.1060\n",
      "[CellOT] epoch=1500 f_loss=568.9043 g_loss=1024.4250 | train mmd=0.1811 | test_mmd=0.0556\n",
      "[CellOT] epoch=1550 f_loss=415.5641 g_loss=1704.7363 | train mmd=0.2792 | test_mmd=0.0851\n",
      "[CellOT] epoch=1600 f_loss=238.1704 g_loss=1118.0691 | train mmd=0.1757 | test_mmd=0.0612\n",
      "[CellOT] epoch=1650 f_loss=370.4422 g_loss=2180.2307 | train mmd=0.2156 | test_mmd=0.0994\n",
      "[CellOT] epoch=1700 f_loss=79.4840 g_loss=1120.7756 | train mmd=0.2696 | test_mmd=0.0818\n",
      "[CellOT] epoch=1750 f_loss=261.2350 g_loss=1145.3689 | train mmd=0.2512 | test_mmd=0.0485\n",
      "[CellOT] epoch=1800 f_loss=205.6528 g_loss=1118.4323 | train mmd=0.1786 | test_mmd=0.0516\n",
      "[CellOT] epoch=1850 f_loss=-89.5147 g_loss=1625.0641 | train mmd=0.3576 | test_mmd=0.1247\n",
      "[CellOT] epoch=1900 f_loss=115.2371 g_loss=1370.7910 | train mmd=0.4767 | test_mmd=0.1023\n",
      "[CellOT] epoch=1950 f_loss=3.6570 g_loss=1652.7178 | train mmd=0.3990 | test_mmd=0.0895\n",
      "[CellOT] epoch=2000 f_loss=58.1714 g_loss=1512.9821 | train mmd=0.1745 | test_mmd=0.0951\n",
      "[CellOT] Final CellOT MMD: 0.1028\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 3 metrics: {'mmd2_gamma_median': 0.0951107428571476, 'mmd2_gamma_0.5': 0.16878705105389902, 'mmd2_gamma_1.0': 0.17778776267954916, 'wasserstein_distance': 1.5996778559747065, 'R2_feature_means': 0.7863187864518125}\n",
      "**************** Run: 4 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=0 f_loss=-3209145.0000 g_loss=3642849.5000 | train mmd=0.1601 | test_mmd=0.7867\n",
      "[CellOT] epoch=50 f_loss=-643.6461 g_loss=356.5972 | train mmd=0.6096 | test_mmd=0.3763\n",
      "[CellOT] epoch=100 f_loss=-311.8356 g_loss=442.9236 | train mmd=0.9893 | test_mmd=0.3443\n",
      "[CellOT] epoch=150 f_loss=-631.0927 g_loss=648.7800 | train mmd=1.0380 | test_mmd=0.3564\n",
      "[CellOT] epoch=200 f_loss=-850.0884 g_loss=841.6785 | train mmd=0.9770 | test_mmd=0.3154\n",
      "[CellOT] epoch=250 f_loss=-1011.1660 g_loss=1123.8286 | train mmd=0.9065 | test_mmd=0.2674\n",
      "[CellOT] epoch=300 f_loss=-1176.4768 g_loss=1307.1101 | train mmd=0.8985 | test_mmd=0.2512\n",
      "[CellOT] epoch=350 f_loss=-1379.5040 g_loss=1499.7383 | train mmd=0.8748 | test_mmd=0.2391\n",
      "[CellOT] epoch=400 f_loss=-1484.2303 g_loss=1721.9285 | train mmd=0.8583 | test_mmd=0.2187\n",
      "[CellOT] epoch=450 f_loss=-1817.4160 g_loss=1919.3810 | train mmd=0.9079 | test_mmd=0.2199\n",
      "[CellOT] epoch=500 f_loss=-1779.7649 g_loss=2010.3762 | train mmd=0.7519 | test_mmd=0.1662\n",
      "[CellOT] epoch=550 f_loss=-4808.9263 g_loss=2332.8706 | train mmd=0.7065 | test_mmd=0.1635\n",
      "[CellOT] epoch=600 f_loss=-2261.6155 g_loss=2574.5815 | train mmd=0.8704 | test_mmd=0.1828\n",
      "[CellOT] epoch=650 f_loss=-2126.1899 g_loss=2417.0415 | train mmd=0.6924 | test_mmd=0.1175\n",
      "[CellOT] epoch=700 f_loss=-2321.1089 g_loss=2782.6260 | train mmd=0.7888 | test_mmd=0.1329\n",
      "[CellOT] epoch=750 f_loss=-3582.1152 g_loss=2970.6792 | train mmd=0.6446 | test_mmd=0.1008\n",
      "[CellOT] epoch=800 f_loss=-2602.1475 g_loss=3114.6758 | train mmd=0.7681 | test_mmd=0.1152\n",
      "[CellOT] epoch=850 f_loss=-2367.0098 g_loss=3127.9392 | train mmd=0.6926 | test_mmd=0.0892\n",
      "[CellOT] epoch=900 f_loss=-2567.6174 g_loss=3279.1399 | train mmd=0.6977 | test_mmd=0.0888\n",
      "[CellOT] epoch=950 f_loss=-2108.2449 g_loss=3078.5144 | train mmd=0.5749 | test_mmd=0.0610\n",
      "[CellOT] epoch=1000 f_loss=-2198.5771 g_loss=3080.4736 | train mmd=0.5645 | test_mmd=0.0602\n",
      "[CellOT] epoch=1050 f_loss=-1774.1212 g_loss=5674.8076 | train mmd=0.3258 | test_mmd=0.1000\n",
      "[CellOT] epoch=1100 f_loss=-1716.3081 g_loss=3304.9048 | train mmd=0.3735 | test_mmd=0.0309\n",
      "[CellOT] epoch=1150 f_loss=-2155.1794 g_loss=3691.2771 | train mmd=0.5588 | test_mmd=0.0455\n",
      "[CellOT] epoch=1200 f_loss=-1251.7076 g_loss=2763.2427 | train mmd=0.3322 | test_mmd=0.0408\n",
      "[CellOT] epoch=1250 f_loss=-1674.0044 g_loss=3319.4468 | train mmd=0.4530 | test_mmd=0.0283\n",
      "[CellOT] epoch=1300 f_loss=-1644.9808 g_loss=3349.2778 | train mmd=0.5047 | test_mmd=0.0289\n",
      "[CellOT] epoch=1350 f_loss=-536.8835 g_loss=2306.3054 | train mmd=0.2658 | test_mmd=0.0516\n",
      "[CellOT] epoch=1400 f_loss=-159.9514 g_loss=1710.5212 | train mmd=0.2882 | test_mmd=0.0370\n",
      "[CellOT] epoch=1450 f_loss=220.3893 g_loss=1550.9906 | train mmd=0.2209 | test_mmd=0.0378\n",
      "[CellOT] epoch=1500 f_loss=397.0757 g_loss=1085.8076 | train mmd=0.2537 | test_mmd=0.0153\n",
      "[CellOT] epoch=1550 f_loss=813.9817 g_loss=1262.5861 | train mmd=0.2330 | test_mmd=0.0799\n",
      "[CellOT] epoch=1600 f_loss=1116.3341 g_loss=2241.1567 | train mmd=0.2876 | test_mmd=0.1109\n",
      "[CellOT] epoch=1650 f_loss=416.2554 g_loss=1756.3676 | train mmd=0.2238 | test_mmd=0.0432\n",
      "[CellOT] epoch=1700 f_loss=312.1913 g_loss=2300.9834 | train mmd=0.2448 | test_mmd=0.1084\n",
      "[CellOT] epoch=1750 f_loss=307.8044 g_loss=3022.0269 | train mmd=0.2771 | test_mmd=0.1022\n",
      "[CellOT] epoch=1800 f_loss=-99.7053 g_loss=2515.1709 | train mmd=0.3077 | test_mmd=0.0693\n",
      "[CellOT] epoch=1850 f_loss=243.2071 g_loss=861.1255 | train mmd=0.2391 | test_mmd=0.0758\n",
      "[CellOT] epoch=1900 f_loss=101.1180 g_loss=889.5950 | train mmd=0.5447 | test_mmd=0.1853\n",
      "[CellOT] epoch=1950 f_loss=22.2880 g_loss=1006.5049 | train mmd=0.4285 | test_mmd=0.1038\n",
      "[CellOT] epoch=2000 f_loss=52.7452 g_loss=1061.4349 | train mmd=0.2824 | test_mmd=0.0989\n",
      "[CellOT] Final CellOT MMD: 0.1275\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 4 metrics: {'mmd2_gamma_median': 0.09885318795974674, 'mmd2_gamma_0.5': 0.27657499021733334, 'mmd2_gamma_1.0': 0.2903037552024297, 'wasserstein_distance': 1.4329898368845793, 'R2_feature_means': 0.8305392513822213}\n",
      "**************** Run: 5 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=0 f_loss=-3482043.0000 g_loss=4025303.0000 | train mmd=0.1601 | test_mmd=0.7867\n",
      "[CellOT] epoch=50 f_loss=124.9986 g_loss=247.5221 | train mmd=0.6484 | test_mmd=0.3812\n",
      "[CellOT] epoch=100 f_loss=-290.8881 g_loss=469.1166 | train mmd=1.0336 | test_mmd=0.3795\n",
      "[CellOT] epoch=150 f_loss=-536.6507 g_loss=566.0032 | train mmd=0.9662 | test_mmd=0.3288\n",
      "[CellOT] epoch=200 f_loss=-748.4483 g_loss=798.9669 | train mmd=0.9026 | test_mmd=0.2906\n",
      "[CellOT] epoch=250 f_loss=-872.3799 g_loss=951.8104 | train mmd=0.5165 | test_mmd=0.1729\n",
      "[CellOT] epoch=300 f_loss=-1140.9062 g_loss=1291.6793 | train mmd=0.8971 | test_mmd=0.2570\n",
      "[CellOT] epoch=350 f_loss=-1398.7079 g_loss=1371.6565 | train mmd=0.9079 | test_mmd=0.2437\n",
      "[CellOT] epoch=400 f_loss=-1516.4723 g_loss=1682.2598 | train mmd=0.8280 | test_mmd=0.2096\n",
      "[CellOT] epoch=450 f_loss=-1697.8699 g_loss=1969.4075 | train mmd=0.8198 | test_mmd=0.1928\n",
      "[CellOT] epoch=500 f_loss=-2080.6235 g_loss=2142.1936 | train mmd=0.8040 | test_mmd=0.1823\n",
      "[CellOT] epoch=550 f_loss=-2803.1084 g_loss=2210.3464 | train mmd=0.7074 | test_mmd=0.1469\n",
      "[CellOT] epoch=600 f_loss=-2532.9849 g_loss=2810.3701 | train mmd=0.7452 | test_mmd=0.1475\n",
      "[CellOT] epoch=650 f_loss=-2703.9473 g_loss=2700.7607 | train mmd=0.6223 | test_mmd=0.1063\n",
      "[CellOT] epoch=700 f_loss=-2803.8389 g_loss=3094.7854 | train mmd=0.6763 | test_mmd=0.1085\n",
      "[CellOT] epoch=750 f_loss=-3181.4565 g_loss=3398.2261 | train mmd=0.7107 | test_mmd=0.1033\n",
      "[CellOT] epoch=800 f_loss=-2442.5312 g_loss=2845.8855 | train mmd=0.4758 | test_mmd=0.0624\n",
      "[CellOT] epoch=850 f_loss=-2507.9746 g_loss=3014.0991 | train mmd=0.4281 | test_mmd=0.0920\n",
      "[CellOT] epoch=900 f_loss=-2724.2168 g_loss=3255.8354 | train mmd=0.4625 | test_mmd=0.0656\n",
      "[CellOT] epoch=950 f_loss=-1649.1067 g_loss=3128.6943 | train mmd=0.2578 | test_mmd=0.1088\n",
      "[CellOT] epoch=1000 f_loss=-1522.4008 g_loss=2733.7661 | train mmd=0.2347 | test_mmd=0.1423\n",
      "[CellOT] epoch=1050 f_loss=-1448.4360 g_loss=2752.3542 | train mmd=0.2752 | test_mmd=0.1123\n",
      "[CellOT] epoch=1100 f_loss=-700.5250 g_loss=1756.2733 | train mmd=0.1382 | test_mmd=0.2340\n",
      "[CellOT] epoch=1150 f_loss=-716.1895 g_loss=2037.4189 | train mmd=0.1708 | test_mmd=0.2095\n",
      "[CellOT] epoch=1200 f_loss=337.6357 g_loss=-1.4584 | train mmd=0.1704 | test_mmd=0.4291\n",
      "[CellOT] epoch=1250 f_loss=225.5047 g_loss=21.7706 | train mmd=0.1806 | test_mmd=0.1830\n",
      "[CellOT] epoch=1300 f_loss=50.1654 g_loss=-0.2711 | train mmd=0.1605 | test_mmd=0.0730\n",
      "[CellOT] epoch=1350 f_loss=139.0654 g_loss=11.7755 | train mmd=0.1310 | test_mmd=0.0498\n",
      "[CellOT] epoch=1400 f_loss=162.8329 g_loss=5.0097 | train mmd=0.1044 | test_mmd=0.0283\n",
      "[CellOT] epoch=1450 f_loss=125.5072 g_loss=1.7410 | train mmd=0.0640 | test_mmd=0.0146\n",
      "[CellOT] epoch=1500 f_loss=44.6507 g_loss=1.0980 | train mmd=0.0866 | test_mmd=0.0207\n",
      "[CellOT] epoch=1550 f_loss=67.3931 g_loss=0.9742 | train mmd=0.1210 | test_mmd=0.0430\n",
      "[CellOT] epoch=1600 f_loss=1.3920 g_loss=2.9964 | train mmd=0.0732 | test_mmd=0.0110\n",
      "[CellOT] epoch=1650 f_loss=62.6968 g_loss=4.9316 | train mmd=0.0462 | test_mmd=0.0072\n",
      "[CellOT] epoch=1700 f_loss=0.7668 g_loss=5.2675 | train mmd=0.0383 | test_mmd=0.0096\n",
      "[CellOT] epoch=1750 f_loss=1.1838 g_loss=6.2225 | train mmd=0.0290 | test_mmd=0.0063\n",
      "[CellOT] epoch=1800 f_loss=45.5137 g_loss=7.2224 | train mmd=0.1154 | test_mmd=0.0293\n",
      "[CellOT] epoch=1850 f_loss=0.9942 g_loss=7.8211 | train mmd=0.0308 | test_mmd=0.0050\n",
      "[CellOT] epoch=1900 f_loss=0.8286 g_loss=8.5135 | train mmd=0.0342 | test_mmd=0.0071\n",
      "[CellOT] epoch=1950 f_loss=0.0707 g_loss=9.0764 | train mmd=0.0455 | test_mmd=0.0059\n",
      "[CellOT] epoch=2000 f_loss=1.1572 g_loss=9.7081 | train mmd=0.0762 | test_mmd=0.0116\n",
      "[CellOT] Final CellOT MMD: 0.0333\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 5 metrics: {'mmd2_gamma_median': 0.011584601593292776, 'mmd2_gamma_0.5': 0.056440442690812054, 'mmd2_gamma_1.0': 0.07988950667200273, 'wasserstein_distance': 0.9973311402006968, 'R2_feature_means': 0.9871571297684208}\n",
      "**************** Run: 6 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=0 f_loss=-2358018.2500 g_loss=3021984.5000 | train mmd=0.1601 | test_mmd=0.7867\n",
      "[CellOT] epoch=50 f_loss=506.9464 g_loss=332.2629 | train mmd=1.0378 | test_mmd=0.4065\n",
      "[CellOT] epoch=100 f_loss=-275.6234 g_loss=343.0029 | train mmd=1.0340 | test_mmd=0.3752\n",
      "[CellOT] epoch=150 f_loss=-492.0607 g_loss=500.3222 | train mmd=0.9909 | test_mmd=0.3450\n",
      "[CellOT] epoch=200 f_loss=-619.7970 g_loss=628.6159 | train mmd=0.8862 | test_mmd=0.2758\n",
      "[CellOT] epoch=250 f_loss=-784.4609 g_loss=804.2644 | train mmd=0.8538 | test_mmd=0.2532\n",
      "[CellOT] epoch=300 f_loss=-1029.1141 g_loss=1002.2910 | train mmd=0.6621 | test_mmd=0.2257\n",
      "[CellOT] epoch=350 f_loss=-1021.4385 g_loss=1151.4873 | train mmd=0.7626 | test_mmd=0.2003\n",
      "[CellOT] epoch=400 f_loss=-1281.4912 g_loss=1309.3503 | train mmd=0.3543 | test_mmd=0.3314\n",
      "[CellOT] epoch=450 f_loss=-1394.8204 g_loss=1531.4861 | train mmd=0.8613 | test_mmd=0.2081\n",
      "[CellOT] epoch=500 f_loss=-1479.9810 g_loss=1637.7222 | train mmd=0.8325 | test_mmd=0.1864\n",
      "[CellOT] epoch=550 f_loss=-1568.9954 g_loss=1678.3613 | train mmd=0.7734 | test_mmd=0.1478\n",
      "[CellOT] epoch=600 f_loss=-1615.6716 g_loss=4121.6221 | train mmd=0.5926 | test_mmd=0.1483\n",
      "[CellOT] epoch=650 f_loss=-1589.6820 g_loss=2028.8683 | train mmd=0.6227 | test_mmd=0.1224\n",
      "[CellOT] epoch=700 f_loss=-1704.1575 g_loss=2179.9854 | train mmd=0.6903 | test_mmd=0.1045\n",
      "[CellOT] epoch=750 f_loss=-1740.3157 g_loss=2346.4998 | train mmd=0.7309 | test_mmd=0.1121\n",
      "[CellOT] epoch=800 f_loss=-1425.8940 g_loss=3685.2505 | train mmd=0.4702 | test_mmd=0.0894\n",
      "[CellOT] epoch=850 f_loss=-1945.7412 g_loss=2388.1479 | train mmd=0.7153 | test_mmd=0.0855\n",
      "[CellOT] epoch=900 f_loss=-1463.9664 g_loss=2117.9700 | train mmd=0.4923 | test_mmd=0.0469\n",
      "[CellOT] epoch=950 f_loss=-1758.1138 g_loss=2433.4546 | train mmd=0.5249 | test_mmd=0.0529\n",
      "[CellOT] epoch=1000 f_loss=-1449.8687 g_loss=2590.0635 | train mmd=0.5061 | test_mmd=0.0410\n",
      "[CellOT] epoch=1050 f_loss=-1368.0551 g_loss=2356.9734 | train mmd=0.4703 | test_mmd=0.0322\n",
      "[CellOT] epoch=1100 f_loss=-708.8469 g_loss=1883.4833 | train mmd=0.2684 | test_mmd=0.0282\n",
      "[CellOT] epoch=1150 f_loss=-852.4979 g_loss=2388.5117 | train mmd=0.3406 | test_mmd=0.0217\n",
      "[CellOT] epoch=1200 f_loss=-1382.6968 g_loss=2061.6877 | train mmd=0.3762 | test_mmd=0.0290\n",
      "[CellOT] epoch=1250 f_loss=-422.7556 g_loss=2288.0752 | train mmd=0.2821 | test_mmd=0.0295\n",
      "[CellOT] epoch=1300 f_loss=-660.6122 g_loss=1287.8907 | train mmd=0.3682 | test_mmd=0.0210\n",
      "[CellOT] epoch=1350 f_loss=199.3125 g_loss=1579.5488 | train mmd=0.2626 | test_mmd=0.0461\n",
      "[CellOT] epoch=1400 f_loss=-1000.5532 g_loss=1031.1863 | train mmd=0.2082 | test_mmd=0.0480\n",
      "[CellOT] epoch=1450 f_loss=416.6672 g_loss=1656.7509 | train mmd=0.2366 | test_mmd=0.0334\n",
      "[CellOT] epoch=1500 f_loss=457.3968 g_loss=1503.3005 | train mmd=0.2909 | test_mmd=0.0221\n",
      "[CellOT] epoch=1550 f_loss=625.9239 g_loss=1530.7966 | train mmd=0.1904 | test_mmd=0.1087\n",
      "[CellOT] epoch=1600 f_loss=33.7507 g_loss=2635.7180 | train mmd=0.3463 | test_mmd=0.0647\n",
      "[CellOT] epoch=1650 f_loss=525.1289 g_loss=2495.1162 | train mmd=0.2161 | test_mmd=0.0681\n",
      "[CellOT] epoch=1700 f_loss=415.1136 g_loss=2594.4141 | train mmd=0.2973 | test_mmd=0.0298\n",
      "[CellOT] epoch=1750 f_loss=-496.1447 g_loss=3277.1841 | train mmd=0.1842 | test_mmd=0.0987\n",
      "[CellOT] epoch=1800 f_loss=-653.7449 g_loss=3669.4722 | train mmd=0.4102 | test_mmd=0.1008\n",
      "[CellOT] epoch=1850 f_loss=57.8746 g_loss=3439.1670 | train mmd=0.2298 | test_mmd=0.0327\n",
      "[CellOT] epoch=1900 f_loss=210.9481 g_loss=3361.2466 | train mmd=0.3133 | test_mmd=0.0268\n",
      "[CellOT] epoch=1950 f_loss=331.5764 g_loss=3607.9937 | train mmd=0.3014 | test_mmd=0.0271\n",
      "[CellOT] epoch=2000 f_loss=347.0491 g_loss=3765.1470 | train mmd=0.3219 | test_mmd=0.0253\n",
      "[CellOT] Final CellOT MMD: 0.1648\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 6 metrics: {'mmd2_gamma_median': 0.02527716283569892, 'mmd2_gamma_0.5': 0.21374506523091963, 'mmd2_gamma_1.0': 0.3326307963701889, 'wasserstein_distance': 1.2808875303564489, 'R2_feature_means': 0.9844479601599859}\n",
      "**************** Run: 7 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=0 f_loss=-3797288.7500 g_loss=4570948.0000 | train mmd=0.1601 | test_mmd=0.7867\n",
      "[CellOT] epoch=50 f_loss=92.5746 g_loss=272.7787 | train mmd=0.7658 | test_mmd=0.3174\n",
      "[CellOT] epoch=100 f_loss=-403.9055 g_loss=466.2184 | train mmd=1.0597 | test_mmd=0.3842\n",
      "[CellOT] epoch=150 f_loss=-720.7181 g_loss=755.8170 | train mmd=1.0280 | test_mmd=0.3512\n",
      "[CellOT] epoch=200 f_loss=-2811.5835 g_loss=978.4957 | train mmd=0.4331 | test_mmd=0.7070\n",
      "[CellOT] epoch=250 f_loss=-1086.5035 g_loss=1201.7582 | train mmd=0.8556 | test_mmd=0.2408\n",
      "[CellOT] epoch=300 f_loss=-1401.6929 g_loss=1388.8378 | train mmd=0.8127 | test_mmd=0.2256\n",
      "[CellOT] epoch=350 f_loss=-1721.1833 g_loss=1705.2415 | train mmd=0.8613 | test_mmd=0.2200\n",
      "[CellOT] epoch=400 f_loss=-1824.8655 g_loss=2026.4519 | train mmd=0.7917 | test_mmd=0.1901\n",
      "[CellOT] epoch=450 f_loss=-1813.3228 g_loss=1885.9927 | train mmd=0.7428 | test_mmd=0.1665\n",
      "[CellOT] epoch=500 f_loss=-2073.2517 g_loss=1929.2773 | train mmd=0.7675 | test_mmd=0.1534\n",
      "[CellOT] epoch=550 f_loss=-2049.8154 g_loss=2286.8892 | train mmd=0.6664 | test_mmd=0.1148\n",
      "[CellOT] epoch=600 f_loss=-2507.5459 g_loss=2934.5786 | train mmd=0.7393 | test_mmd=0.1353\n",
      "[CellOT] epoch=650 f_loss=-2514.1821 g_loss=3006.5671 | train mmd=0.6873 | test_mmd=0.1043\n",
      "[CellOT] epoch=700 f_loss=-2369.2754 g_loss=3292.8545 | train mmd=0.6380 | test_mmd=0.0878\n",
      "[CellOT] epoch=750 f_loss=-2613.3315 g_loss=3239.7075 | train mmd=0.5963 | test_mmd=0.0775\n",
      "[CellOT] epoch=800 f_loss=-2882.7476 g_loss=3347.5181 | train mmd=0.6449 | test_mmd=0.0783\n",
      "[CellOT] epoch=850 f_loss=-3055.6069 g_loss=3457.2363 | train mmd=0.4709 | test_mmd=0.0797\n",
      "[CellOT] epoch=900 f_loss=-2688.4399 g_loss=4131.6216 | train mmd=0.4975 | test_mmd=0.0506\n",
      "[CellOT] epoch=950 f_loss=-2411.8613 g_loss=3801.3928 | train mmd=0.5367 | test_mmd=0.0448\n",
      "[CellOT] epoch=1000 f_loss=-3514.1128 g_loss=3396.1108 | train mmd=0.4076 | test_mmd=0.0449\n",
      "[CellOT] epoch=1050 f_loss=-1960.8960 g_loss=3539.6204 | train mmd=0.4205 | test_mmd=0.0228\n",
      "[CellOT] epoch=1100 f_loss=-1610.2517 g_loss=3377.1299 | train mmd=0.3133 | test_mmd=0.0254\n",
      "[CellOT] epoch=1150 f_loss=-1068.2418 g_loss=2771.6968 | train mmd=0.3135 | test_mmd=0.0342\n",
      "[CellOT] epoch=1200 f_loss=-1008.1769 g_loss=2624.5508 | train mmd=0.2944 | test_mmd=0.0468\n",
      "[CellOT] epoch=1250 f_loss=-907.6312 g_loss=2456.7312 | train mmd=0.2970 | test_mmd=0.0297\n",
      "[CellOT] epoch=1300 f_loss=-587.0894 g_loss=1756.5111 | train mmd=0.2599 | test_mmd=0.0511\n",
      "[CellOT] epoch=1350 f_loss=-434.9658 g_loss=1113.4119 | train mmd=0.2981 | test_mmd=0.0172\n",
      "[CellOT] epoch=1400 f_loss=621.1764 g_loss=507.4366 | train mmd=0.1544 | test_mmd=0.0698\n",
      "[CellOT] epoch=1450 f_loss=278.2275 g_loss=50.0980 | train mmd=0.0960 | test_mmd=0.0244\n",
      "[CellOT] epoch=1500 f_loss=38.5188 g_loss=-0.3283 | train mmd=0.1576 | test_mmd=0.0995\n",
      "[CellOT] epoch=1550 f_loss=63.7726 g_loss=0.8669 | train mmd=0.1550 | test_mmd=0.0918\n",
      "[CellOT] epoch=1600 f_loss=9.2744 g_loss=0.4486 | train mmd=0.1015 | test_mmd=0.0244\n",
      "[CellOT] epoch=1650 f_loss=8.6251 g_loss=4.0071 | train mmd=0.1161 | test_mmd=0.0174\n",
      "[CellOT] epoch=1700 f_loss=8.1905 g_loss=4.9982 | train mmd=0.0518 | test_mmd=0.0115\n",
      "[CellOT] epoch=1750 f_loss=1.3113 g_loss=5.4128 | train mmd=0.0696 | test_mmd=0.0143\n",
      "[CellOT] epoch=1800 f_loss=8.7724 g_loss=7.1019 | train mmd=0.0751 | test_mmd=0.0146\n",
      "[CellOT] epoch=1850 f_loss=0.1303 g_loss=5.3640 | train mmd=0.0691 | test_mmd=0.0125\n",
      "[CellOT] epoch=1900 f_loss=0.2615 g_loss=5.6523 | train mmd=0.0739 | test_mmd=0.0136\n",
      "[CellOT] epoch=1950 f_loss=6.9491 g_loss=6.4730 | train mmd=0.0731 | test_mmd=0.0150\n",
      "[CellOT] epoch=2000 f_loss=0.3044 g_loss=6.3890 | train mmd=0.0769 | test_mmd=0.0138\n",
      "[CellOT] Final CellOT MMD: 0.0276\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 7 metrics: {'mmd2_gamma_median': 0.013809382412517168, 'mmd2_gamma_0.5': 0.06253296470766811, 'mmd2_gamma_1.0': 0.07737710071771314, 'wasserstein_distance': 0.8385125618197652, 'R2_feature_means': 0.9833128632929382}\n",
      "**************** Run: 8 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=0 f_loss=-3013546.5000 g_loss=3396041.2500 | train mmd=0.1601 | test_mmd=0.7867\n",
      "[CellOT] epoch=50 f_loss=365.9742 g_loss=315.4385 | train mmd=1.0206 | test_mmd=0.3872\n",
      "[CellOT] epoch=100 f_loss=-372.9450 g_loss=414.0101 | train mmd=1.0855 | test_mmd=0.3904\n",
      "[CellOT] epoch=150 f_loss=-601.5657 g_loss=647.3611 | train mmd=0.9846 | test_mmd=0.3338\n",
      "[CellOT] epoch=200 f_loss=-831.6998 g_loss=855.2887 | train mmd=0.8911 | test_mmd=0.2775\n",
      "[CellOT] epoch=250 f_loss=-1025.2926 g_loss=979.8942 | train mmd=0.8998 | test_mmd=0.2580\n",
      "[CellOT] epoch=300 f_loss=-1238.6202 g_loss=1361.9333 | train mmd=0.9295 | test_mmd=0.2640\n",
      "[CellOT] epoch=350 f_loss=-1427.1322 g_loss=1484.4691 | train mmd=0.7667 | test_mmd=0.2231\n",
      "[CellOT] epoch=400 f_loss=-1586.4211 g_loss=1714.5986 | train mmd=0.7917 | test_mmd=0.1954\n",
      "[CellOT] epoch=450 f_loss=-1727.8186 g_loss=1961.0310 | train mmd=0.7887 | test_mmd=0.1891\n",
      "[CellOT] epoch=500 f_loss=-1766.9767 g_loss=2613.6235 | train mmd=0.7478 | test_mmd=0.1675\n",
      "[CellOT] epoch=550 f_loss=-2270.0137 g_loss=2409.0198 | train mmd=0.6846 | test_mmd=0.1456\n",
      "[CellOT] epoch=600 f_loss=-2194.7175 g_loss=2719.7087 | train mmd=0.7136 | test_mmd=0.1262\n",
      "[CellOT] epoch=650 f_loss=-2224.8638 g_loss=2660.7031 | train mmd=0.6957 | test_mmd=0.1138\n",
      "[CellOT] epoch=700 f_loss=-2506.5366 g_loss=2908.0732 | train mmd=0.7700 | test_mmd=0.1167\n",
      "[CellOT] epoch=750 f_loss=-2410.5149 g_loss=3098.1099 | train mmd=0.7362 | test_mmd=0.1025\n",
      "[CellOT] epoch=800 f_loss=-4245.1387 g_loss=2480.6714 | train mmd=0.2853 | test_mmd=0.1963\n",
      "[CellOT] epoch=850 f_loss=-2403.2639 g_loss=3105.6948 | train mmd=0.6256 | test_mmd=0.0670\n",
      "[CellOT] epoch=900 f_loss=-2238.2134 g_loss=3224.2549 | train mmd=0.5650 | test_mmd=0.0545\n",
      "[CellOT] epoch=950 f_loss=-2639.2905 g_loss=3350.7642 | train mmd=0.4018 | test_mmd=0.0435\n",
      "[CellOT] epoch=1000 f_loss=-1781.0518 g_loss=2991.5425 | train mmd=0.4641 | test_mmd=0.0319\n",
      "[CellOT] epoch=1050 f_loss=-1840.7451 g_loss=2733.0205 | train mmd=0.4065 | test_mmd=0.0183\n",
      "[CellOT] epoch=1100 f_loss=-1866.0902 g_loss=2520.6899 | train mmd=0.3765 | test_mmd=0.0184\n",
      "[CellOT] epoch=1150 f_loss=-201.6398 g_loss=1662.2283 | train mmd=0.1545 | test_mmd=0.0749\n",
      "[CellOT] epoch=1200 f_loss=-1305.7657 g_loss=2137.4756 | train mmd=0.3412 | test_mmd=0.0152\n",
      "[CellOT] epoch=1250 f_loss=20.5624 g_loss=840.8364 | train mmd=0.1295 | test_mmd=0.1575\n",
      "[CellOT] epoch=1300 f_loss=268.6239 g_loss=66.4167 | train mmd=0.0892 | test_mmd=0.0165\n",
      "[CellOT] epoch=1350 f_loss=134.2559 g_loss=40.8642 | train mmd=0.0715 | test_mmd=0.0248\n",
      "[CellOT] epoch=1400 f_loss=22.8715 g_loss=31.1041 | train mmd=0.1432 | test_mmd=0.0180\n",
      "[CellOT] epoch=1450 f_loss=24.3526 g_loss=44.8852 | train mmd=0.2284 | test_mmd=0.0217\n",
      "[CellOT] epoch=1500 f_loss=8.6982 g_loss=78.0101 | train mmd=0.1735 | test_mmd=0.0397\n",
      "[CellOT] epoch=1550 f_loss=25.9671 g_loss=71.1679 | train mmd=0.1959 | test_mmd=0.0275\n",
      "[CellOT] epoch=1600 f_loss=9.0308 g_loss=114.5440 | train mmd=0.3024 | test_mmd=0.1642\n",
      "[CellOT] epoch=1650 f_loss=31.9938 g_loss=145.8695 | train mmd=0.2121 | test_mmd=0.0357\n",
      "[CellOT] epoch=1700 f_loss=-145.3837 g_loss=120.5049 | train mmd=0.3631 | test_mmd=0.0891\n",
      "[CellOT] epoch=1750 f_loss=29.9379 g_loss=132.7570 | train mmd=0.1637 | test_mmd=0.0343\n",
      "[CellOT] epoch=1800 f_loss=8.6753 g_loss=135.9148 | train mmd=0.1835 | test_mmd=0.1329\n",
      "[CellOT] epoch=1850 f_loss=39.9454 g_loss=141.5007 | train mmd=0.2407 | test_mmd=0.0648\n",
      "[CellOT] epoch=1900 f_loss=20.6228 g_loss=159.6441 | train mmd=0.1300 | test_mmd=0.0458\n",
      "[CellOT] epoch=1950 f_loss=-0.5033 g_loss=211.5263 | train mmd=0.1151 | test_mmd=0.0391\n",
      "[CellOT] epoch=2000 f_loss=51.4193 g_loss=153.9920 | train mmd=0.1215 | test_mmd=0.0316\n",
      "[CellOT] Final CellOT MMD: 0.0661\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 8 metrics: {'mmd2_gamma_median': 0.03163042469378596, 'mmd2_gamma_0.5': 0.08549024412434392, 'mmd2_gamma_1.0': 0.12208553532097177, 'wasserstein_distance': 1.075354688293289, 'R2_feature_means': 0.9569547094893588}\n",
      "**************** Run: 9 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=0 f_loss=-2063642.7500 g_loss=2226211.5000 | train mmd=0.1601 | test_mmd=0.7867\n",
      "[CellOT] epoch=50 f_loss=324.8153 g_loss=660.0584 | train mmd=0.8867 | test_mmd=0.3684\n",
      "[CellOT] epoch=100 f_loss=-395.8270 g_loss=443.9370 | train mmd=1.0701 | test_mmd=0.3818\n",
      "[CellOT] epoch=150 f_loss=-653.6917 g_loss=705.6902 | train mmd=0.8551 | test_mmd=0.2672\n",
      "[CellOT] epoch=200 f_loss=-785.4043 g_loss=837.3347 | train mmd=0.8421 | test_mmd=0.2496\n",
      "[CellOT] epoch=250 f_loss=-1115.4060 g_loss=1107.8623 | train mmd=0.9260 | test_mmd=0.2651\n",
      "[CellOT] epoch=300 f_loss=-1269.4304 g_loss=1370.5396 | train mmd=0.8579 | test_mmd=0.2262\n",
      "[CellOT] epoch=350 f_loss=-1506.0884 g_loss=1576.3849 | train mmd=0.8286 | test_mmd=0.2076\n",
      "[CellOT] epoch=400 f_loss=-1602.0137 g_loss=1734.2954 | train mmd=0.7390 | test_mmd=0.1743\n",
      "[CellOT] epoch=450 f_loss=-1803.5212 g_loss=2083.9075 | train mmd=0.7125 | test_mmd=0.1591\n",
      "[CellOT] epoch=500 f_loss=-1917.6194 g_loss=2236.3462 | train mmd=0.7393 | test_mmd=0.1585\n",
      "[CellOT] epoch=550 f_loss=-2300.3945 g_loss=2418.5347 | train mmd=0.7178 | test_mmd=0.1468\n",
      "[CellOT] epoch=600 f_loss=-2262.9956 g_loss=2937.2930 | train mmd=0.7504 | test_mmd=0.1304\n",
      "[CellOT] epoch=650 f_loss=-2601.0107 g_loss=3021.7271 | train mmd=0.8008 | test_mmd=0.1385\n",
      "[CellOT] epoch=700 f_loss=-2663.0415 g_loss=3321.2656 | train mmd=0.6950 | test_mmd=0.1029\n",
      "[CellOT] epoch=750 f_loss=-2613.1816 g_loss=3340.8135 | train mmd=0.6524 | test_mmd=0.0856\n",
      "[CellOT] epoch=800 f_loss=-2268.1709 g_loss=3423.2100 | train mmd=0.5159 | test_mmd=0.0590\n",
      "[CellOT] epoch=850 f_loss=-1537.4241 g_loss=3471.5181 | train mmd=0.3411 | test_mmd=0.0581\n",
      "[CellOT] epoch=900 f_loss=-1466.9398 g_loss=2508.7422 | train mmd=0.3570 | test_mmd=0.0718\n",
      "[CellOT] epoch=950 f_loss=-4318.9082 g_loss=3390.9863 | train mmd=0.3650 | test_mmd=0.1288\n",
      "[CellOT] epoch=1000 f_loss=-1494.6316 g_loss=3196.3577 | train mmd=0.3590 | test_mmd=0.0397\n",
      "[CellOT] epoch=1050 f_loss=-1832.8741 g_loss=3928.2988 | train mmd=0.4175 | test_mmd=0.0252\n",
      "[CellOT] epoch=1100 f_loss=-1657.7643 g_loss=3239.0845 | train mmd=0.3223 | test_mmd=0.0250\n",
      "[CellOT] epoch=1150 f_loss=-1427.7274 g_loss=3656.3110 | train mmd=0.3111 | test_mmd=0.0413\n",
      "[CellOT] epoch=1200 f_loss=-308.8232 g_loss=2347.1035 | train mmd=0.2478 | test_mmd=0.0345\n",
      "[CellOT] epoch=1250 f_loss=-220.3128 g_loss=2019.1461 | train mmd=0.2119 | test_mmd=0.0208\n",
      "[CellOT] epoch=1300 f_loss=696.1444 g_loss=1245.9918 | train mmd=0.2435 | test_mmd=0.0551\n",
      "[CellOT] epoch=1350 f_loss=758.9490 g_loss=1077.8457 | train mmd=0.1790 | test_mmd=0.0516\n",
      "[CellOT] epoch=1400 f_loss=603.4556 g_loss=892.9194 | train mmd=0.3364 | test_mmd=0.1169\n",
      "[CellOT] epoch=1450 f_loss=-128.9744 g_loss=930.8124 | train mmd=0.4252 | test_mmd=0.1003\n",
      "[CellOT] epoch=1500 f_loss=756.4648 g_loss=1018.8707 | train mmd=0.1863 | test_mmd=0.0777\n",
      "[CellOT] epoch=1550 f_loss=469.7604 g_loss=1148.0220 | train mmd=0.1962 | test_mmd=0.0583\n",
      "[CellOT] epoch=1600 f_loss=423.5399 g_loss=1453.3138 | train mmd=0.2289 | test_mmd=0.1711\n",
      "[CellOT] epoch=1650 f_loss=238.8575 g_loss=1690.8145 | train mmd=0.4317 | test_mmd=0.0985\n",
      "[CellOT] epoch=1700 f_loss=-213.8409 g_loss=1232.3848 | train mmd=0.2657 | test_mmd=0.2182\n",
      "[CellOT] epoch=1750 f_loss=198.5748 g_loss=1383.4066 | train mmd=0.2471 | test_mmd=0.0491\n",
      "[CellOT] epoch=1800 f_loss=98.2378 g_loss=1463.8445 | train mmd=0.3236 | test_mmd=0.1105\n",
      "[CellOT] epoch=1850 f_loss=98.5955 g_loss=1609.6345 | train mmd=0.4087 | test_mmd=0.1006\n",
      "[CellOT] epoch=1900 f_loss=127.5235 g_loss=1591.2186 | train mmd=0.1605 | test_mmd=0.0762\n",
      "[CellOT] epoch=1950 f_loss=54.8878 g_loss=1773.9309 | train mmd=0.3543 | test_mmd=0.0913\n",
      "[CellOT] epoch=2000 f_loss=-77.7318 g_loss=1896.8843 | train mmd=0.4109 | test_mmd=0.1074\n",
      "[CellOT] Final CellOT MMD: 0.1806\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/umap/umap_.py:1952: UserWarning: n_jobs value 1 overridden to 1 by setting random_state. Use no seed for parallelism.\n",
      "  warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 9 metrics: {'mmd2_gamma_median': 0.10739596174711519, 'mmd2_gamma_0.5': 0.38911870801195236, 'mmd2_gamma_1.0': 0.4342716317370062, 'wasserstein_distance': 1.4914037240985916, 'R2_feature_means': 0.815647696031292}\n",
      "                        mean     std\n",
      "mmd2_gamma_median     0.0443  0.0394\n",
      "mmd2_gamma_0.5        0.1692  0.1104\n",
      "mmd2_gamma_1.0        0.2175  0.1323\n",
      "wasserstein_distance  1.2233  0.2653\n",
      "R2_feature_means      0.9295  0.0830\n"
     ]
    },
    {
     "ename": "AttributeError",
     "evalue": "'DummyProcess' object has no attribute 'terminate'",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mRuntimeError\u001b[0m                              Traceback (most recent call last)",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/multiprocessing/pool.py:212\u001b[0m, in \u001b[0;36mPool.__init__\u001b[0;34m(self, processes, initializer, initargs, maxtasksperchild, context)\u001b[0m\n\u001b[1;32m    211\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[0;32m--> 212\u001b[0m     \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_repopulate_pool\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    213\u001b[0m \u001b[38;5;28;01mexcept\u001b[39;00m \u001b[38;5;167;01mException\u001b[39;00m:\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/multiprocessing/pool.py:303\u001b[0m, in \u001b[0;36mPool._repopulate_pool\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m    302\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m\u001b[38;5;250m \u001b[39m\u001b[38;5;21m_repopulate_pool\u001b[39m(\u001b[38;5;28mself\u001b[39m):\n\u001b[0;32m--> 303\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_repopulate_pool_static\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_ctx\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mProcess\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    304\u001b[0m \u001b[43m                                        \u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_processes\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    305\u001b[0m \u001b[43m                                        \u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_pool\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_inqueue\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    306\u001b[0m \u001b[43m                                        \u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_outqueue\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_initializer\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    307\u001b[0m \u001b[43m                                        \u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_initargs\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    308\u001b[0m \u001b[43m                                        \u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_maxtasksperchild\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    309\u001b[0m \u001b[43m                                        \u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_wrap_exception\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/multiprocessing/pool.py:326\u001b[0m, in \u001b[0;36mPool._repopulate_pool_static\u001b[0;34m(ctx, Process, processes, pool, inqueue, outqueue, initializer, initargs, maxtasksperchild, wrap_exception)\u001b[0m\n\u001b[1;32m    325\u001b[0m w\u001b[38;5;241m.\u001b[39mdaemon \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mTrue\u001b[39;00m\n\u001b[0;32m--> 326\u001b[0m \u001b[43mw\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mstart\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    327\u001b[0m pool\u001b[38;5;241m.\u001b[39mappend(w)\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/multiprocessing/dummy/__init__.py:51\u001b[0m, in \u001b[0;36mDummyProcess.start\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m     50\u001b[0m     \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_parent\u001b[38;5;241m.\u001b[39m_children[\u001b[38;5;28mself\u001b[39m] \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mNone\u001b[39;00m\n\u001b[0;32m---> 51\u001b[0m \u001b[43mthreading\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mThread\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mstart\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/threading.py:899\u001b[0m, in \u001b[0;36mThread.start\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m    898\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[0;32m--> 899\u001b[0m     \u001b[43m_start_new_thread\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_bootstrap\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    900\u001b[0m \u001b[38;5;28;01mexcept\u001b[39;00m \u001b[38;5;167;01mException\u001b[39;00m:\n",
      "\u001b[0;31mRuntimeError\u001b[0m: can't start new thread",
      "\nDuring handling of the above exception, another exception occurred:\n",
      "\u001b[0;31mAttributeError\u001b[0m                            Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[10], line 51\u001b[0m\n\u001b[1;32m     48\u001b[0m \u001b[38;5;66;03m# Instantiate UMAP\u001b[39;00m\n\u001b[1;32m     49\u001b[0m umap_model \u001b[38;5;241m=\u001b[39m UMAP(n_components\u001b[38;5;241m=\u001b[39m\u001b[38;5;241m2\u001b[39m, random_state\u001b[38;5;241m=\u001b[39m\u001b[38;5;241m42\u001b[39m)\n\u001b[0;32m---> 51\u001b[0m all_sample_umap \u001b[38;5;241m=\u001b[39m \u001b[43mumap_model\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mfit_transform\u001b[49m\u001b[43m(\u001b[49m\u001b[43mnp\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mvstack\u001b[49m\u001b[43m(\u001b[49m\u001b[43m[\u001b[49m\u001b[43msource\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mtarget\u001b[49m\u001b[43m]\u001b[49m\u001b[43m)\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     52\u001b[0m source_umap \u001b[38;5;241m=\u001b[39m umap_model\u001b[38;5;241m.\u001b[39mtransform(source)\n\u001b[1;32m     53\u001b[0m target_umap \u001b[38;5;241m=\u001b[39m umap_model\u001b[38;5;241m.\u001b[39mtransform(target)\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/site-packages/umap/umap_.py:2928\u001b[0m, in \u001b[0;36mUMAP.fit_transform\u001b[0;34m(self, X, y, force_all_finite, **kwargs)\u001b[0m\n\u001b[1;32m   2890\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m\u001b[38;5;250m \u001b[39m\u001b[38;5;21mfit_transform\u001b[39m(\u001b[38;5;28mself\u001b[39m, X, y\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mNone\u001b[39;00m, force_all_finite\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mTrue\u001b[39;00m, \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mkwargs):\n\u001b[1;32m   2891\u001b[0m \u001b[38;5;250m    \u001b[39m\u001b[38;5;124;03m\"\"\"Fit X into an embedded space and return that transformed\u001b[39;00m\n\u001b[1;32m   2892\u001b[0m \u001b[38;5;124;03m    output.\u001b[39;00m\n\u001b[1;32m   2893\u001b[0m \n\u001b[0;32m   (...)\u001b[0m\n\u001b[1;32m   2926\u001b[0m \u001b[38;5;124;03m        Local radii of data points in the embedding (log-transformed).\u001b[39;00m\n\u001b[1;32m   2927\u001b[0m \u001b[38;5;124;03m    \"\"\"\u001b[39;00m\n\u001b[0;32m-> 2928\u001b[0m     \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mfit\u001b[49m\u001b[43m(\u001b[49m\u001b[43mX\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43my\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mforce_all_finite\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   2929\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mtransform_mode \u001b[38;5;241m==\u001b[39m \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124membedding\u001b[39m\u001b[38;5;124m\"\u001b[39m:\n\u001b[1;32m   2930\u001b[0m         \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39moutput_dens:\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/site-packages/umap/umap_.py:2635\u001b[0m, in \u001b[0;36mUMAP.fit\u001b[0;34m(self, X, y, force_all_finite, **kwargs)\u001b[0m\n\u001b[1;32m   2629\u001b[0m     nn_metric \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_input_distance_func\n\u001b[1;32m   2630\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mknn_dists \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m:\n\u001b[1;32m   2631\u001b[0m     (\n\u001b[1;32m   2632\u001b[0m         \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_knn_indices,\n\u001b[1;32m   2633\u001b[0m         \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_knn_dists,\n\u001b[1;32m   2634\u001b[0m         \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_knn_search_index,\n\u001b[0;32m-> 2635\u001b[0m     ) \u001b[38;5;241m=\u001b[39m \u001b[43mnearest_neighbors\u001b[49m\u001b[43m(\u001b[49m\n\u001b[1;32m   2636\u001b[0m \u001b[43m        \u001b[49m\u001b[43mX\u001b[49m\u001b[43m[\u001b[49m\u001b[43mindex\u001b[49m\u001b[43m]\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m   2637\u001b[0m \u001b[43m        \u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_n_neighbors\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m   2638\u001b[0m \u001b[43m        \u001b[49m\u001b[43mnn_metric\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m   2639\u001b[0m \u001b[43m        \u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_metric_kwds\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m   2640\u001b[0m \u001b[43m        \u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mangular_rp_forest\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m   2641\u001b[0m \u001b[43m        \u001b[49m\u001b[43mrandom_state\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m   2642\u001b[0m \u001b[43m        \u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mlow_memory\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m   2643\u001b[0m \u001b[43m        \u001b[49m\u001b[43muse_pynndescent\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43;01mTrue\u001b[39;49;00m\u001b[43m,\u001b[49m\n\u001b[1;32m   2644\u001b[0m \u001b[43m        \u001b[49m\u001b[43mn_jobs\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mn_jobs\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m   2645\u001b[0m \u001b[43m        \u001b[49m\u001b[43mverbose\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mverbose\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m   2646\u001b[0m \u001b[43m    \u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   2647\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[1;32m   2648\u001b[0m     \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_knn_indices \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mknn_indices\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/site-packages/umap/umap_.py:330\u001b[0m, in \u001b[0;36mnearest_neighbors\u001b[0;34m(X, n_neighbors, metric, metric_kwds, angular, random_state, low_memory, use_pynndescent, n_jobs, verbose)\u001b[0m\n\u001b[1;32m    327\u001b[0m     n_trees \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mmin\u001b[39m(\u001b[38;5;241m64\u001b[39m, \u001b[38;5;241m5\u001b[39m \u001b[38;5;241m+\u001b[39m \u001b[38;5;28mint\u001b[39m(\u001b[38;5;28mround\u001b[39m((X\u001b[38;5;241m.\u001b[39mshape[\u001b[38;5;241m0\u001b[39m]) \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39m \u001b[38;5;241m0.5\u001b[39m \u001b[38;5;241m/\u001b[39m \u001b[38;5;241m20.0\u001b[39m)))\n\u001b[1;32m    328\u001b[0m     n_iters \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mmax\u001b[39m(\u001b[38;5;241m5\u001b[39m, \u001b[38;5;28mint\u001b[39m(\u001b[38;5;28mround\u001b[39m(np\u001b[38;5;241m.\u001b[39mlog2(X\u001b[38;5;241m.\u001b[39mshape[\u001b[38;5;241m0\u001b[39m]))))\n\u001b[0;32m--> 330\u001b[0m     knn_search_index \u001b[38;5;241m=\u001b[39m \u001b[43mNNDescent\u001b[49m\u001b[43m(\u001b[49m\n\u001b[1;32m    331\u001b[0m \u001b[43m        \u001b[49m\u001b[43mX\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    332\u001b[0m \u001b[43m        \u001b[49m\u001b[43mn_neighbors\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mn_neighbors\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    333\u001b[0m \u001b[43m        \u001b[49m\u001b[43mmetric\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mmetric\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    334\u001b[0m \u001b[43m        \u001b[49m\u001b[43mmetric_kwds\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mmetric_kwds\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    335\u001b[0m \u001b[43m        \u001b[49m\u001b[43mrandom_state\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mrandom_state\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    336\u001b[0m \u001b[43m        \u001b[49m\u001b[43mn_trees\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mn_trees\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    337\u001b[0m \u001b[43m        \u001b[49m\u001b[43mn_iters\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mn_iters\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    338\u001b[0m \u001b[43m        \u001b[49m\u001b[43mmax_candidates\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m60\u001b[39;49m\u001b[43m,\u001b[49m\n\u001b[1;32m    339\u001b[0m \u001b[43m        \u001b[49m\u001b[43mlow_memory\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mlow_memory\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    340\u001b[0m \u001b[43m        \u001b[49m\u001b[43mn_jobs\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mn_jobs\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    341\u001b[0m \u001b[43m        \u001b[49m\u001b[43mverbose\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mverbose\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    342\u001b[0m \u001b[43m        \u001b[49m\u001b[43mcompressed\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43;01mFalse\u001b[39;49;00m\u001b[43m,\u001b[49m\n\u001b[1;32m    343\u001b[0m \u001b[43m    \u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    344\u001b[0m     knn_indices, knn_dists \u001b[38;5;241m=\u001b[39m knn_search_index\u001b[38;5;241m.\u001b[39mneighbor_graph\n\u001b[1;32m    346\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m verbose:\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/site-packages/pynndescent/pynndescent_.py:806\u001b[0m, in \u001b[0;36mNNDescent.__init__\u001b[0;34m(self, data, metric, metric_kwds, n_neighbors, n_trees, leaf_size, pruning_degree_multiplier, diversify_prob, n_search_trees, tree_init, init_graph, init_dist, random_state, low_memory, max_candidates, max_rptree_depth, n_iters, delta, n_jobs, compressed, parallel_batch_queries, verbose)\u001b[0m\n\u001b[1;32m    793\u001b[0m         \u001b[38;5;28mprint\u001b[39m(ts(), \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mBuilding RP forest with\u001b[39m\u001b[38;5;124m\"\u001b[39m, \u001b[38;5;28mstr\u001b[39m(n_trees), \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mtrees\u001b[39m\u001b[38;5;124m\"\u001b[39m)\n\u001b[1;32m    794\u001b[0m     \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_rp_forest \u001b[38;5;241m=\u001b[39m make_forest(\n\u001b[1;32m    795\u001b[0m         data,\n\u001b[1;32m    796\u001b[0m         n_neighbors,\n\u001b[0;32m   (...)\u001b[0m\n\u001b[1;32m    804\u001b[0m         max_depth\u001b[38;5;241m=\u001b[39m\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mmax_rptree_depth,\n\u001b[1;32m    805\u001b[0m     )\n\u001b[0;32m--> 806\u001b[0m     leaf_array \u001b[38;5;241m=\u001b[39m \u001b[43mrptree_leaf_array\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_rp_forest\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    807\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[1;32m    808\u001b[0m     \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_rp_forest \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mNone\u001b[39;00m\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/site-packages/pynndescent/rp_trees.py:1436\u001b[0m, in \u001b[0;36mrptree_leaf_array\u001b[0;34m(rp_forest)\u001b[0m\n\u001b[1;32m   1434\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m\u001b[38;5;250m \u001b[39m\u001b[38;5;21mrptree_leaf_array\u001b[39m(rp_forest):\n\u001b[1;32m   1435\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mlen\u001b[39m(rp_forest) \u001b[38;5;241m>\u001b[39m \u001b[38;5;241m0\u001b[39m:\n\u001b[0;32m-> 1436\u001b[0m         \u001b[38;5;28;01mreturn\u001b[39;00m np\u001b[38;5;241m.\u001b[39mvstack(\u001b[43mrptree_leaf_array_parallel\u001b[49m\u001b[43m(\u001b[49m\u001b[43mrp_forest\u001b[49m\u001b[43m)\u001b[49m)\n\u001b[1;32m   1437\u001b[0m     \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[1;32m   1438\u001b[0m         \u001b[38;5;28;01mreturn\u001b[39;00m np\u001b[38;5;241m.\u001b[39marray([[\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39m]])\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/site-packages/pynndescent/rp_trees.py:1428\u001b[0m, in \u001b[0;36mrptree_leaf_array_parallel\u001b[0;34m(rp_forest)\u001b[0m\n\u001b[1;32m   1426\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m\u001b[38;5;250m \u001b[39m\u001b[38;5;21mrptree_leaf_array_parallel\u001b[39m(rp_forest):\n\u001b[1;32m   1427\u001b[0m     max_leaf_size \u001b[38;5;241m=\u001b[39m np\u001b[38;5;241m.\u001b[39mmax([rp_tree\u001b[38;5;241m.\u001b[39mleaf_size \u001b[38;5;28;01mfor\u001b[39;00m rp_tree \u001b[38;5;129;01min\u001b[39;00m rp_forest])\n\u001b[0;32m-> 1428\u001b[0m     result \u001b[38;5;241m=\u001b[39m \u001b[43mjoblib\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mParallel\u001b[49m\u001b[43m(\u001b[49m\u001b[43mn_jobs\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;241;43m-\u001b[39;49m\u001b[38;5;241;43m1\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mrequire\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;124;43m\"\u001b[39;49m\u001b[38;5;124;43msharedmem\u001b[39;49m\u001b[38;5;124;43m\"\u001b[39;49m\u001b[43m)\u001b[49m\u001b[43m(\u001b[49m\n\u001b[1;32m   1429\u001b[0m \u001b[43m        \u001b[49m\u001b[43mjoblib\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mdelayed\u001b[49m\u001b[43m(\u001b[49m\u001b[43mget_leaves_from_tree\u001b[49m\u001b[43m)\u001b[49m\u001b[43m(\u001b[49m\u001b[43mrp_tree\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mmax_leaf_size\u001b[49m\u001b[43m)\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43;01mfor\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[43mrp_tree\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;129;43;01min\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[43mrp_forest\u001b[49m\n\u001b[1;32m   1430\u001b[0m \u001b[43m    \u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1431\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m result\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/site-packages/joblib/parallel.py:2070\u001b[0m, in \u001b[0;36mParallel.__call__\u001b[0;34m(self, iterable)\u001b[0m\n\u001b[1;32m   2064\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_call_ref \u001b[38;5;241m=\u001b[39m weakref\u001b[38;5;241m.\u001b[39mref(output)\n\u001b[1;32m   2066\u001b[0m \u001b[38;5;66;03m# The first item from the output is blank, but it makes the interpreter\u001b[39;00m\n\u001b[1;32m   2067\u001b[0m \u001b[38;5;66;03m# progress until it enters the Try/Except block of the generator and\u001b[39;00m\n\u001b[1;32m   2068\u001b[0m \u001b[38;5;66;03m# reaches the first `yield` statement. This starts the asynchronous\u001b[39;00m\n\u001b[1;32m   2069\u001b[0m \u001b[38;5;66;03m# dispatch of the tasks to the workers.\u001b[39;00m\n\u001b[0;32m-> 2070\u001b[0m \u001b[38;5;28;43mnext\u001b[39;49m\u001b[43m(\u001b[49m\u001b[43moutput\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   2072\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m output \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mreturn_generator \u001b[38;5;28;01melse\u001b[39;00m \u001b[38;5;28mlist\u001b[39m(output)\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/site-packages/joblib/parallel.py:1675\u001b[0m, in \u001b[0;36mParallel._get_outputs\u001b[0;34m(self, iterator, pre_dispatch)\u001b[0m\n\u001b[1;32m   1673\u001b[0m detach_generator_exit \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mFalse\u001b[39;00m\n\u001b[1;32m   1674\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[0;32m-> 1675\u001b[0m     \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_start\u001b[49m\u001b[43m(\u001b[49m\u001b[43miterator\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mpre_dispatch\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1676\u001b[0m     \u001b[38;5;66;03m# first yield returns None, for internal use only. This ensures\u001b[39;00m\n\u001b[1;32m   1677\u001b[0m     \u001b[38;5;66;03m# that we enter the try/except block and start dispatching the\u001b[39;00m\n\u001b[1;32m   1678\u001b[0m     \u001b[38;5;66;03m# tasks.\u001b[39;00m\n\u001b[1;32m   1679\u001b[0m     \u001b[38;5;28;01myield\u001b[39;00m\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/site-packages/joblib/parallel.py:1658\u001b[0m, in \u001b[0;36mParallel._start\u001b[0;34m(self, iterator, pre_dispatch)\u001b[0m\n\u001b[1;32m   1649\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m\u001b[38;5;250m \u001b[39m\u001b[38;5;21m_start\u001b[39m(\u001b[38;5;28mself\u001b[39m, iterator, pre_dispatch):\n\u001b[1;32m   1650\u001b[0m     \u001b[38;5;66;03m# Only set self._iterating to True if at least a batch\u001b[39;00m\n\u001b[1;32m   1651\u001b[0m     \u001b[38;5;66;03m# was dispatched. In particular this covers the edge\u001b[39;00m\n\u001b[0;32m   (...)\u001b[0m\n\u001b[1;32m   1655\u001b[0m     \u001b[38;5;66;03m# was very quick and its callback already dispatched all the\u001b[39;00m\n\u001b[1;32m   1656\u001b[0m     \u001b[38;5;66;03m# remaining jobs.\u001b[39;00m\n\u001b[1;32m   1657\u001b[0m     \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_iterating \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mFalse\u001b[39;00m\n\u001b[0;32m-> 1658\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mdispatch_one_batch\u001b[49m\u001b[43m(\u001b[49m\u001b[43miterator\u001b[49m\u001b[43m)\u001b[49m:\n\u001b[1;32m   1659\u001b[0m         \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_iterating \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_original_iterator \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m\n\u001b[1;32m   1661\u001b[0m     \u001b[38;5;28;01mwhile\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mdispatch_one_batch(iterator):\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/site-packages/joblib/parallel.py:1540\u001b[0m, in \u001b[0;36mParallel.dispatch_one_batch\u001b[0;34m(self, iterator)\u001b[0m\n\u001b[1;32m   1538\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28;01mFalse\u001b[39;00m\n\u001b[1;32m   1539\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[0;32m-> 1540\u001b[0m     \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_dispatch\u001b[49m\u001b[43m(\u001b[49m\u001b[43mtasks\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1541\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28;01mTrue\u001b[39;00m\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/site-packages/joblib/parallel.py:1437\u001b[0m, in \u001b[0;36mParallel._dispatch\u001b[0;34m(self, batch)\u001b[0m\n\u001b[1;32m   1430\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_register_new_job(batch_tracker)\n\u001b[1;32m   1432\u001b[0m \u001b[38;5;66;03m# If return_ordered is False, the batch_tracker is not stored in the\u001b[39;00m\n\u001b[1;32m   1433\u001b[0m \u001b[38;5;66;03m# jobs queue at the time of submission. Instead, it will be appended to\u001b[39;00m\n\u001b[1;32m   1434\u001b[0m \u001b[38;5;66;03m# the queue by itself as soon as the callback is triggered to be able\u001b[39;00m\n\u001b[1;32m   1435\u001b[0m \u001b[38;5;66;03m# to return the results in the order of completion.\u001b[39;00m\n\u001b[0;32m-> 1437\u001b[0m job \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_backend\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43msubmit\u001b[49m\u001b[43m(\u001b[49m\u001b[43mbatch\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mcallback\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mbatch_tracker\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1438\u001b[0m batch_tracker\u001b[38;5;241m.\u001b[39mregister_job(job)\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/site-packages/joblib/_parallel_backends.py:339\u001b[0m, in \u001b[0;36mPoolManagerMixin.submit\u001b[0;34m(self, func, callback)\u001b[0m\n\u001b[1;32m    335\u001b[0m \u001b[38;5;250m\u001b[39m\u001b[38;5;124;03m\"\"\"Schedule a func to be run\"\"\"\u001b[39;00m\n\u001b[1;32m    336\u001b[0m \u001b[38;5;66;03m# Here, we need a wrapper to avoid crashes on KeyboardInterruptErrors.\u001b[39;00m\n\u001b[1;32m    337\u001b[0m \u001b[38;5;66;03m# We also call the callback on error, to make sure the pool does not\u001b[39;00m\n\u001b[1;32m    338\u001b[0m \u001b[38;5;66;03m# wait on crashed jobs.\u001b[39;00m\n\u001b[0;32m--> 339\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_get_pool\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\u001b[38;5;241m.\u001b[39mapply_async(\n\u001b[1;32m    340\u001b[0m     _TracebackCapturingWrapper(func),\n\u001b[1;32m    341\u001b[0m     (),\n\u001b[1;32m    342\u001b[0m     callback\u001b[38;5;241m=\u001b[39mcallback,\n\u001b[1;32m    343\u001b[0m     error_callback\u001b[38;5;241m=\u001b[39mcallback,\n\u001b[1;32m    344\u001b[0m )\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/site-packages/joblib/_parallel_backends.py:507\u001b[0m, in \u001b[0;36mThreadingBackend._get_pool\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m    501\u001b[0m \u001b[38;5;250m\u001b[39m\u001b[38;5;124;03m\"\"\"Lazily initialize the thread pool\u001b[39;00m\n\u001b[1;32m    502\u001b[0m \n\u001b[1;32m    503\u001b[0m \u001b[38;5;124;03mThe actual pool of worker threads is only initialized at the first\u001b[39;00m\n\u001b[1;32m    504\u001b[0m \u001b[38;5;124;03mcall to apply_async.\u001b[39;00m\n\u001b[1;32m    505\u001b[0m \u001b[38;5;124;03m\"\"\"\u001b[39;00m\n\u001b[1;32m    506\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_pool \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m:\n\u001b[0;32m--> 507\u001b[0m     \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_pool \u001b[38;5;241m=\u001b[39m \u001b[43mThreadPool\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_n_jobs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    508\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_pool\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/multiprocessing/pool.py:927\u001b[0m, in \u001b[0;36mThreadPool.__init__\u001b[0;34m(self, processes, initializer, initargs)\u001b[0m\n\u001b[1;32m    926\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m\u001b[38;5;250m \u001b[39m\u001b[38;5;21m__init__\u001b[39m(\u001b[38;5;28mself\u001b[39m, processes\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mNone\u001b[39;00m, initializer\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mNone\u001b[39;00m, initargs\u001b[38;5;241m=\u001b[39m()):\n\u001b[0;32m--> 927\u001b[0m     \u001b[43mPool\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[38;5;21;43m__init__\u001b[39;49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mprocesses\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43minitializer\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43minitargs\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m~/here/miniconda3/envs/scgen-env/lib/python3.9/multiprocessing/pool.py:216\u001b[0m, in \u001b[0;36mPool.__init__\u001b[0;34m(self, processes, initializer, initargs, maxtasksperchild, context)\u001b[0m\n\u001b[1;32m    214\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m p \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_pool:\n\u001b[1;32m    215\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m p\u001b[38;5;241m.\u001b[39mexitcode \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m:\n\u001b[0;32m--> 216\u001b[0m         \u001b[43mp\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mterminate\u001b[49m()\n\u001b[1;32m    217\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m p \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_pool:\n\u001b[1;32m    218\u001b[0m     p\u001b[38;5;241m.\u001b[39mjoin()\n",
      "\u001b[0;31mAttributeError\u001b[0m: 'DummyProcess' object has no attribute 'terminate'"
     ]
    }
   ],
   "source": [
    "drug = \"Vem\"\n",
    "X_pre, X_post = prepare_pair_from_mat('WM902B', 'DMSO','24h', drug, '72h')\n",
    "jfe_indices = [1, 6, 0, 5, 4, 7, 8, 2, 3, 19]  \n",
    "\n",
    "print(\"X_pre cells:\", X_pre.shape)\n",
    "print(\"X_post cells:\", X_post.shape)\n",
    "\n",
    "X_tr_pre, X_te_pre, Y_tr_post, Y_te_post = split_train_test(X_pre, X_post, 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",
    "# 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",
    "\n",
    "\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 = run_cellot_pair(X_tr_pre[:, jfe_indices], Y_tr_post[:, jfe_indices], X_te_pre[:, jfe_indices], Y_te_post[:, jfe_indices], n_epochs=2000)\n",
    "    metrics = summarize_metrics(out[\"y_pred\"], Y_te_post[:, jfe_indices], median_gamma)\n",
    "    print(f\"Run {run} metrics: {metrics}\")\n",
    "    all_metrics.append(metrics)\n",
    "\n",
    "# Results summary\n",
    "df = pd.DataFrame(all_metrics)\n",
    "print(df.describe().T[['mean', 'std']].round(4))\n",
    "\n",
    "\n",
    "from umap import UMAP\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "source = Y_tr_post[:, jfe_indices]\n",
    "target = Y_te_post[:, jfe_indices]\n",
    "predicted = out.get('y_pred') \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([source, target]))\n",
    "source_umap = umap_model.transform(source)\n",
    "target_umap = umap_model.transform(target)\n",
    "y_pred_umap = umap_model.transform(predicted)\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=\"#C88131\")\n",
    "ax.scatter(y_pred_umap[:, 0], y_pred_umap[:, 1], s=10, alpha=0.7, label='predicted cells', color=\"#1F4D8D\")\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/cellot_on_4i_drug_{drug}.png\", dpi=300)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9747cc38",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "610e4869",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Cell line:  SKMEL19\n",
      "['DMSO' 'Vem' 'Vem+Tram']\n",
      "X_pre cells: (2677, 20)\n",
      "X_post cells: (2677, 20)\n",
      "(2141, 20)\n",
      "(536, 20)\n",
      "(2141, 20)\n",
      "(536, 20)\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Median heuristic gamma: 0.10487158680337198\n",
      "**************** Run: 0 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=0 f_loss=-4652797.0000 g_loss=5776148.0000 | train mmd=0.1318 | test_mmd=0.6987\n",
      "[CellOT] epoch=50 f_loss=269.4297 g_loss=1047.7224 | train mmd=0.6801 | test_mmd=0.2498\n",
      "[CellOT] epoch=100 f_loss=-113.9228 g_loss=345.6044 | train mmd=0.7146 | test_mmd=0.1999\n",
      "[CellOT] epoch=150 f_loss=-320.9928 g_loss=456.1716 | train mmd=0.5992 | test_mmd=0.1435\n",
      "[CellOT] epoch=200 f_loss=-429.0464 g_loss=578.6300 | train mmd=0.4610 | test_mmd=0.0924\n",
      "[CellOT] epoch=250 f_loss=-496.7188 g_loss=801.4954 | train mmd=0.3231 | test_mmd=0.0917\n",
      "[CellOT] epoch=300 f_loss=-594.7444 g_loss=769.6606 | train mmd=0.5553 | test_mmd=0.1042\n",
      "[CellOT] epoch=350 f_loss=-642.6067 g_loss=886.4951 | train mmd=0.3355 | test_mmd=0.0638\n",
      "[CellOT] epoch=400 f_loss=-650.0369 g_loss=987.3616 | train mmd=0.4782 | test_mmd=0.0762\n",
      "[CellOT] epoch=450 f_loss=-709.6603 g_loss=1119.4773 | train mmd=0.3206 | test_mmd=0.0450\n",
      "[CellOT] epoch=500 f_loss=-850.8151 g_loss=1173.0146 | train mmd=0.4670 | test_mmd=0.0697\n",
      "[CellOT] epoch=550 f_loss=-814.7144 g_loss=1307.5809 | train mmd=0.3984 | test_mmd=0.0518\n",
      "[CellOT] epoch=600 f_loss=-703.3341 g_loss=1325.8351 | train mmd=0.3895 | test_mmd=0.0501\n",
      "[CellOT] epoch=650 f_loss=-711.0400 g_loss=1345.1567 | train mmd=0.3592 | test_mmd=0.0349\n",
      "[CellOT] epoch=700 f_loss=-1401.2524 g_loss=1140.8545 | train mmd=0.2465 | test_mmd=0.0460\n",
      "[CellOT] epoch=750 f_loss=-534.7891 g_loss=1297.8276 | train mmd=0.3017 | test_mmd=0.0232\n",
      "[CellOT] epoch=800 f_loss=-496.6761 g_loss=1213.4634 | train mmd=0.1960 | test_mmd=0.0295\n",
      "[CellOT] epoch=850 f_loss=-508.8551 g_loss=1340.0046 | train mmd=0.2409 | test_mmd=0.0328\n",
      "[CellOT] epoch=900 f_loss=-114.0420 g_loss=937.1354 | train mmd=0.1416 | test_mmd=0.0929\n",
      "[CellOT] epoch=950 f_loss=-252.7941 g_loss=1294.5757 | train mmd=0.2818 | test_mmd=0.0232\n",
      "[CellOT] epoch=1000 f_loss=-1235.7600 g_loss=5236.6221 | train mmd=0.3796 | test_mmd=0.1268\n",
      "[CellOT] epoch=1050 f_loss=-144.1848 g_loss=1035.8323 | train mmd=0.2603 | test_mmd=0.0275\n",
      "[CellOT] epoch=1100 f_loss=392.3425 g_loss=619.7416 | train mmd=0.2053 | test_mmd=0.0748\n",
      "[CellOT] epoch=1150 f_loss=22.5370 g_loss=841.0580 | train mmd=0.1981 | test_mmd=0.0264\n",
      "[CellOT] epoch=1200 f_loss=-1765.1865 g_loss=765.3708 | train mmd=0.2215 | test_mmd=0.0645\n",
      "[CellOT] epoch=1250 f_loss=165.9413 g_loss=679.9312 | train mmd=0.2063 | test_mmd=0.0262\n",
      "[CellOT] epoch=1300 f_loss=373.6150 g_loss=735.4655 | train mmd=0.2170 | test_mmd=0.0299\n",
      "[CellOT] epoch=1350 f_loss=310.5590 g_loss=689.8095 | train mmd=0.2911 | test_mmd=0.0873\n",
      "[CellOT] epoch=1400 f_loss=200.8583 g_loss=657.5186 | train mmd=0.2926 | test_mmd=0.0766\n",
      "[CellOT] epoch=1450 f_loss=32.7505 g_loss=725.7121 | train mmd=0.2232 | test_mmd=0.1161\n",
      "[CellOT] epoch=1500 f_loss=74.1795 g_loss=744.7770 | train mmd=0.1638 | test_mmd=0.0257\n",
      "[CellOT] epoch=1550 f_loss=74.5453 g_loss=800.4076 | train mmd=0.1977 | test_mmd=0.0316\n",
      "[CellOT] epoch=1600 f_loss=90.7867 g_loss=867.3439 | train mmd=0.1334 | test_mmd=0.0406\n",
      "[CellOT] epoch=1650 f_loss=28.4837 g_loss=986.4269 | train mmd=0.1830 | test_mmd=0.0701\n",
      "[CellOT] epoch=1700 f_loss=-308.3263 g_loss=932.9881 | train mmd=0.4491 | test_mmd=0.1973\n",
      "[CellOT] epoch=1750 f_loss=55.1687 g_loss=933.8301 | train mmd=0.2703 | test_mmd=0.0294\n",
      "[CellOT] epoch=1800 f_loss=50.6717 g_loss=1004.8019 | train mmd=0.1814 | test_mmd=0.0866\n",
      "[CellOT] epoch=1850 f_loss=16.9264 g_loss=1105.7495 | train mmd=0.3796 | test_mmd=0.1019\n",
      "[CellOT] epoch=1900 f_loss=27.8872 g_loss=1095.1075 | train mmd=0.3572 | test_mmd=0.2568\n",
      "[CellOT] epoch=1950 f_loss=25.8795 g_loss=1116.4465 | train mmd=0.2740 | test_mmd=0.0303\n",
      "[CellOT] epoch=2000 f_loss=13.3014 g_loss=1172.6533 | train mmd=0.2110 | test_mmd=0.0472\n",
      "[CellOT] Final CellOT MMD: 0.0977\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n",
      "[CellOT] epoch=0 f_loss=-4154182.0000 g_loss=4497466.0000 | train mmd=0.1318 | test_mmd=0.6987\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 0 metrics: {'mmd2_gamma_median': 0.04720326284342091, 'mmd2_gamma_0.5': 0.11202558641149418, 'mmd2_gamma_1.0': 0.18028495574159964, 'wasserstein_distance': 1.3199191192744086, 'R2_feature_means': 0.8965939008851268}\n",
      "**************** Run: 1 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=50 f_loss=507.7241 g_loss=1100.2688 | train mmd=0.7535 | test_mmd=0.2360\n",
      "[CellOT] epoch=100 f_loss=-52.2796 g_loss=329.4726 | train mmd=0.7015 | test_mmd=0.1924\n",
      "[CellOT] epoch=150 f_loss=-349.8020 g_loss=451.5379 | train mmd=0.7248 | test_mmd=0.1859\n",
      "[CellOT] epoch=200 f_loss=-529.7360 g_loss=568.5678 | train mmd=0.6491 | test_mmd=0.1438\n",
      "[CellOT] epoch=250 f_loss=-573.2266 g_loss=1367.8479 | train mmd=0.5751 | test_mmd=0.1212\n",
      "[CellOT] epoch=300 f_loss=-723.7170 g_loss=879.2603 | train mmd=0.6043 | test_mmd=0.1194\n",
      "[CellOT] epoch=350 f_loss=-4933.0107 g_loss=709.9821 | train mmd=0.7247 | test_mmd=0.9006\n",
      "[CellOT] epoch=400 f_loss=-816.3004 g_loss=967.8734 | train mmd=0.4625 | test_mmd=0.0727\n",
      "[CellOT] epoch=450 f_loss=-806.9844 g_loss=1180.0946 | train mmd=0.4930 | test_mmd=0.0785\n",
      "[CellOT] epoch=500 f_loss=-732.7183 g_loss=2265.0254 | train mmd=0.2615 | test_mmd=0.0419\n",
      "[CellOT] epoch=550 f_loss=-1286.0210 g_loss=1274.7920 | train mmd=0.4435 | test_mmd=0.0755\n",
      "[CellOT] epoch=600 f_loss=-945.4094 g_loss=1557.8627 | train mmd=0.4355 | test_mmd=0.0507\n",
      "[CellOT] epoch=650 f_loss=-969.7445 g_loss=1417.8542 | train mmd=0.4257 | test_mmd=0.0696\n",
      "[CellOT] epoch=700 f_loss=-634.3313 g_loss=1440.1448 | train mmd=0.2432 | test_mmd=0.0398\n",
      "[CellOT] epoch=750 f_loss=-839.5948 g_loss=7318.4277 | train mmd=0.4265 | test_mmd=0.3043\n",
      "[CellOT] epoch=800 f_loss=-572.7909 g_loss=1555.7468 | train mmd=0.2318 | test_mmd=0.0464\n",
      "[CellOT] epoch=850 f_loss=-789.4807 g_loss=1662.3066 | train mmd=0.2835 | test_mmd=0.0326\n",
      "[CellOT] epoch=900 f_loss=-588.6101 g_loss=1723.5422 | train mmd=0.3669 | test_mmd=0.0256\n",
      "[CellOT] epoch=950 f_loss=-326.5732 g_loss=1722.0844 | train mmd=0.2997 | test_mmd=0.0268\n",
      "[CellOT] epoch=1000 f_loss=-412.5696 g_loss=1653.0471 | train mmd=0.3179 | test_mmd=0.0272\n",
      "[CellOT] epoch=1050 f_loss=-302.7099 g_loss=1659.4763 | train mmd=0.2387 | test_mmd=0.0175\n",
      "[CellOT] epoch=1100 f_loss=-30.9988 g_loss=1550.8425 | train mmd=0.2156 | test_mmd=0.0177\n",
      "[CellOT] epoch=1150 f_loss=134.8300 g_loss=1223.3580 | train mmd=0.2260 | test_mmd=0.0520\n",
      "[CellOT] epoch=1200 f_loss=108.5760 g_loss=1135.9595 | train mmd=0.2437 | test_mmd=0.0256\n",
      "[CellOT] epoch=1250 f_loss=322.8239 g_loss=1108.0536 | train mmd=0.1907 | test_mmd=0.0664\n",
      "[CellOT] epoch=1300 f_loss=-96.8000 g_loss=896.3709 | train mmd=0.2787 | test_mmd=0.0398\n",
      "[CellOT] epoch=1350 f_loss=66.1264 g_loss=626.4651 | train mmd=0.2009 | test_mmd=0.0481\n",
      "[CellOT] epoch=1400 f_loss=361.4371 g_loss=783.5211 | train mmd=0.2685 | test_mmd=0.0649\n",
      "[CellOT] epoch=1450 f_loss=520.6074 g_loss=551.9397 | train mmd=0.1736 | test_mmd=0.0389\n",
      "[CellOT] epoch=1500 f_loss=329.3718 g_loss=590.1223 | train mmd=0.2100 | test_mmd=0.0539\n",
      "[CellOT] epoch=1550 f_loss=141.7249 g_loss=731.0364 | train mmd=0.1794 | test_mmd=0.0580\n",
      "[CellOT] epoch=1600 f_loss=69.9445 g_loss=991.4020 | train mmd=0.2343 | test_mmd=0.1130\n",
      "[CellOT] epoch=1650 f_loss=52.9309 g_loss=889.8240 | train mmd=0.1997 | test_mmd=0.0656\n",
      "[CellOT] epoch=1700 f_loss=62.2393 g_loss=1019.3563 | train mmd=0.4972 | test_mmd=0.2248\n",
      "[CellOT] epoch=1750 f_loss=76.2744 g_loss=1025.7200 | train mmd=0.3566 | test_mmd=0.1792\n",
      "[CellOT] epoch=1800 f_loss=27.3962 g_loss=1120.2483 | train mmd=0.1885 | test_mmd=0.1522\n",
      "[CellOT] epoch=1850 f_loss=39.5521 g_loss=1124.6974 | train mmd=0.1834 | test_mmd=0.1204\n",
      "[CellOT] epoch=1900 f_loss=32.6498 g_loss=1180.5408 | train mmd=0.1735 | test_mmd=0.0339\n",
      "[CellOT] epoch=1950 f_loss=-7.0422 g_loss=1217.9585 | train mmd=0.3986 | test_mmd=0.2252\n",
      "[CellOT] epoch=2000 f_loss=17.1302 g_loss=1274.6409 | train mmd=0.2399 | test_mmd=0.0843\n",
      "[CellOT] Final CellOT MMD: 0.1006\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n",
      "[CellOT] epoch=0 f_loss=-4210270.0000 g_loss=4869083.5000 | train mmd=0.1318 | test_mmd=0.6987\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 1 metrics: {'mmd2_gamma_median': 0.08434185283547069, 'mmd2_gamma_0.5': 0.2151429930464709, 'mmd2_gamma_1.0': 0.24677923051445444, 'wasserstein_distance': 1.229153091030545, 'R2_feature_means': 0.909138430524356}\n",
      "**************** Run: 2 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=50 f_loss=-828.2010 g_loss=413.4328 | train mmd=0.4234 | test_mmd=0.5085\n",
      "[CellOT] epoch=100 f_loss=-154.3372 g_loss=271.3845 | train mmd=0.6687 | test_mmd=0.1781\n",
      "[CellOT] epoch=150 f_loss=-334.4213 g_loss=432.4973 | train mmd=0.5289 | test_mmd=0.1461\n",
      "[CellOT] epoch=200 f_loss=-451.0247 g_loss=525.6388 | train mmd=0.4781 | test_mmd=0.1142\n",
      "[CellOT] epoch=250 f_loss=-507.6888 g_loss=645.4148 | train mmd=0.5636 | test_mmd=0.1175\n",
      "[CellOT] epoch=300 f_loss=-760.6851 g_loss=758.6970 | train mmd=0.4328 | test_mmd=0.1107\n",
      "[CellOT] epoch=350 f_loss=-645.2349 g_loss=906.5718 | train mmd=0.5090 | test_mmd=0.1076\n",
      "[CellOT] epoch=400 f_loss=-847.5602 g_loss=1126.8468 | train mmd=0.5633 | test_mmd=0.1062\n",
      "[CellOT] epoch=450 f_loss=-790.4156 g_loss=1168.5181 | train mmd=0.3920 | test_mmd=0.0712\n",
      "[CellOT] epoch=500 f_loss=-848.1615 g_loss=1291.6272 | train mmd=0.5089 | test_mmd=0.0749\n",
      "[CellOT] epoch=550 f_loss=-889.3603 g_loss=1830.3447 | train mmd=0.4600 | test_mmd=0.0705\n",
      "[CellOT] epoch=600 f_loss=-929.0165 g_loss=1404.3311 | train mmd=0.4532 | test_mmd=0.0587\n",
      "[CellOT] epoch=650 f_loss=-781.2191 g_loss=1588.5474 | train mmd=0.4087 | test_mmd=0.0504\n",
      "[CellOT] epoch=700 f_loss=-863.4508 g_loss=1485.6627 | train mmd=0.2948 | test_mmd=0.0631\n",
      "[CellOT] epoch=750 f_loss=-766.4240 g_loss=1692.8939 | train mmd=0.3923 | test_mmd=0.0552\n",
      "[CellOT] epoch=800 f_loss=-453.4001 g_loss=1259.6313 | train mmd=0.2384 | test_mmd=0.0431\n",
      "[CellOT] epoch=850 f_loss=-486.1242 g_loss=1397.3655 | train mmd=0.3495 | test_mmd=0.0275\n",
      "[CellOT] epoch=900 f_loss=-1239.7589 g_loss=1319.1260 | train mmd=0.2688 | test_mmd=0.0454\n",
      "[CellOT] epoch=950 f_loss=-617.9871 g_loss=1098.5747 | train mmd=0.2427 | test_mmd=0.0273\n",
      "[CellOT] epoch=1000 f_loss=-375.6228 g_loss=1690.5076 | train mmd=0.2160 | test_mmd=0.0190\n",
      "[CellOT] epoch=1050 f_loss=-147.3509 g_loss=1479.1492 | train mmd=0.2677 | test_mmd=0.0503\n",
      "[CellOT] epoch=1100 f_loss=-186.3689 g_loss=1511.0381 | train mmd=0.2665 | test_mmd=0.0211\n",
      "[CellOT] epoch=1150 f_loss=-329.2742 g_loss=2646.2349 | train mmd=0.1850 | test_mmd=0.0984\n",
      "[CellOT] epoch=1200 f_loss=-265.8681 g_loss=1479.1375 | train mmd=0.2797 | test_mmd=0.0418\n",
      "[CellOT] epoch=1250 f_loss=34.3900 g_loss=1319.8228 | train mmd=0.2845 | test_mmd=0.0508\n",
      "[CellOT] epoch=1300 f_loss=230.2418 g_loss=1138.7716 | train mmd=0.2438 | test_mmd=0.0518\n",
      "[CellOT] epoch=1350 f_loss=264.4026 g_loss=842.4949 | train mmd=0.2468 | test_mmd=0.0339\n",
      "[CellOT] epoch=1400 f_loss=348.2764 g_loss=577.3723 | train mmd=0.2115 | test_mmd=0.0607\n",
      "[CellOT] epoch=1450 f_loss=388.8079 g_loss=512.4824 | train mmd=0.3144 | test_mmd=0.0747\n",
      "[CellOT] epoch=1500 f_loss=190.0981 g_loss=778.1786 | train mmd=0.3081 | test_mmd=0.1476\n",
      "[CellOT] epoch=1550 f_loss=83.7100 g_loss=832.5861 | train mmd=0.2169 | test_mmd=0.0406\n",
      "[CellOT] epoch=1600 f_loss=0.4631 g_loss=1156.3740 | train mmd=0.3807 | test_mmd=0.1969\n",
      "[CellOT] epoch=1650 f_loss=86.9157 g_loss=1134.8197 | train mmd=0.1843 | test_mmd=0.1026\n",
      "[CellOT] epoch=1700 f_loss=33.8951 g_loss=1242.4569 | train mmd=0.3837 | test_mmd=0.0970\n",
      "[CellOT] epoch=1750 f_loss=50.6653 g_loss=1273.2710 | train mmd=0.2214 | test_mmd=0.0430\n",
      "[CellOT] epoch=1800 f_loss=61.5632 g_loss=1457.2496 | train mmd=0.2641 | test_mmd=0.0676\n",
      "[CellOT] epoch=1850 f_loss=28.0554 g_loss=1441.6187 | train mmd=0.3048 | test_mmd=0.1040\n",
      "[CellOT] epoch=1900 f_loss=55.9233 g_loss=1513.5585 | train mmd=0.2558 | test_mmd=0.1097\n",
      "[CellOT] epoch=1950 f_loss=2.1204 g_loss=1649.6371 | train mmd=0.3038 | test_mmd=0.1316\n",
      "[CellOT] epoch=2000 f_loss=26.8204 g_loss=1651.0415 | train mmd=0.3962 | test_mmd=0.2310\n",
      "[CellOT] Final CellOT MMD: 0.1827\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n",
      "[CellOT] epoch=0 f_loss=-4846829.0000 g_loss=5491219.0000 | train mmd=0.1318 | test_mmd=0.6987\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 2 metrics: {'mmd2_gamma_median': 0.23103313970075812, 'mmd2_gamma_0.5': 0.3880725958589976, 'mmd2_gamma_1.0': 0.37971293014588514, 'wasserstein_distance': 1.6701325581132482, 'R2_feature_means': 0.6920864336448382}\n",
      "**************** Run: 3 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=50 f_loss=-925.5743 g_loss=226.6459 | train mmd=0.4656 | test_mmd=0.4316\n",
      "[CellOT] epoch=100 f_loss=-102.7266 g_loss=522.9037 | train mmd=0.6877 | test_mmd=0.1789\n",
      "[CellOT] epoch=150 f_loss=-467.6417 g_loss=568.8029 | train mmd=0.6845 | test_mmd=0.1857\n",
      "[CellOT] epoch=200 f_loss=-695.6797 g_loss=767.9146 | train mmd=0.6269 | test_mmd=0.1406\n",
      "[CellOT] epoch=250 f_loss=-1323.9581 g_loss=965.8571 | train mmd=0.3316 | test_mmd=0.1229\n",
      "[CellOT] epoch=300 f_loss=-1166.3610 g_loss=1012.3077 | train mmd=0.3919 | test_mmd=0.1022\n",
      "[CellOT] epoch=350 f_loss=-995.8383 g_loss=1033.1221 | train mmd=0.4292 | test_mmd=0.0769\n",
      "[CellOT] epoch=400 f_loss=-1041.4880 g_loss=1130.2089 | train mmd=0.2449 | test_mmd=0.0613\n",
      "[CellOT] epoch=450 f_loss=-2339.6958 g_loss=8243.7031 | train mmd=0.5211 | test_mmd=0.2914\n",
      "[CellOT] epoch=500 f_loss=-1218.6289 g_loss=2686.7266 | train mmd=0.3221 | test_mmd=0.0645\n",
      "[CellOT] epoch=550 f_loss=-1118.5837 g_loss=1668.1960 | train mmd=0.3784 | test_mmd=0.0518\n",
      "[CellOT] epoch=600 f_loss=-2178.1934 g_loss=1158.6115 | train mmd=0.2677 | test_mmd=0.0490\n",
      "[CellOT] epoch=650 f_loss=-6764.6392 g_loss=1301.4592 | train mmd=0.2287 | test_mmd=0.1057\n",
      "[CellOT] epoch=700 f_loss=-584.3344 g_loss=1840.1240 | train mmd=0.3360 | test_mmd=0.0341\n",
      "[CellOT] epoch=750 f_loss=-787.5393 g_loss=2169.2708 | train mmd=0.3416 | test_mmd=0.0347\n",
      "[CellOT] epoch=800 f_loss=-770.9095 g_loss=1956.4231 | train mmd=0.3106 | test_mmd=0.0288\n",
      "[CellOT] epoch=850 f_loss=-2040.6622 g_loss=2463.0803 | train mmd=0.2433 | test_mmd=0.0317\n",
      "[CellOT] epoch=900 f_loss=-1016.6774 g_loss=2590.2212 | train mmd=0.2840 | test_mmd=0.0390\n",
      "[CellOT] epoch=950 f_loss=-131.3933 g_loss=2574.1038 | train mmd=0.1660 | test_mmd=0.0507\n",
      "[CellOT] epoch=1000 f_loss=-514.1306 g_loss=2570.3018 | train mmd=0.2662 | test_mmd=0.0155\n",
      "[CellOT] epoch=1050 f_loss=-610.9059 g_loss=2652.7168 | train mmd=0.2896 | test_mmd=0.0257\n",
      "[CellOT] epoch=1100 f_loss=-82.9647 g_loss=2602.2852 | train mmd=0.2441 | test_mmd=0.0185\n",
      "[CellOT] epoch=1150 f_loss=-143.4293 g_loss=2289.3972 | train mmd=0.1877 | test_mmd=0.0269\n",
      "[CellOT] epoch=1200 f_loss=427.2751 g_loss=2095.3906 | train mmd=0.1774 | test_mmd=0.0596\n",
      "[CellOT] epoch=1250 f_loss=401.1932 g_loss=1702.0820 | train mmd=0.2427 | test_mmd=0.1245\n",
      "[CellOT] epoch=1300 f_loss=426.2703 g_loss=1517.8701 | train mmd=0.2418 | test_mmd=0.0223\n",
      "[CellOT] epoch=1350 f_loss=609.7295 g_loss=2334.8115 | train mmd=0.2376 | test_mmd=0.0207\n",
      "[CellOT] epoch=1400 f_loss=304.0577 g_loss=1237.3442 | train mmd=0.2055 | test_mmd=0.0292\n",
      "[CellOT] epoch=1450 f_loss=706.6134 g_loss=1250.2390 | train mmd=0.2188 | test_mmd=0.0377\n",
      "[CellOT] epoch=1500 f_loss=577.2168 g_loss=856.9242 | train mmd=0.2790 | test_mmd=0.0683\n",
      "[CellOT] epoch=1550 f_loss=69.0823 g_loss=1238.4625 | train mmd=0.6120 | test_mmd=0.3739\n",
      "[CellOT] epoch=1600 f_loss=44.6026 g_loss=2577.3835 | train mmd=0.2475 | test_mmd=0.1983\n",
      "[CellOT] epoch=1650 f_loss=182.4719 g_loss=2553.3921 | train mmd=0.1905 | test_mmd=0.0724\n",
      "[CellOT] epoch=1700 f_loss=146.0084 g_loss=2693.1025 | train mmd=0.1662 | test_mmd=0.0361\n",
      "[CellOT] epoch=1750 f_loss=-45.5099 g_loss=2666.2188 | train mmd=0.4250 | test_mmd=0.1846\n",
      "[CellOT] epoch=1800 f_loss=150.7887 g_loss=2887.2329 | train mmd=0.2415 | test_mmd=0.0350\n",
      "[CellOT] epoch=1850 f_loss=127.8384 g_loss=2850.1060 | train mmd=0.1993 | test_mmd=0.0316\n",
      "[CellOT] epoch=1900 f_loss=-168.9885 g_loss=3035.3296 | train mmd=0.3393 | test_mmd=0.1148\n",
      "[CellOT] epoch=1950 f_loss=81.2449 g_loss=3098.8247 | train mmd=0.3652 | test_mmd=0.2705\n",
      "[CellOT] epoch=2000 f_loss=82.0691 g_loss=3184.7349 | train mmd=0.4066 | test_mmd=0.1371\n",
      "[CellOT] Final CellOT MMD: 0.1747\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 3 metrics: {'mmd2_gamma_median': 0.1370630947819671, 'mmd2_gamma_0.5': 0.35480769263505485, 'mmd2_gamma_1.0': 0.40409427143840626, 'wasserstein_distance': 1.3734818434062626, 'R2_feature_means': 0.8471621223261324}\n",
      "**************** Run: 4 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=0 f_loss=-4354760.0000 g_loss=5243214.5000 | train mmd=0.1318 | test_mmd=0.6987\n",
      "[CellOT] epoch=50 f_loss=560.2656 g_loss=478.3854 | train mmd=0.7246 | test_mmd=0.2211\n",
      "[CellOT] epoch=100 f_loss=-31.6559 g_loss=370.7814 | train mmd=0.7601 | test_mmd=0.2035\n",
      "[CellOT] epoch=150 f_loss=-357.4481 g_loss=529.3525 | train mmd=0.5625 | test_mmd=0.1533\n",
      "[CellOT] epoch=200 f_loss=-553.2610 g_loss=630.3130 | train mmd=0.5444 | test_mmd=0.1173\n",
      "[CellOT] epoch=250 f_loss=-631.7449 g_loss=1121.7159 | train mmd=0.5020 | test_mmd=0.1025\n",
      "[CellOT] epoch=300 f_loss=-711.2075 g_loss=1337.5256 | train mmd=0.4786 | test_mmd=0.0902\n",
      "[CellOT] epoch=350 f_loss=-720.3048 g_loss=1172.0515 | train mmd=0.4528 | test_mmd=0.0943\n",
      "[CellOT] epoch=400 f_loss=-698.2388 g_loss=1534.8392 | train mmd=0.4355 | test_mmd=0.0755\n",
      "[CellOT] epoch=450 f_loss=-829.3126 g_loss=1004.0705 | train mmd=0.3715 | test_mmd=0.0680\n",
      "[CellOT] epoch=500 f_loss=-1369.0059 g_loss=1056.3840 | train mmd=0.2221 | test_mmd=0.0781\n",
      "[CellOT] epoch=550 f_loss=-825.2868 g_loss=1454.7869 | train mmd=0.3842 | test_mmd=0.0510\n",
      "[CellOT] epoch=600 f_loss=-777.1581 g_loss=1176.0228 | train mmd=0.4344 | test_mmd=0.0592\n",
      "[CellOT] epoch=650 f_loss=-6264.6797 g_loss=1073.9717 | train mmd=0.2628 | test_mmd=0.0925\n",
      "[CellOT] epoch=700 f_loss=-769.1660 g_loss=4380.0303 | train mmd=0.3667 | test_mmd=0.0620\n",
      "[CellOT] epoch=750 f_loss=-1312.9747 g_loss=6612.5332 | train mmd=0.2197 | test_mmd=0.0821\n",
      "[CellOT] epoch=800 f_loss=-2241.1934 g_loss=1338.7650 | train mmd=0.3068 | test_mmd=0.0321\n",
      "[CellOT] epoch=850 f_loss=-409.0229 g_loss=1494.6702 | train mmd=0.3582 | test_mmd=0.0345\n",
      "[CellOT] epoch=900 f_loss=-601.4794 g_loss=2085.6685 | train mmd=0.3475 | test_mmd=0.0327\n",
      "[CellOT] epoch=950 f_loss=-543.8893 g_loss=2299.2566 | train mmd=0.2946 | test_mmd=0.0211\n",
      "[CellOT] epoch=1000 f_loss=-456.6196 g_loss=2038.6158 | train mmd=0.3065 | test_mmd=0.0203\n",
      "[CellOT] epoch=1050 f_loss=-254.5528 g_loss=2237.4656 | train mmd=0.2676 | test_mmd=0.0130\n",
      "[CellOT] epoch=1100 f_loss=-1059.0153 g_loss=1956.7329 | train mmd=0.1908 | test_mmd=0.0330\n",
      "[CellOT] epoch=1150 f_loss=-465.1064 g_loss=1578.7330 | train mmd=0.2509 | test_mmd=0.0259\n",
      "[CellOT] epoch=1200 f_loss=-3.9874 g_loss=1775.1221 | train mmd=0.2214 | test_mmd=0.0179\n",
      "[CellOT] epoch=1250 f_loss=178.4828 g_loss=1341.6188 | train mmd=0.2479 | test_mmd=0.0351\n",
      "[CellOT] epoch=1300 f_loss=335.7393 g_loss=1240.5206 | train mmd=0.1832 | test_mmd=0.0501\n",
      "[CellOT] epoch=1350 f_loss=267.2713 g_loss=1414.1554 | train mmd=0.2380 | test_mmd=0.0257\n",
      "[CellOT] epoch=1400 f_loss=281.0035 g_loss=1309.0710 | train mmd=0.2345 | test_mmd=0.0308\n",
      "[CellOT] epoch=1450 f_loss=543.8733 g_loss=1282.7990 | train mmd=0.2364 | test_mmd=0.0410\n",
      "[CellOT] epoch=1500 f_loss=553.0117 g_loss=1371.8895 | train mmd=0.3429 | test_mmd=0.0486\n",
      "[CellOT] epoch=1550 f_loss=295.2369 g_loss=1268.3081 | train mmd=0.2948 | test_mmd=0.0630\n",
      "[CellOT] epoch=1600 f_loss=516.9127 g_loss=1277.6042 | train mmd=0.2452 | test_mmd=0.0194\n",
      "[CellOT] epoch=1650 f_loss=528.3057 g_loss=2228.9600 | train mmd=0.2260 | test_mmd=0.0603\n",
      "[CellOT] epoch=1700 f_loss=291.2680 g_loss=2967.5718 | train mmd=0.2368 | test_mmd=0.0975\n",
      "[CellOT] epoch=1750 f_loss=226.1560 g_loss=3198.4705 | train mmd=0.2094 | test_mmd=0.0466\n",
      "[CellOT] epoch=1800 f_loss=191.8992 g_loss=3220.0640 | train mmd=0.1677 | test_mmd=0.1490\n",
      "[CellOT] epoch=1850 f_loss=30.8819 g_loss=2981.5483 | train mmd=0.1972 | test_mmd=0.1013\n",
      "[CellOT] epoch=1900 f_loss=-31.8740 g_loss=3294.3711 | train mmd=0.3711 | test_mmd=0.1571\n",
      "[CellOT] epoch=1950 f_loss=94.0710 g_loss=3419.2329 | train mmd=0.2014 | test_mmd=0.1067\n",
      "[CellOT] epoch=2000 f_loss=127.2728 g_loss=3222.2881 | train mmd=0.1961 | test_mmd=0.0711\n",
      "[CellOT] Final CellOT MMD: 0.0988\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n",
      "[CellOT] epoch=0 f_loss=-4200819.0000 g_loss=4553803.0000 | train mmd=0.1318 | test_mmd=0.6987\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 4 metrics: {'mmd2_gamma_median': 0.07111727514704236, 'mmd2_gamma_0.5': 0.11805163617344827, 'mmd2_gamma_1.0': 0.16953794831635233, 'wasserstein_distance': 1.5154504728072804, 'R2_feature_means': 0.8958845757655961}\n",
      "**************** Run: 5 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=50 f_loss=-537.9965 g_loss=343.1578 | train mmd=0.5205 | test_mmd=0.6117\n",
      "[CellOT] epoch=100 f_loss=-107.1925 g_loss=326.7331 | train mmd=0.7490 | test_mmd=0.2046\n",
      "[CellOT] epoch=150 f_loss=-306.4725 g_loss=2087.0010 | train mmd=0.4614 | test_mmd=0.1263\n",
      "[CellOT] epoch=200 f_loss=-397.7710 g_loss=599.2192 | train mmd=0.5620 | test_mmd=0.1270\n",
      "[CellOT] epoch=250 f_loss=-356.0660 g_loss=994.5317 | train mmd=0.4159 | test_mmd=0.0829\n",
      "[CellOT] epoch=300 f_loss=-1855.8271 g_loss=648.3389 | train mmd=0.4571 | test_mmd=0.4061\n",
      "[CellOT] epoch=350 f_loss=-924.0509 g_loss=3115.3345 | train mmd=0.4046 | test_mmd=0.0821\n",
      "[CellOT] epoch=400 f_loss=-681.9590 g_loss=795.0090 | train mmd=0.5177 | test_mmd=0.0839\n",
      "[CellOT] epoch=450 f_loss=-781.8024 g_loss=1625.3828 | train mmd=0.4894 | test_mmd=0.0822\n",
      "[CellOT] epoch=500 f_loss=-814.6223 g_loss=997.2875 | train mmd=0.3690 | test_mmd=0.0895\n",
      "[CellOT] epoch=550 f_loss=-639.0245 g_loss=1042.2451 | train mmd=0.4027 | test_mmd=0.0512\n",
      "[CellOT] epoch=600 f_loss=-717.7189 g_loss=1099.2202 | train mmd=0.4209 | test_mmd=0.0523\n",
      "[CellOT] epoch=650 f_loss=-644.1848 g_loss=1073.7334 | train mmd=0.3896 | test_mmd=0.0438\n",
      "[CellOT] epoch=700 f_loss=-268.0464 g_loss=922.4964 | train mmd=0.1939 | test_mmd=0.0178\n",
      "[CellOT] epoch=750 f_loss=-567.6317 g_loss=1154.6597 | train mmd=0.3184 | test_mmd=0.0305\n",
      "[CellOT] epoch=800 f_loss=-604.5000 g_loss=1164.0242 | train mmd=0.2883 | test_mmd=0.0240\n",
      "[CellOT] epoch=850 f_loss=-660.8529 g_loss=892.3837 | train mmd=0.1944 | test_mmd=0.2435\n",
      "[CellOT] epoch=900 f_loss=-151.4650 g_loss=1680.3907 | train mmd=0.2892 | test_mmd=0.0278\n",
      "[CellOT] epoch=950 f_loss=-357.6363 g_loss=1383.5586 | train mmd=0.2343 | test_mmd=0.0133\n",
      "[CellOT] epoch=1000 f_loss=-287.6099 g_loss=1379.6080 | train mmd=0.2494 | test_mmd=0.0300\n",
      "[CellOT] epoch=1050 f_loss=-242.4830 g_loss=1348.4449 | train mmd=0.1944 | test_mmd=0.0515\n",
      "[CellOT] epoch=1100 f_loss=-1302.5210 g_loss=1113.8378 | train mmd=0.1930 | test_mmd=0.1162\n",
      "[CellOT] epoch=1150 f_loss=-100.2666 g_loss=1605.6223 | train mmd=0.3021 | test_mmd=0.0307\n",
      "[CellOT] epoch=1200 f_loss=88.2743 g_loss=1287.5000 | train mmd=0.1953 | test_mmd=0.0248\n",
      "[CellOT] epoch=1250 f_loss=171.2321 g_loss=1118.6746 | train mmd=0.2304 | test_mmd=0.0170\n",
      "[CellOT] epoch=1300 f_loss=533.9749 g_loss=992.7357 | train mmd=0.1923 | test_mmd=0.0762\n",
      "[CellOT] epoch=1350 f_loss=430.1687 g_loss=844.9741 | train mmd=0.2181 | test_mmd=0.0386\n",
      "[CellOT] epoch=1400 f_loss=495.3187 g_loss=1157.2672 | train mmd=0.2379 | test_mmd=0.0853\n",
      "[CellOT] epoch=1450 f_loss=216.8394 g_loss=1340.9509 | train mmd=0.1928 | test_mmd=0.0779\n",
      "[CellOT] epoch=1500 f_loss=94.6078 g_loss=1618.4309 | train mmd=0.1884 | test_mmd=0.0604\n",
      "[CellOT] epoch=1550 f_loss=36.1810 g_loss=1756.8855 | train mmd=0.3236 | test_mmd=0.2605\n",
      "[CellOT] epoch=1600 f_loss=64.1099 g_loss=1843.0332 | train mmd=0.1903 | test_mmd=0.0362\n",
      "[CellOT] epoch=1650 f_loss=55.9913 g_loss=1951.1245 | train mmd=0.3128 | test_mmd=0.2922\n",
      "[CellOT] epoch=1700 f_loss=44.7353 g_loss=1994.0547 | train mmd=0.1768 | test_mmd=0.0668\n",
      "[CellOT] epoch=1750 f_loss=36.6236 g_loss=2098.5347 | train mmd=0.4930 | test_mmd=0.4041\n",
      "[CellOT] epoch=1800 f_loss=44.0954 g_loss=2100.6870 | train mmd=0.3930 | test_mmd=0.2864\n",
      "[CellOT] epoch=1850 f_loss=26.3656 g_loss=2153.3503 | train mmd=0.2544 | test_mmd=0.0299\n",
      "[CellOT] epoch=1900 f_loss=32.6539 g_loss=2179.4128 | train mmd=0.2380 | test_mmd=0.0326\n",
      "[CellOT] epoch=1950 f_loss=24.4599 g_loss=2194.2766 | train mmd=0.2123 | test_mmd=0.0358\n",
      "[CellOT] epoch=2000 f_loss=29.2689 g_loss=2213.4958 | train mmd=0.2780 | test_mmd=0.0225\n",
      "[CellOT] Final CellOT MMD: 0.1340\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n",
      "[CellOT] epoch=0 f_loss=-4009068.0000 g_loss=4175682.7500 | train mmd=0.1318 | test_mmd=0.6987\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 5 metrics: {'mmd2_gamma_median': 0.02250478343934348, 'mmd2_gamma_0.5': 0.14238667880902334, 'mmd2_gamma_1.0': 0.2607156154224899, 'wasserstein_distance': 1.2774247516260513, 'R2_feature_means': 0.9356671873706367}\n",
      "**************** Run: 6 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=50 f_loss=447.1933 g_loss=1814.9343 | train mmd=0.6411 | test_mmd=0.2407\n",
      "[CellOT] epoch=100 f_loss=-47.9660 g_loss=478.9678 | train mmd=0.7639 | test_mmd=0.2128\n",
      "[CellOT] epoch=150 f_loss=-353.4204 g_loss=442.7979 | train mmd=0.6865 | test_mmd=0.1719\n",
      "[CellOT] epoch=200 f_loss=-499.7928 g_loss=441.5974 | train mmd=0.6163 | test_mmd=0.1340\n",
      "[CellOT] epoch=250 f_loss=-518.4906 g_loss=673.7175 | train mmd=0.5952 | test_mmd=0.1379\n",
      "[CellOT] epoch=300 f_loss=-526.5754 g_loss=2037.1870 | train mmd=0.4504 | test_mmd=0.0714\n",
      "[CellOT] epoch=350 f_loss=-822.7052 g_loss=845.6585 | train mmd=0.5393 | test_mmd=0.1055\n",
      "[CellOT] epoch=400 f_loss=-658.7954 g_loss=933.5019 | train mmd=0.5547 | test_mmd=0.1033\n",
      "[CellOT] epoch=450 f_loss=-731.3666 g_loss=1004.9679 | train mmd=0.5048 | test_mmd=0.0849\n",
      "[CellOT] epoch=500 f_loss=-888.3563 g_loss=1165.4889 | train mmd=0.5245 | test_mmd=0.0859\n",
      "[CellOT] epoch=550 f_loss=-778.6096 g_loss=1341.7500 | train mmd=0.4945 | test_mmd=0.0790\n",
      "[CellOT] epoch=600 f_loss=-822.0183 g_loss=1821.7478 | train mmd=0.4688 | test_mmd=0.0647\n",
      "[CellOT] epoch=650 f_loss=-1013.5879 g_loss=1228.8118 | train mmd=0.3259 | test_mmd=0.0381\n",
      "[CellOT] epoch=700 f_loss=-698.7974 g_loss=1535.0051 | train mmd=0.3951 | test_mmd=0.0485\n",
      "[CellOT] epoch=750 f_loss=-1029.6256 g_loss=1446.0510 | train mmd=0.3420 | test_mmd=0.0505\n",
      "[CellOT] epoch=800 f_loss=-929.6704 g_loss=6745.7100 | train mmd=0.2953 | test_mmd=0.1170\n",
      "[CellOT] epoch=850 f_loss=-1385.4329 g_loss=1475.5670 | train mmd=0.3126 | test_mmd=0.0506\n",
      "[CellOT] epoch=900 f_loss=-4780.5830 g_loss=886.7439 | train mmd=0.3340 | test_mmd=0.4324\n",
      "[CellOT] epoch=950 f_loss=-377.1341 g_loss=1469.6714 | train mmd=0.3280 | test_mmd=0.0266\n",
      "[CellOT] epoch=1000 f_loss=500.9033 g_loss=1318.4368 | train mmd=0.2719 | test_mmd=0.3132\n",
      "[CellOT] epoch=1050 f_loss=-94.1775 g_loss=1810.2542 | train mmd=0.1602 | test_mmd=0.0475\n",
      "[CellOT] epoch=1100 f_loss=-91.7639 g_loss=1270.1614 | train mmd=0.2452 | test_mmd=0.0348\n",
      "[CellOT] epoch=1150 f_loss=107.4267 g_loss=1077.9744 | train mmd=0.2201 | test_mmd=0.0296\n",
      "[CellOT] epoch=1200 f_loss=609.6718 g_loss=416.2313 | train mmd=0.2002 | test_mmd=0.0967\n",
      "[CellOT] epoch=1250 f_loss=178.9676 g_loss=750.7120 | train mmd=0.2143 | test_mmd=0.0616\n",
      "[CellOT] epoch=1300 f_loss=306.8965 g_loss=695.0730 | train mmd=0.1969 | test_mmd=0.1021\n",
      "[CellOT] epoch=1350 f_loss=323.6334 g_loss=868.5793 | train mmd=0.2530 | test_mmd=0.0384\n",
      "[CellOT] epoch=1400 f_loss=146.9212 g_loss=1038.5500 | train mmd=0.1908 | test_mmd=0.0617\n",
      "[CellOT] epoch=1450 f_loss=291.4962 g_loss=1430.4458 | train mmd=0.2504 | test_mmd=0.1557\n",
      "[CellOT] epoch=1500 f_loss=174.6942 g_loss=1848.5090 | train mmd=0.3065 | test_mmd=0.1300\n",
      "[CellOT] epoch=1550 f_loss=97.8155 g_loss=1975.9501 | train mmd=0.2796 | test_mmd=0.2259\n",
      "[CellOT] epoch=1600 f_loss=74.1432 g_loss=2024.7441 | train mmd=0.4140 | test_mmd=0.2377\n",
      "[CellOT] epoch=1650 f_loss=58.2842 g_loss=2098.7329 | train mmd=0.1870 | test_mmd=0.0643\n",
      "[CellOT] epoch=1700 f_loss=54.2828 g_loss=2094.6738 | train mmd=0.1901 | test_mmd=0.0704\n",
      "[CellOT] epoch=1750 f_loss=39.9390 g_loss=2233.9873 | train mmd=0.2118 | test_mmd=0.0632\n",
      "[CellOT] epoch=1800 f_loss=67.9095 g_loss=2136.2852 | train mmd=0.2513 | test_mmd=0.0344\n",
      "[CellOT] epoch=1850 f_loss=52.0851 g_loss=2134.1177 | train mmd=0.2909 | test_mmd=0.0617\n",
      "[CellOT] epoch=1900 f_loss=36.0756 g_loss=2209.6538 | train mmd=0.1956 | test_mmd=0.0253\n",
      "[CellOT] epoch=1950 f_loss=34.5328 g_loss=2254.4358 | train mmd=0.2437 | test_mmd=0.0270\n",
      "[CellOT] epoch=2000 f_loss=21.0426 g_loss=2324.1794 | train mmd=0.2385 | test_mmd=0.0492\n",
      "[CellOT] Final CellOT MMD: 0.1103\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n",
      "[CellOT] epoch=0 f_loss=-6248561.0000 g_loss=6763722.0000 | train mmd=0.1318 | test_mmd=0.6987\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 6 metrics: {'mmd2_gamma_median': 0.0492490679937172, 'mmd2_gamma_0.5': 0.12959212917543939, 'mmd2_gamma_1.0': 0.21225094951275186, 'wasserstein_distance': 1.283433538128941, 'R2_feature_means': 0.8825633759540992}\n",
      "**************** Run: 7 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=50 f_loss=-623.7385 g_loss=211.6183 | train mmd=0.4392 | test_mmd=0.4084\n",
      "[CellOT] epoch=100 f_loss=-499.0898 g_loss=310.3237 | train mmd=0.5067 | test_mmd=0.2404\n",
      "[CellOT] epoch=150 f_loss=-411.0854 g_loss=531.8906 | train mmd=0.7083 | test_mmd=0.1890\n",
      "[CellOT] epoch=200 f_loss=-555.7245 g_loss=712.5181 | train mmd=0.6152 | test_mmd=0.1366\n",
      "[CellOT] epoch=250 f_loss=-631.2363 g_loss=3041.9626 | train mmd=0.4687 | test_mmd=0.1072\n",
      "[CellOT] epoch=300 f_loss=-846.8442 g_loss=914.0194 | train mmd=0.4728 | test_mmd=0.0954\n",
      "[CellOT] epoch=350 f_loss=-872.9218 g_loss=1054.2227 | train mmd=0.5769 | test_mmd=0.1125\n",
      "[CellOT] epoch=400 f_loss=-873.0077 g_loss=1250.6722 | train mmd=0.5666 | test_mmd=0.1055\n",
      "[CellOT] epoch=450 f_loss=-997.1779 g_loss=1214.0287 | train mmd=0.5173 | test_mmd=0.0926\n",
      "[CellOT] epoch=500 f_loss=-974.1638 g_loss=1251.1792 | train mmd=0.5074 | test_mmd=0.0808\n",
      "[CellOT] epoch=550 f_loss=-1520.2791 g_loss=1377.1538 | train mmd=0.4492 | test_mmd=0.0817\n",
      "[CellOT] epoch=600 f_loss=-706.9872 g_loss=1934.7220 | train mmd=0.2768 | test_mmd=0.0742\n",
      "[CellOT] epoch=650 f_loss=-863.6371 g_loss=1749.7778 | train mmd=0.4218 | test_mmd=0.0592\n",
      "[CellOT] epoch=700 f_loss=-976.1327 g_loss=1730.5632 | train mmd=0.3740 | test_mmd=0.0399\n",
      "[CellOT] epoch=750 f_loss=-1106.4164 g_loss=1734.1709 | train mmd=0.3766 | test_mmd=0.0403\n",
      "[CellOT] epoch=800 f_loss=-937.1191 g_loss=2038.2723 | train mmd=0.3778 | test_mmd=0.0319\n",
      "[CellOT] epoch=850 f_loss=-5027.1680 g_loss=1539.0422 | train mmd=0.1710 | test_mmd=0.2607\n",
      "[CellOT] epoch=900 f_loss=-336.2407 g_loss=2000.9060 | train mmd=0.2962 | test_mmd=0.0342\n",
      "[CellOT] epoch=950 f_loss=-450.1757 g_loss=1795.8340 | train mmd=0.2618 | test_mmd=0.0311\n",
      "[CellOT] epoch=1000 f_loss=-2250.4941 g_loss=2325.8215 | train mmd=0.1710 | test_mmd=0.1130\n",
      "[CellOT] epoch=1050 f_loss=-5395.2417 g_loss=3841.9683 | train mmd=0.2951 | test_mmd=0.2634\n",
      "[CellOT] epoch=1100 f_loss=-2556.6133 g_loss=3401.0874 | train mmd=0.1988 | test_mmd=0.0918\n",
      "[CellOT] epoch=1150 f_loss=-908.7478 g_loss=2500.2031 | train mmd=0.2959 | test_mmd=0.0366\n",
      "[CellOT] epoch=1200 f_loss=-1905.8197 g_loss=3494.1914 | train mmd=0.2222 | test_mmd=0.1041\n",
      "[CellOT] epoch=1250 f_loss=-2895.1997 g_loss=4013.1699 | train mmd=0.2281 | test_mmd=0.1347\n",
      "[CellOT] epoch=1300 f_loss=-1272.4933 g_loss=3525.5527 | train mmd=0.2377 | test_mmd=0.0473\n",
      "[CellOT] epoch=1350 f_loss=-1148.0220 g_loss=3148.7507 | train mmd=0.2523 | test_mmd=0.0301\n",
      "[CellOT] epoch=1400 f_loss=-41.3825 g_loss=3054.4316 | train mmd=0.3019 | test_mmd=0.0365\n",
      "[CellOT] epoch=1450 f_loss=-27.8065 g_loss=3213.4106 | train mmd=0.2926 | test_mmd=0.0493\n",
      "[CellOT] epoch=1500 f_loss=-846.5278 g_loss=3582.7930 | train mmd=0.2655 | test_mmd=0.0518\n",
      "[CellOT] epoch=1550 f_loss=329.3246 g_loss=3725.7539 | train mmd=0.2926 | test_mmd=0.0552\n",
      "[CellOT] epoch=1600 f_loss=-819.8430 g_loss=4762.7744 | train mmd=0.2664 | test_mmd=0.1116\n",
      "[CellOT] epoch=1650 f_loss=388.8235 g_loss=4591.7563 | train mmd=0.2580 | test_mmd=0.0246\n",
      "[CellOT] epoch=1700 f_loss=-209.8672 g_loss=4698.9463 | train mmd=0.2458 | test_mmd=0.0343\n",
      "[CellOT] epoch=1750 f_loss=172.4193 g_loss=5186.1440 | train mmd=0.2626 | test_mmd=0.0708\n",
      "[CellOT] epoch=1800 f_loss=458.2555 g_loss=5658.7520 | train mmd=0.2573 | test_mmd=0.0275\n",
      "[CellOT] epoch=1850 f_loss=196.4827 g_loss=5775.8545 | train mmd=0.2847 | test_mmd=0.0660\n",
      "[CellOT] epoch=1900 f_loss=575.9205 g_loss=5071.2700 | train mmd=0.2850 | test_mmd=0.0726\n",
      "[CellOT] epoch=1950 f_loss=377.7955 g_loss=5301.9766 | train mmd=0.2502 | test_mmd=0.0368\n",
      "[CellOT] epoch=2000 f_loss=139.0201 g_loss=5395.9277 | train mmd=0.4503 | test_mmd=0.1545\n",
      "[CellOT] Final CellOT MMD: 0.2174\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n",
      "[CellOT] epoch=0 f_loss=-4329976.0000 g_loss=5189057.0000 | train mmd=0.1318 | test_mmd=0.6987\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 7 metrics: {'mmd2_gamma_median': 0.15445049516689147, 'mmd2_gamma_0.5': 0.3698191756375424, 'mmd2_gamma_1.0': 0.46003015865353175, 'wasserstein_distance': 1.477418023179496, 'R2_feature_means': 0.7687227798981966}\n",
      "**************** Run: 8 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=50 f_loss=474.6043 g_loss=900.6925 | train mmd=0.6858 | test_mmd=0.2071\n",
      "[CellOT] epoch=100 f_loss=-59.3909 g_loss=372.3025 | train mmd=0.7133 | test_mmd=0.1908\n",
      "[CellOT] epoch=150 f_loss=-461.1306 g_loss=564.9434 | train mmd=0.7198 | test_mmd=0.2015\n",
      "[CellOT] epoch=200 f_loss=-611.3236 g_loss=813.3672 | train mmd=0.6388 | test_mmd=0.1452\n",
      "[CellOT] epoch=250 f_loss=-832.1200 g_loss=799.4951 | train mmd=0.5356 | test_mmd=0.1257\n",
      "[CellOT] epoch=300 f_loss=-726.9006 g_loss=1378.4517 | train mmd=0.5363 | test_mmd=0.0985\n",
      "[CellOT] epoch=350 f_loss=-943.7498 g_loss=1153.0740 | train mmd=0.5643 | test_mmd=0.1044\n",
      "[CellOT] epoch=400 f_loss=-920.7848 g_loss=1824.8516 | train mmd=0.5069 | test_mmd=0.0852\n",
      "[CellOT] epoch=450 f_loss=-917.0747 g_loss=1273.7422 | train mmd=0.4949 | test_mmd=0.0798\n",
      "[CellOT] epoch=500 f_loss=-1155.0222 g_loss=1268.0625 | train mmd=0.4650 | test_mmd=0.0719\n",
      "[CellOT] epoch=550 f_loss=-2561.8147 g_loss=1318.1702 | train mmd=0.3777 | test_mmd=0.0784\n",
      "[CellOT] epoch=600 f_loss=-1661.1113 g_loss=1452.9114 | train mmd=0.4223 | test_mmd=0.0642\n",
      "[CellOT] epoch=650 f_loss=-867.7535 g_loss=1525.3378 | train mmd=0.4529 | test_mmd=0.0530\n",
      "[CellOT] epoch=700 f_loss=-767.7173 g_loss=2767.0000 | train mmd=0.3199 | test_mmd=0.0698\n",
      "[CellOT] epoch=750 f_loss=-2550.9839 g_loss=5891.1616 | train mmd=0.2187 | test_mmd=0.1814\n",
      "[CellOT] epoch=800 f_loss=-961.1629 g_loss=1933.7795 | train mmd=0.3889 | test_mmd=0.0374\n",
      "[CellOT] epoch=850 f_loss=-636.6307 g_loss=1911.2789 | train mmd=0.3352 | test_mmd=0.0421\n",
      "[CellOT] epoch=900 f_loss=-702.1194 g_loss=1947.9832 | train mmd=0.2736 | test_mmd=0.0305\n",
      "[CellOT] epoch=950 f_loss=-413.8508 g_loss=2098.5991 | train mmd=0.2573 | test_mmd=0.0181\n",
      "[CellOT] epoch=1000 f_loss=-73.6239 g_loss=2036.9427 | train mmd=0.2225 | test_mmd=0.0419\n",
      "[CellOT] epoch=1050 f_loss=-612.0801 g_loss=2420.2744 | train mmd=0.3174 | test_mmd=0.0279\n",
      "[CellOT] epoch=1100 f_loss=47.6696 g_loss=1862.8945 | train mmd=0.1615 | test_mmd=0.0312\n",
      "[CellOT] epoch=1150 f_loss=-18.7386 g_loss=2165.6919 | train mmd=0.2752 | test_mmd=0.0195\n",
      "[CellOT] epoch=1200 f_loss=-219.5988 g_loss=2314.8352 | train mmd=0.2426 | test_mmd=0.0236\n",
      "[CellOT] epoch=1250 f_loss=47.8415 g_loss=1704.8591 | train mmd=0.2705 | test_mmd=0.0436\n",
      "[CellOT] epoch=1300 f_loss=168.9017 g_loss=1880.8967 | train mmd=0.2681 | test_mmd=0.0372\n",
      "[CellOT] epoch=1350 f_loss=694.6135 g_loss=1379.8911 | train mmd=0.1980 | test_mmd=0.0830\n",
      "[CellOT] epoch=1400 f_loss=505.4340 g_loss=1405.7261 | train mmd=0.2128 | test_mmd=0.0189\n",
      "[CellOT] epoch=1450 f_loss=835.1284 g_loss=996.7733 | train mmd=0.2124 | test_mmd=0.0655\n",
      "[CellOT] epoch=1500 f_loss=678.8163 g_loss=1071.8054 | train mmd=0.2373 | test_mmd=0.0904\n",
      "[CellOT] epoch=1550 f_loss=668.7963 g_loss=1153.5669 | train mmd=0.2333 | test_mmd=0.0815\n",
      "[CellOT] epoch=1600 f_loss=200.1380 g_loss=1701.5229 | train mmd=0.1940 | test_mmd=0.0353\n",
      "[CellOT] epoch=1650 f_loss=180.9294 g_loss=1751.2709 | train mmd=0.4215 | test_mmd=0.3053\n",
      "[CellOT] epoch=1700 f_loss=112.4665 g_loss=1881.8451 | train mmd=0.2608 | test_mmd=0.1769\n",
      "[CellOT] epoch=1750 f_loss=69.8634 g_loss=1972.3248 | train mmd=0.1745 | test_mmd=0.0314\n",
      "[CellOT] epoch=1800 f_loss=-9.0217 g_loss=2134.9492 | train mmd=0.3395 | test_mmd=0.0962\n",
      "[CellOT] epoch=1850 f_loss=69.9153 g_loss=2095.8105 | train mmd=0.2034 | test_mmd=0.0554\n",
      "[CellOT] epoch=1900 f_loss=51.3049 g_loss=2156.1418 | train mmd=0.2310 | test_mmd=0.0394\n",
      "[CellOT] epoch=1950 f_loss=54.3023 g_loss=2291.6631 | train mmd=0.4840 | test_mmd=0.1843\n",
      "[CellOT] epoch=2000 f_loss=55.0038 g_loss=2249.1846 | train mmd=0.2402 | test_mmd=0.0497\n",
      "[CellOT] Final CellOT MMD: 0.1155\n",
      "VERS torch=1.13.1+cu117 (CellOT), device=cuda\n",
      "[CellOT] epoch=0 f_loss=-3814278.5000 g_loss=4332641.0000 | train mmd=0.1318 | test_mmd=0.6987\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 8 metrics: {'mmd2_gamma_median': 0.04972045938959724, 'mmd2_gamma_0.5': 0.13109321058229373, 'mmd2_gamma_1.0': 0.20714151318104512, 'wasserstein_distance': 1.5463285963016522, 'R2_feature_means': 0.8148773062668517}\n",
      "**************** Run: 9 ****************\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[CellOT] epoch=50 f_loss=312.4943 g_loss=1612.6641 | train mmd=0.6148 | test_mmd=0.2516\n",
      "[CellOT] epoch=100 f_loss=-146.9672 g_loss=310.5554 | train mmd=0.7526 | test_mmd=0.1997\n",
      "[CellOT] epoch=150 f_loss=-414.0605 g_loss=428.3445 | train mmd=0.6539 | test_mmd=0.1692\n",
      "[CellOT] epoch=200 f_loss=-408.3353 g_loss=783.7712 | train mmd=0.3626 | test_mmd=0.0924\n",
      "[CellOT] epoch=250 f_loss=-552.4115 g_loss=797.3804 | train mmd=0.5477 | test_mmd=0.1067\n",
      "[CellOT] epoch=300 f_loss=-763.6899 g_loss=726.6216 | train mmd=0.4968 | test_mmd=0.0922\n",
      "[CellOT] epoch=350 f_loss=-719.5190 g_loss=857.0045 | train mmd=0.5056 | test_mmd=0.0843\n",
      "[CellOT] epoch=400 f_loss=-706.7183 g_loss=1776.5159 | train mmd=0.4702 | test_mmd=0.0828\n",
      "[CellOT] epoch=450 f_loss=-2932.8159 g_loss=992.7184 | train mmd=0.5398 | test_mmd=0.2578\n",
      "[CellOT] epoch=500 f_loss=-1738.7175 g_loss=975.9956 | train mmd=0.3218 | test_mmd=0.0905\n",
      "[CellOT] epoch=550 f_loss=-912.1703 g_loss=4134.1035 | train mmd=0.4061 | test_mmd=0.0613\n",
      "[CellOT] epoch=600 f_loss=-703.0805 g_loss=1169.2903 | train mmd=0.4292 | test_mmd=0.0542\n",
      "[CellOT] epoch=650 f_loss=-673.5555 g_loss=1894.4944 | train mmd=0.3902 | test_mmd=0.0476\n",
      "[CellOT] epoch=700 f_loss=-829.1600 g_loss=1260.7775 | train mmd=0.4250 | test_mmd=0.0502\n",
      "[CellOT] epoch=750 f_loss=-485.0497 g_loss=1316.9070 | train mmd=0.3718 | test_mmd=0.0451\n",
      "[CellOT] epoch=800 f_loss=-636.0373 g_loss=1445.3214 | train mmd=0.3451 | test_mmd=0.0306\n",
      "[CellOT] epoch=850 f_loss=-473.5591 g_loss=1606.6189 | train mmd=0.3028 | test_mmd=0.0355\n",
      "[CellOT] epoch=900 f_loss=-798.2427 g_loss=1923.6301 | train mmd=0.3464 | test_mmd=0.0273\n",
      "[CellOT] epoch=950 f_loss=-2105.3181 g_loss=1936.9521 | train mmd=0.1950 | test_mmd=0.0393\n",
      "[CellOT] epoch=1000 f_loss=-440.7547 g_loss=1840.3035 | train mmd=0.2804 | test_mmd=0.0220\n",
      "[CellOT] epoch=1050 f_loss=-476.8397 g_loss=1620.1923 | train mmd=0.2605 | test_mmd=0.0250\n",
      "[CellOT] epoch=1100 f_loss=-96.8177 g_loss=1716.0891 | train mmd=0.2770 | test_mmd=0.0223\n",
      "[CellOT] epoch=1150 f_loss=-6.5062 g_loss=1548.5986 | train mmd=0.2480 | test_mmd=0.0131\n",
      "[CellOT] epoch=1200 f_loss=173.0877 g_loss=1301.1908 | train mmd=0.2278 | test_mmd=0.0690\n",
      "[CellOT] epoch=1250 f_loss=467.4453 g_loss=1083.6863 | train mmd=0.1707 | test_mmd=0.0445\n",
      "[CellOT] epoch=1300 f_loss=229.3623 g_loss=1134.8876 | train mmd=0.2234 | test_mmd=0.0229\n",
      "[CellOT] epoch=1350 f_loss=302.9356 g_loss=1075.2197 | train mmd=0.2412 | test_mmd=0.0282\n",
      "[CellOT] epoch=1400 f_loss=737.0509 g_loss=945.9741 | train mmd=0.2282 | test_mmd=0.0952\n",
      "[CellOT] epoch=1450 f_loss=614.5059 g_loss=1061.8064 | train mmd=0.2318 | test_mmd=0.0567\n",
      "[CellOT] epoch=1500 f_loss=-235.8474 g_loss=1315.6707 | train mmd=0.2444 | test_mmd=0.0310\n",
      "[CellOT] epoch=1550 f_loss=479.2139 g_loss=1897.3536 | train mmd=0.2506 | test_mmd=0.1407\n",
      "[CellOT] epoch=1600 f_loss=269.2750 g_loss=2439.4272 | train mmd=0.2172 | test_mmd=0.0704\n",
      "[CellOT] epoch=1650 f_loss=292.0265 g_loss=2646.9617 | train mmd=0.2238 | test_mmd=0.0463\n",
      "[CellOT] epoch=1700 f_loss=167.4053 g_loss=2830.3232 | train mmd=0.2787 | test_mmd=0.0576\n",
      "[CellOT] epoch=1750 f_loss=71.3547 g_loss=3297.6084 | train mmd=0.3046 | test_mmd=0.0945\n",
      "[CellOT] epoch=1800 f_loss=84.7475 g_loss=3591.1919 | train mmd=0.3923 | test_mmd=0.1476\n",
      "[CellOT] epoch=1850 f_loss=105.4869 g_loss=3313.1965 | train mmd=0.2180 | test_mmd=0.0272\n",
      "[CellOT] epoch=1900 f_loss=107.3670 g_loss=3477.6753 | train mmd=0.3481 | test_mmd=0.1074\n",
      "[CellOT] epoch=1950 f_loss=139.3652 g_loss=3466.1611 | train mmd=0.2493 | test_mmd=0.0961\n",
      "[CellOT] epoch=2000 f_loss=152.7076 g_loss=3716.7185 | train mmd=0.5245 | test_mmd=0.2756\n",
      "[CellOT] Final CellOT MMD: 0.2517\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/umap/umap_.py:1952: UserWarning: n_jobs value 1 overridden to 1 by setting random_state. Use no seed for parallelism.\n",
      "  warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run 9 metrics: {'mmd2_gamma_median': 0.2756241886614621, 'mmd2_gamma_0.5': 0.5374148793980765, 'mmd2_gamma_1.0': 0.5414677293700438, 'wasserstein_distance': 1.8026562837088465, 'R2_feature_means': 0.6958805982748926}\n",
      "                        mean     std\n",
      "mmd2_gamma_median     0.1122  0.0855\n",
      "mmd2_gamma_0.5        0.2498  0.1509\n",
      "mmd2_gamma_1.0        0.3062  0.1303\n",
      "wasserstein_distance  1.4495  0.1874\n",
      "R2_feature_means      0.8339  0.0883\n"
     ]
    },
    {
     "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/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": "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ObOv/DHNK29j9pU/pnbJ5VN2LJ/Px5e//GW+9a1v8eSTT/Luu+/yhS98gXA4zOc+97m0bf/pn/6JZ599lm3btvHpT3+a6upqrrzySgBuueUWnnrqKfbu3ctbb73Fc889l5wUvvKVrxAIBLjuuut444032LNnD0899RSf+cxnCiLXT37yk/z617/mb3/7W9LNA+D3+/nmN7/JN77xDe6991727NnDW2+9xS9+8QvuvfdeAL70pS+xa9cuvvWtb7Fjxw7uv/9+7rnnnjGP2dzczLPPPktnZyf9/f05t1uxYgXLli3jyiuv5Omnn6alpYVXXnmF//N//k9yJTl37txkMHjz5s188pOfxDTNUcd78cUXaWtrS2Y7fec73+GVV17hq1/9Kps2bWLXrl2sWbMmGdydN28el156KV//zg95e0cr7+zt5iu3/h88Hs+ozBuHtxxXeSPOsnpu//7t3Pkf/8kvf/Ub9nb0s2nLNn71u3uRFCcfXHkZy84+i6uvuzHn+RhaFNPQrEIwI07CzaSHAry+8W1++l93s2XnAbqGDB59ch09PT1pBoING3kxmUACWYJzqYhEIuLUU08Vn/zkJ3Nuc/vttwustXPav6kM7h4ORCIR8bWvfU1UV1cLl8slzjnnHPH6668nv08Edx999FFx8sknC6fTKc4880yxefPm5DZf/epXxezZs4XL5RI1NTXihhtuEL29vcnvd+7cKT760Y+K8vJy4fF4xPz588Utt9wiTNMUQljB3a9//etZx/fuu+8KQMycOTO5fQKmaYo777xTzJs3TzgcDlFTUyMuueQS8cILLyS3efTRR8WcOXOEy+US5513nvj9738/ZnB37dq1Ys6cOUJVVTFz5kwhxOigZgJDQ0Pia1/7mmhoaBAOh0NMnz5dXH/99WL//v1CCCH27t0rLrroIuHxeMT06dPFL3/5y1Hnu2HDBrFo0SLhcrlE6qP/6isvixUXLxclJSXC5/OJRYsWiX/+539Oft/R0SEuu+wy4XK5xIwZM8Tq1auzBooz8etf/zp5zerr68VX/v5LySBsvvMx4lHxj9+8WSw8eb4Itr8ngu3vieuvuVJ85NIVItixXbz50hNixUXniZrqSuFyucSJJ56YFsROxdH8ztgYPwoN7kpCTDwHTJIkHn744aRFmgpN0/jYxz5Ga2srzz//PKWl2ZtNxGIx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",
      "text/plain": [
       "<Figure size 400x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "drug = \"Vem\"\n",
    "X_pre, X_post = prepare_pair_from_mat('SKMEL19', 'DMSO','24h', drug, '72h')\n",
    "jfe_indices = [1, 6, 0, 5, 4, 7, 8, 2, 3, 19]  \n",
    "\n",
    "print(\"X_pre cells:\", X_pre.shape)\n",
    "print(\"X_post cells:\", X_post.shape)\n",
    "\n",
    "X_tr_pre, X_te_pre, Y_tr_post, Y_te_post = split_train_test(X_pre, X_post, 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",
    "# 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",
    "\n",
    "\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 = run_cellot_pair(X_tr_pre[:, jfe_indices], Y_tr_post[:, jfe_indices], X_te_pre[:, jfe_indices], Y_te_post[:, jfe_indices], n_epochs=2000)\n",
    "    metrics = summarize_metrics(out[\"y_pred\"], Y_te_post[:, jfe_indices], median_gamma)\n",
    "    print(f\"Run {run} metrics: {metrics}\")\n",
    "    all_metrics.append(metrics)\n",
    "\n",
    "# Results summary\n",
    "df = pd.DataFrame(all_metrics)\n",
    "print(df.describe().T[['mean', 'std']].round(4))\n",
    "\n",
    "\n",
    "from umap import UMAP\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "source = Y_tr_post[:, jfe_indices]\n",
    "target = Y_te_post[:, jfe_indices]\n",
    "predicted = out.get('y_pred') \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([source, target]))\n",
    "source_umap = umap_model.transform(source)\n",
    "target_umap = umap_model.transform(target)\n",
    "y_pred_umap = umap_model.transform(predicted)\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=\"#C88131\")\n",
    "ax.scatter(y_pred_umap[:, 0], y_pred_umap[:, 1], s=10, alpha=0.7, label='predicted cells', color=\"#1F4D8D\")\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/cellot_on_4i_drug_{drug}.png\", dpi=300)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3cc05c34",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6856ea1b",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2061ee67",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "711c8633",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b3847dd9",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "258f2758",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d698f206",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.25"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
