{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The autoreload extension is already loaded. To reload it, use:\n",
      "  %reload_ext autoreload\n"
     ]
    }
   ],
   "source": [
    "%matplotlib inline\n",
    "import sys, os\n",
    "sys.path.append(\"../\")\n",
    "import numpy as np\n",
    "import scipy as sp\n",
    "import numpy.linalg as nla\n",
    "import matplotlib as mpl\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.pylab as pl\n",
    "from timeit import timeit\n",
    "import ot\n",
    "import ot.plot\n",
    "from ot.datasets import make_1D_gauss as gauss\n",
    "from drot.solver import drot, drot2, PDHG\n",
    "from drot.proximal import *\n",
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Optimal transport"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "def one_dimensional_gaussian_ot(n):\n",
    "    t = np.arange(0, n)/n\n",
    "    Gaussian = lambda t0, sigma: np.exp(-(t-t0)**2 / (2*sigma**2))\n",
    "    normalize = lambda p: p / np.sum(p)\n",
    "    sigma = 0.1;\n",
    "    p = Gaussian(.75, sigma)\n",
    "    q = Gaussian(.15, sigma)\n",
    "    vmin = .02;\n",
    "    p = normalize(p + np.max(p) * vmin)\n",
    "    q = normalize(q + np.max(q) * vmin)\n",
    "    [Y,X] = np.meshgrid(t,t)\n",
    "    C = np.array((X-Y)**2, order='F')\n",
    "    # C = np.array(np.abs(X-Y), order='F')\n",
    "    return n, C, p, q\n",
    "\n",
    "def two_dimensional_gaussian_ot(m, n):\n",
    "    mu_s = np.array([0, 0.5])\n",
    "    cov_s = np.array([[1, 0.5], [0.5, 1]])\n",
    "    mu_t = np.array([4, 10])\n",
    "    cov_t = np.array([[1, -.8], [-.8, 1]])\n",
    "    xs = ot.datasets.make_2D_samples_gauss(m, mu_s, cov_s)\n",
    "    xt = ot.datasets.make_2D_samples_gauss(n, mu_t, cov_t)\n",
    "    p, q = np.ones((m,)) / m, np.ones((n,)) / n  \n",
    "    C = np.array(ot.dist(xs, xt), order='F')\n",
    "    C /= C.max()\n",
    "\n",
    "    return m, n, C, p, q"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 40,
   "metadata": {
    "scrolled": true,
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Solve time:  0.1322770118713379\n",
      "Solve time:  0.06894207000732422\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x7f847b54df10>"
      ]
     },
     "execution_count": 40,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 720x576 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "max_iters = 500\n",
    "step = 0.1\n",
    "\n",
    "m, n, C, p, q = two_dimensional_gaussian_ot(100, 100)\n",
    "x0 = np.ones((m, n), order='F') / n\n",
    "eta, tau = (1/(m+n))**0.5, (1/(m+n))**0.5\n",
    "\n",
    "assert C.flags['F_CONTIGUOUS']   \n",
    "assert x0.flags['F_CONTIGUOUS']   \n",
    "\n",
    "proxf = lambda x, stz: trace_nonnegative_prox(x, C, stz)\n",
    "proxg = lambda x, stz: generalized_doubly_stochastic_matrices_projection(x, p, q)\n",
    "# proxg_ = lambda x, y, stz: primal_dual_trace_nonnegative_prox(x, C, y, stz)\n",
    "# proxh = lambda x, xnew, y, stz: primal_dual_linear_prox(x, xnew, y, p, q, stz)\n",
    "\n",
    "drout  = drot(x0, C, p, q, max_iters=max_iters, step=step, compute_r_primal=True, \n",
    "              compute_r_dual=True, adapt_stepsize=False, fixed_restart=False, \n",
    "              milestones=[100, 500, 1000], eps_abs=1e-10, eps_rel=1e-10, verbose=False)\n",
    "\n",
    "dr2out  = drot2(x0, proxf, proxg, np.hstack((p,q)), max_iters, step=step, relaxation=1.0, compute_r_primal=True, \n",
    "              compute_r_dual=True, adapt_stepsize=False, fixed_restart=True, milestones=[100, 500, 1000], \n",
    "              eps_abs=1e-10, eps_rel=1e-10)\n",
    "# cpout  = PDHG(x0, proxg_, proxh, max_iters, eta=eta, tau=tau, relaxation=1.0)\n",
    "\n",
    "xopt = drout[\"sol\"]\n",
    "x2opt = dr2out[\"sol\"]\n",
    "# xopt_cp = cpout[\"sol\"]\n",
    "\n",
    "plt.figure(1, figsize=(10,8))\n",
    "plt.plot(range(drout[\"num_iters\"]), [r for r in drout['primal']], label='DR-primal')\n",
    "# plt.plot(range(drout[\"num_iters\"]), [r for r in drout['dual']], label='DR-dual')\n",
    "plt.ylabel(\"primal residual\") \n",
    "plt.yscale('log')\n",
    "plt.legend()\n",
    "\n",
    "plt.figure(1, figsize=(10,8))\n",
    "plt.plot(range(dr2out[\"num_iters\"]), [r for r in dr2out['primal']], label='DR2-primal')\n",
    "plt.plot(range(dr2out[\"num_iters\"]), [r for r in dr2out['dual']], label='DR2-dual')\n",
    "plt.ylabel(\"primal residual\") \n",
    "plt.yscale('log')\n",
    "plt.legend()\n",
    "\n",
    "# # plt.figure(2, figsize=(10,8))\n",
    "# plt.plot(range(cpout[\"num_iters\"]), [r for r in cpout['dual']], label='PDHG-dual')\n",
    "# # plt.ylabel(\"primal residual\") \n",
    "# plt.yscale('log')\n",
    "# plt.legend()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "C_ = C.copy()\n",
    "G0 = ot.emd(p, q, C_)\n",
    "Gs = ot.sinkhorn(p, q, C_, 1e-3, verbose=False)\n",
    "nla.norm(xopt - G0, ord='fro'),\\\n",
    "nla.norm(Gs - G0, ord='fro')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "np.sum(G0 >= 1e-14) "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "np.trace(xopt.T.dot(C)), np.trace(G0.T.dot(C_)), np.trace(Gs.T.dot(C_))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "pl.figure(1)\n",
    "pl.imshow(xopt, interpolation='nearest')\n",
    "pl.title('OT matrix DR')\n",
    "\n",
    "pl.figure(2)\n",
    "pl.imshow(G0, interpolation='nearest')\n",
    "pl.title('OT matrix G0')\n",
    "\n",
    "pl.figure(4)\n",
    "pl.imshow(xopt_cp, interpolation='nearest')\n",
    "pl.title('OT matrix PDHG')\n",
    "\n",
    "pl.figure(3)\n",
    "pl.imshow(Gs, interpolation='nearest')\n",
    "pl.title('OT matrix Sinkhorn')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "pl.figure(3, figsize=(6, 4))\n",
    "ot.plot.plot1D_mat(p, q, xopt.T, 'OT matrix DR')\n",
    "# pl.title(\"---\")\n",
    "\n",
    "pl.figure(6, figsize=(6, 4))\n",
    "ot.plot.plot1D_mat(p, q, G0, 'OT matrix LP')\n",
    "\n",
    "pl.figure(9, figsize=(6, 4))\n",
    "ot.plot.plot1D_mat(p, q, Gs, 'OT matrix Sinkhorn')\n",
    "pl.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def Sinkhorn(C, reg, a, b, delta=1e-9, lam=1e-6):\n",
    "\n",
    "    K = np.exp(-C / reg)\n",
    "    u = np.ones(np.shape(a)[0])\n",
    "    v = np.ones(np.shape(b)[0])\n",
    "\n",
    "    u_trans = np.dot(K, v) + lam  # add regularization to avoid divide 0\n",
    "    v_trans = np.dot(K.T, u) + lam  # add regularization to avoid divide 0\n",
    "\n",
    "    err_1 = np.sum(np.abs(u * u_trans - a))\n",
    "    err_2 = np.sum(np.abs(v * v_trans - b))\n",
    "\n",
    "    while True:\n",
    "        if err_1 + err_2 > delta:\n",
    "            u = a / u_trans\n",
    "            v_trans = np.dot(K.T, u) + lam\n",
    "\n",
    "            v = b / v_trans\n",
    "            u_trans = np.dot(K, v) + lam\n",
    "\n",
    "            err_1 = np.sum(np.abs(u * u_trans - a))\n",
    "            err_2 = np.sum(np.abs(v * v_trans - b))\n",
    "        else:\n",
    "            return u, v\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# proxf_ = lambda x: nonneg_projection(x -  step * C_)\n",
    "# proxg_ = lambda x: generalized_doubly_stochastic_matrices_projection_(x, p, q)"
   ]
  }
 ],
 "metadata": {
  "interpreter": {
   "hash": "f4884be1e91315d779e4ffae1c5ca2afdbecf8830e8ecc49f4fe74e58837c66e"
  },
  "kernelspec": {
   "display_name": "Python [conda env:common]",
   "language": "python",
   "name": "conda-env-common-py"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.8.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
