{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "5cea4e6a",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(44808, 50) (44808,) float32\n",
      "Training Statistics\n",
      "size:  (44808, 50)\n",
      "Mean: 3.4530467e-09\n",
      "Max:  262.05157\n",
      "Min:  -122.139206\n",
      "STD:  3.7548966\n"
     ]
    }
   ],
   "source": [
    "import sys\n",
    "sys.path.append(\"../code\")\n",
    "\n",
    "\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "from sklearn.metrics.pairwise import euclidean_distances\n",
    "from scipy import optimize\n",
    "\n",
    "\n",
    "\n",
    "from sklearn.neighbors import KNeighborsClassifier\n",
    "neigh = KNeighborsClassifier(n_neighbors=1)\n",
    "\n",
    "from sklearn.datasets import make_moons\n",
    "\n",
    "from scipy.io import savemat, loadmat\n",
    "\n",
    "d = loadmat('data/macosko.mat')\n",
    "\n",
    "X_data = d['X_data']\n",
    "y_data = d['y_data'].reshape(-1)\n",
    "\n",
    "X_train = X_data\n",
    "y_train = y_data\n",
    "\n",
    "\n",
    "n = X_train.shape[0]\n",
    "\n",
    "classes = ['Amacrine cells',\n",
    "           'Astrocytes',\n",
    "           'Bipolar cells',\n",
    "           'Cones',\n",
    "           'Fibroblasts',\n",
    "           'Horizontal cells',\n",
    "           'Microglia',\n",
    "           'Muller glia',\n",
    "           'Pericytes',\n",
    "           'Retinal ganglion cells',\n",
    "           'Rods',\n",
    "           'Vascular endothelium']\n",
    "\n",
    "print(X_train.shape, y_train.shape, X_train.dtype)\n",
    "\n",
    "#Torch Setups\n",
    "from sklearn.decomposition import PCA\n",
    "\n",
    "#import torch\n",
    "#device = torch.device(\"cuda:0\" if torch.cuda.is_available() else \"cpu\")\n",
    "#X_torch = torch.as_tensor(X_train, dtype=torch.float32, device=device)\n",
    "\n",
    "n_components = 2\n",
    "\n",
    "pca = PCA(n_components = n_components)\n",
    "x_init = pca.fit_transform(X_train)\n",
    "x_init = x_init - np.mean(x_init, axis=0)\n",
    "\n",
    "def print_stats(X):\n",
    "    print('size: ', X.shape)\n",
    "    print('Mean:', np.mean(X))\n",
    "    print('Max: ', np.max(X))\n",
    "    print('Min: ', np.min(X))\n",
    "    print('STD: ', np.std(X))\n",
    "    \n",
    "    return\n",
    "\n",
    "print('Training Statistics')\n",
    "print_stats(X_train)\n",
    "#print('Test Statistics')\n",
    "#print_stats(X_test)\n",
    "\n",
    "\n",
    "epochs = 200\n",
    "n_neighbors= 15\n",
    "n_components = 2\n",
    "MIN_DIST = 0.1\n",
    "\n",
    "    \n",
    "%matplotlib inline\n",
    "\n",
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "cc02235e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.collections.PathCollection at 0x7fae54cf3748>"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.figure()\n",
    "plt.scatter(x_init[:,0], x_init[:,1], c=y_train, s=0.01, cmap='Spectral')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "d846eb38",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "\n",
    "import numba\n",
    "from numba import prange\n",
    "\n",
    "import random\n",
    "\n",
    "import scipy.sparse\n",
    "\n",
    "import gc\n",
    "\n",
    "\n",
    "@numba.jit(nopython=True, parallel=True)\n",
    "def euclidean_distances_numba(X, squared = True):\n",
    "    n = X.shape[0]\n",
    "    xcorr = np.zeros((n,n),dtype=X.dtype)\n",
    "    for i in prange(n):\n",
    "        for j in range(i,n):\n",
    "            dist = np.sum( np.square(X[i,:] - X[j,:]) )\n",
    "            if not squared:\n",
    "                dist = np.sqrt(dist)\n",
    "            xcorr[i,j] = dist\n",
    "            xcorr[j,i] = dist\n",
    "    \n",
    "    return xcorr\n",
    "\n",
    "#@numba.jit(nopython=True)\n",
    "def get_weight_function(dists, rho, sigma):\n",
    "    d = dists - rho\n",
    "    #print(d)\n",
    "    d[d<0] = 0\n",
    "    weight = np.exp(- d / sigma )\n",
    "    return weight\n",
    "\n",
    "#@numba.jit(nopython=True)\n",
    "def search_sigma(dists, rho, k, tol = 10**-5, n_iteration=200):\n",
    "    sigma_min = 0\n",
    "    sigma_max = 1000\n",
    "    \n",
    "    cur_sigma = 100\n",
    "    \n",
    "    logk = np.log2(k)\n",
    "    #print(logk)\n",
    "    \n",
    "    for i in range(n_iteration):\n",
    "        \n",
    "        cur_sigma = (sigma_min+sigma_max)/2\n",
    "        probs = get_weight_function(dists,rho,cur_sigma)\n",
    "        weight = np.sum(probs)\n",
    "        #print(weight)\n",
    "        \n",
    "        if np.abs(logk - weight) < tol:\n",
    "            break\n",
    "        \n",
    "        if weight < logk:\n",
    "            sigma_min = cur_sigma\n",
    "        else:\n",
    "            sigma_max = cur_sigma\n",
    "        \n",
    "    return cur_sigma, probs\n",
    "\n",
    "@numba.jit(nopython=True, parallel=True)\n",
    "def symmetrization_step(prob):\n",
    "    n = prob.shape[0]\n",
    "    P = np.zeros((n,n),dtype=np.float32)\n",
    "\n",
    "    for i in prange(n):\n",
    "        #if i%1000 == 0:\n",
    "        #    print('Completed ', i, ' of ', n)\n",
    "        for j in prange(i,n):\n",
    "            p = prob[i,j] + prob[j,i] - prob[i,j] * prob[j,i] #t-conorm\n",
    "            P[i,j] = p\n",
    "            P[j,i] = p\n",
    "            \n",
    "    return P\n",
    "\n",
    "def get_prob_matrix(X, n_neighbors=15):\n",
    "    n = X.shape[0]\n",
    "    dist = euclidean_distances_numba(X, squared = False)\n",
    "    sort_idx = np.argsort(dist,axis=1)\n",
    "    #sort_idx = sort_idx.astype(np.int32)\n",
    "    sort_idx = sort_idx[:,1:n_neighbors+1]\n",
    "    \n",
    "    rho = [ dist[i, sort_idx[i,0] ] for i in range(n)]\n",
    "    rho = np.array(rho)\n",
    "    \n",
    "    \n",
    "\n",
    "    sigmas = []\n",
    "\n",
    "    directed_graph = []\n",
    "\n",
    "\n",
    "    #'''\n",
    "    for i in range(n):\n",
    "        if (i+1)%1000 == 0:\n",
    "            print('Processed ', i+1, ' of ', n, ' samples.')\n",
    "        sigma, weights = search_sigma(dists = dist[i,sort_idx[i,:]],rho = rho[i],k = n_neighbors)\n",
    "\n",
    "        probs = np.zeros(n)\n",
    "        probs[sort_idx[i,:]] = weights\n",
    "        #print(sum(weights), np.log2(n_neighbors))\n",
    "        #print(sort_idx[i,:])\n",
    "        #print(probs[1770:1780])\n",
    "\n",
    "        directed_graph.append(probs)\n",
    "\n",
    "    directed_graph = np.array(directed_graph).astype(np.float32)\n",
    "    prob = directed_graph\n",
    "    \n",
    "    P = symmetrization_step(prob)\n",
    "    #P = prob\n",
    "    \n",
    "    graph = scipy.sparse.coo_matrix(P)\n",
    "    \n",
    "    return graph\n",
    "\n",
    "def make_epochs_per_sample(weights, n_epochs):\n",
    "    \"\"\"Given a set of weights and number of epochs generate the number of\n",
    "    epochs per sample for each weight.\n",
    "    Parameters\n",
    "    ----------\n",
    "    weights: array of shape (n_1_simplices)\n",
    "        The weights ofhow much we wish to sample each 1-simplex.\n",
    "    n_epochs: int\n",
    "        The total number of epochs we want to train for.\n",
    "    Returns\n",
    "    -------\n",
    "    An array of number of epochs per sample, one for each 1-simplex.\n",
    "    Copied from UMAP repo: https://github.com/lmcinnes/umap/\n",
    "    \"\"\"\n",
    "    result = -1.0 * np.ones(weights.shape[0], dtype=np.float64)\n",
    "    n_samples = n_epochs * (weights / weights.max())\n",
    "    result[n_samples > 0] = float(n_epochs) / n_samples[n_samples > 0]\n",
    "    return result\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "7e5e3eec",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Processed  1000  of  44808  samples.\n",
      "Processed  2000  of  44808  samples.\n",
      "Processed  3000  of  44808  samples.\n",
      "Processed  4000  of  44808  samples.\n",
      "Processed  5000  of  44808  samples.\n",
      "Processed  6000  of  44808  samples.\n",
      "Processed  7000  of  44808  samples.\n",
      "Processed  8000  of  44808  samples.\n",
      "Processed  9000  of  44808  samples.\n",
      "Processed  10000  of  44808  samples.\n",
      "Processed  11000  of  44808  samples.\n",
      "Processed  12000  of  44808  samples.\n",
      "Processed  13000  of  44808  samples.\n",
      "Processed  14000  of  44808  samples.\n",
      "Processed  15000  of  44808  samples.\n",
      "Processed  16000  of  44808  samples.\n",
      "Processed  17000  of  44808  samples.\n",
      "Processed  18000  of  44808  samples.\n",
      "Processed  19000  of  44808  samples.\n",
      "Processed  20000  of  44808  samples.\n",
      "Processed  21000  of  44808  samples.\n",
      "Processed  22000  of  44808  samples.\n",
      "Processed  23000  of  44808  samples.\n",
      "Processed  24000  of  44808  samples.\n",
      "Processed  25000  of  44808  samples.\n",
      "Processed  26000  of  44808  samples.\n",
      "Processed  27000  of  44808  samples.\n",
      "Processed  28000  of  44808  samples.\n",
      "Processed  29000  of  44808  samples.\n",
      "Processed  30000  of  44808  samples.\n",
      "Processed  31000  of  44808  samples.\n",
      "Processed  32000  of  44808  samples.\n",
      "Processed  33000  of  44808  samples.\n",
      "Processed  34000  of  44808  samples.\n",
      "Processed  35000  of  44808  samples.\n",
      "Processed  36000  of  44808  samples.\n",
      "Processed  37000  of  44808  samples.\n",
      "Processed  38000  of  44808  samples.\n",
      "Processed  39000  of  44808  samples.\n",
      "Processed  40000  of  44808  samples.\n",
      "Processed  41000  of  44808  samples.\n",
      "Processed  42000  of  44808  samples.\n",
      "Processed  43000  of  44808  samples.\n",
      "Processed  44000  of  44808  samples.\n",
      "1143510\n",
      "prune value:  0.005\n",
      "1141560\n"
     ]
    }
   ],
   "source": [
    "graph = get_prob_matrix(X_train,n_neighbors=n_neighbors)\n",
    "print(len(graph.data))\n",
    "print('prune value: ', graph.data.max() / float(epochs))\n",
    "graph.data[graph.data < (graph.data.max() / float(epochs))] = 0.0\n",
    "graph.eliminate_zeros()\n",
    "print(len(graph.data))\n",
    "epochs_per_sample_og = make_epochs_per_sample(graph.data, epochs)\n",
    "gc.collect()\n",
    "\n",
    "\n",
    "with open('macosko_epoch_per_sample_og.npy', 'wb') as f:\n",
    "    np.save(f, epochs_per_sample_og)\n",
    "\n",
    "from scipy import sparse\n",
    "sparse.save_npz('macosko_graph.npz', graph)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "dae7b3b5",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(1141560,)\n"
     ]
    }
   ],
   "source": [
    "print(epochs_per_sample_og.data.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "c621b84e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(44808, 2)\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<matplotlib.collections.PathCollection at 0x7faf032aaa58>"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from sklearn.decomposition import PCA\n",
    "\n",
    "n_components = 2\n",
    "\n",
    "pca = PCA(n_components = n_components)\n",
    "init = pca.fit_transform(X_train)\n",
    "emb = init.astype(np.float32).copy()\n",
    "\n",
    "neg_sample_rate = 5\n",
    "repulsion_strength=1.0\n",
    "\n",
    "print(emb.shape)\n",
    "\n",
    "plt.figure()\n",
    "plt.scatter(emb[:,0], emb[:,1], c=y_train, s=0.01, cmap='Spectral')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "42908467",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Hyperparameters a = 1.576943460405378 and b = 0.8950608781227859\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x7faf0311b438>]"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "\n",
    "\n",
    "x = np.linspace(0, 3, 300)\n",
    "\n",
    "y = (x>MIN_DIST) * np.exp(-x+MIN_DIST)\n",
    "y[x<=MIN_DIST] = 1.0\n",
    "\n",
    "function = lambda x, a, b: 1 / (1 + a*x**(2*b))\n",
    "\n",
    "p , _ = optimize.curve_fit(function, x, y) \n",
    "\n",
    "a = p[0]\n",
    "b = p[1] \n",
    "print(\"Hyperparameters a = \" + str(a) + \" and b = \" + str(b))\n",
    "\n",
    "x_p = np.arange(0,3,0.01)\n",
    "y_p = np.exp(- (x_p-MIN_DIST) * ( (x_p - MIN_DIST) >=0 ) )\n",
    "y_p2 = 1 / (1 + a*x_p**(2*b))\n",
    "\n",
    "plt.figure()\n",
    "plt.plot(x_p,y_p, label='Target')\n",
    "plt.plot(x_p,y_p2, label='Fitted')\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "31ed258c",
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "epoch  0 of  200\n",
      "epoch  20 of  200\n",
      "epoch  40 of  200\n",
      "epoch  60 of  200\n",
      "epoch  80 of  200\n",
      "epoch  100 of  200\n",
      "epoch  120 of  200\n",
      "epoch  140 of  200\n",
      "epoch  160 of  200\n",
      "epoch  180 of  200\n"
     ]
    }
   ],
   "source": [
    "attr_coeff = []\n",
    "rep_coeff = []\n",
    "\n",
    "idx_to_map = 100\n",
    "\n",
    "\n",
    "@numba.jit(nopython=True)\n",
    "def clip(x,val=4.0):\n",
    "\n",
    "    if x>val:\n",
    "        return val\n",
    "    elif x<-val:\n",
    "        return -val\n",
    "    else:\n",
    "        return x\n",
    "    \n",
    "@numba.jit(nopython=True)\n",
    "def update_attraction(x, y, a, b, dim, lr, P, idx):\n",
    "    dist = np.sum((x - y)**2)\n",
    "\n",
    "    if dist>0.0:\n",
    "        grad_coeff = 2.0*2*a*b*dist**(b-1.0) / (1 + a * dist**b)\n",
    "        \n",
    "\n",
    "    else:\n",
    "        grad_coeff = 0.0\n",
    "    \n",
    "\n",
    "    for d in range(dim):\n",
    "        mv = clip(grad_coeff * P * (x[0,d]-y[0,d]))  # * P[idx,idy]\n",
    "        mv = mv * lr\n",
    "\n",
    "        x[0,d] -= mv\n",
    "        y[0,d] += mv\n",
    "        \n",
    "    return\n",
    "\n",
    "@numba.jit(nopython=True)\n",
    "def update_repulsion(x, y, a, b, dim, lr, P, idx):\n",
    "    dist = np.sum((x - y)**2)\n",
    "\n",
    "    if dist>0.0:\n",
    "        grad_coeff = 2 * repulsion_strength * b / ( (0.001+dist) * (1.0 + a * dist**b) )\n",
    "    else:\n",
    "        grad_coeff = 0\n",
    "        \n",
    "\n",
    "    for d in range(dim):\n",
    "        \n",
    "        grad = clip(grad_coeff  * (x[0,d]-y[0,d]) * (1-P))\n",
    "        mv = grad * lr\n",
    "\n",
    "        x[0,d] += mv\n",
    "\n",
    "    return\n",
    "\n",
    "\n",
    "@numba.jit(nopython=True)\n",
    "def one_step_in_a_set(emb, idx, rows, columns, a, b, dim,\n",
    "                   n_points,\n",
    "                   epochs_per_sample,\n",
    "                   epoch_of_next_sample,\n",
    "                   epochs_per_negative_sample,\n",
    "                   epoch_of_next_negative_sample,\n",
    "                   lr, epoch):\n",
    "    \n",
    "    if epoch_of_next_sample[idx] <= epoch:\n",
    "        x_idx = rows[idx]\n",
    "        y_idx = columns[idx]\n",
    "        \n",
    "        \n",
    "        x = emb[x_idx:x_idx+1,:]\n",
    "        y = emb[y_idx:y_idx+1, :]\n",
    "            \n",
    "        update_attraction(x, y, a, b, dim, lr, 1, idx)\n",
    "        \n",
    "        epoch_of_next_sample[idx] += epochs_per_sample[idx]\n",
    "        \n",
    "        n_neg_samples = int(\n",
    "                (epoch - epoch_of_next_negative_sample[idx]) / epochs_per_negative_sample[idx]\n",
    "            )\n",
    "        \n",
    "        for i in range(n_neg_samples):\n",
    "            y_idx = np.random.choice(n_points)\n",
    "            \n",
    "            if x_idx == y_idx:\n",
    "                continue\n",
    "            \n",
    "            y = emb[y_idx:y_idx+1, :]\n",
    "                \n",
    "            update_repulsion(x, y, a, b, dim, lr, 0, idx)\n",
    "            \n",
    "        epoch_of_next_negative_sample[idx] += (\n",
    "                n_neg_samples * epochs_per_negative_sample[idx]\n",
    "            )\n",
    "            \n",
    "    return \n",
    "\n",
    "@numba.jit(nopython=True,parallel=True)\n",
    "def one_epoch_2sets_2(emb,\n",
    "                     rows, columns,\n",
    "                     n_points,\n",
    "                     n_edges,\n",
    "                     a, b, dim,\n",
    "                     lr, epoch,\n",
    "                     epochs_per_sample,\n",
    "                     epoch_of_next_sample,\n",
    "                     epochs_per_negative_sample,\n",
    "                     epoch_of_next_negative_sample,\n",
    "                     repulsion_strength=1.0):\n",
    "    '''\n",
    "    Set1 = 1 * np.ones(epochs_per_sample_1.shape[0])\n",
    "    Set2 = 2 * np.ones(epochs_per_sample_2.shape[0])\n",
    "    Set = np.random.permutation(np.concatenate((Set1,Set2)))\n",
    "    '''\n",
    "    \n",
    "    for i in prange(n_edges):\n",
    "\n",
    "        one_step_in_a_set(emb=emb, idx=i, \n",
    "                              rows=rows, columns=columns, a=a, b=b, dim=dim,\n",
    "                              n_points=n_points,\n",
    "                              epochs_per_sample=epochs_per_sample,\n",
    "                              epoch_of_next_sample=epoch_of_next_sample,\n",
    "                              epochs_per_negative_sample=epochs_per_negative_sample,\n",
    "                              epoch_of_next_negative_sample=epoch_of_next_negative_sample,\n",
    "                              lr=lr, epoch=epoch)\n",
    "    \n",
    "    return\n",
    "\n",
    "from sklearn.decomposition import PCA\n",
    "\n",
    "n_components = 2\n",
    "\n",
    "pca = PCA(n_components = n_components)\n",
    "init = pca.fit_transform(X_train)\n",
    "emb = init.astype(np.float32).copy()\n",
    "expansion = 10.0 / np.abs(emb).max()\n",
    "emb = (emb * expansion).astype(np.float32)\n",
    "\n",
    "neg_sample_rate = 5\n",
    "repulsion_strength=1.0\n",
    "\n",
    "\n",
    "epochs_per_sample = epochs_per_sample_og.copy()\n",
    "epoch_of_next_sample = epochs_per_sample.copy()\n",
    "epochs_per_negative_sample = epochs_per_sample / neg_sample_rate\n",
    "epoch_of_next_negative_sample = epochs_per_negative_sample.copy()\n",
    "\n",
    "init_lr = 1.0\n",
    "\n",
    "n_edges = epochs_per_sample.shape[0]\n",
    "\n",
    "np.random.seed(500)\n",
    "\n",
    "a = 1.57\n",
    "b = 0.89\n",
    "\n",
    "\n",
    "import timeit\n",
    "\n",
    "\n",
    "\n",
    "for epoch in range(epochs):\n",
    "    \n",
    "    if epoch%20==0:\n",
    "        print('epoch ', epoch, 'of ', epochs)\n",
    "    #print('epoch ', epoch, 'of ', epochs)\n",
    "    start = timeit.default_timer()\n",
    "    \n",
    "    lr = init_lr * (1.0 - float(epoch)/float(epochs))\n",
    "    \n",
    "    one_epoch_2sets_2(emb=emb, \n",
    "                     rows=graph.row, columns=graph.col, \n",
    "                     n_points=X_train.shape[0],\n",
    "                     n_edges = n_edges,\n",
    "                     a=a, b=b, dim=n_components,\n",
    "                     lr=lr, epoch=epoch+1,\n",
    "                     epochs_per_sample=epochs_per_sample,\n",
    "                     epoch_of_next_sample=epoch_of_next_sample,\n",
    "                     epochs_per_negative_sample=epochs_per_negative_sample,\n",
    "                     epoch_of_next_negative_sample=epoch_of_next_negative_sample,\n",
    "                     repulsion_strength=repulsion_strength)\n",
    "    \n",
    "    \n",
    "    #print('Time for epoch ', epoch, ': ', stop - start) \n",
    "    stop = timeit.default_timer()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "39942df5",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "1a1b90ad",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "#%matplotlib notebook\n",
    "\n",
    "plt.figure()\n",
    "#plt.title('nonLinear Transform UMAP')\n",
    "plt.scatter(emb[:,0], emb[:,1], c=y_train, s=0.01, cmap='Spectral')\n",
    "cbar = plt.colorbar(boundaries=np.arange(11)-0.5)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "af3742ea",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "100\n",
      "[26540890 52577207 89039121 44502734 60574653 45269576 97111879 74814285\n",
      " 98980589 93361553 84392149 41599146 94003218 29110931 29531277 31662182\n",
      " 47233640 88572855 17059759 44078392 77237130 56789901 64028781 38732290\n",
      " 74870075 96458877 66118844 48301710 77364540 94627659 34158526 57300059\n",
      " 54582941 13702556 84260041 78761431 97750651 46855129 69829007 23990899\n",
      " 32016285 61235141 43815097 82451877 22386384  7155371 61497185 71542726\n",
      " 42468704 57603092 42819537 14551533 47811076 19212410 72242633 85171772\n",
      " 48139484 23975582 54515692 19213544 32803637 87745077 21466662 48256673\n",
      " 61373942 65948537 76350024 70301638 48420653 42181707 93949579 86776793\n",
      " 51052695 46891247 31026048 66930219  3586062 47285987  8598039 12006788\n",
      " 31634464   941590 32644396 79499836 90484182 17247889 41358122 15635987\n",
      " 98535095 11482198 13126231 40353429 38792227 90692342  9840707 50109818\n",
      " 90946035 91157229  3961630 12920063]\n"
     ]
    }
   ],
   "source": [
    "#with open('random_init_test_orig/macosko_PCA_init.npy', 'wb') as f:\n",
    "#    np.save(f,emb)\n",
    "\n",
    "#no_of_random_seeds = 100\n",
    "#rngs = np.random.randint(low=0,high=99999999,size=no_of_random_seeds)\n",
    "#print(len(np.unique(rngs)))\n",
    "#\n",
    "#with open('random_init_test_orig/macosko_seeds.npy', 'wb') as f:\n",
    "#    np.save(f,rngs)\n",
    "\n",
    "#print(rngs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e975d2fc",
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "from sklearn.metrics import silhouette_score\n",
    "from sklearn.metrics import calinski_harabasz_score\n",
    "from sklearn.metrics import davies_bouldin_score\n",
    "\n",
    "from sklearn.metrics import silhouette_score\n",
    "\n",
    "from sklearn.preprocessing import StandardScaler\n",
    "scaler = StandardScaler()\n",
    "\n",
    "y_ump1_scaled = scaler.fit_transform(emb)\n",
    "\n",
    "Z_ump1 = silhouette_score(emb,y_train)\n",
    "char_ump1 = calinski_harabasz_score(y_ump1_scaled, y_train)\n",
    "dav_ump1 = davies_bouldin_score(y_ump1_scaled, y_train)\n",
    "\n",
    "print('UMAP1: ', Z_ump1, char_ump1, dav_ump1)\n",
    "\n",
    "'''"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a33634c5",
   "metadata": {},
   "outputs": [],
   "source": [
    "#from sklearn.manifold import trustworthiness\n",
    "\n",
    "#print(trustworthiness(X_train, emb))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "dd1c300c",
   "metadata": {},
   "outputs": [],
   "source": [
    "#import os\n",
    "#folder = 'Test_Data/MNIST_UMAP/'\n",
    "#if os.path.isdir(folder):\n",
    "#    print('Folder Already Exists')\n",
    "#else:\n",
    "#    os.mkdir(folder)\n",
    "#\n",
    "#d = {}\n",
    "#d['y'] = emb\n",
    "#savemat(folder+'Epochs_'+str(epoch)+'.mat', d)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7a6fd54c",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.6.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
