{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "GRdoE3U-SybK"
   },
   "source": [
    "## Gaussian mixture model in 2d\n",
    "\n",
    "\n",
    "\n",
    "This notebook compares Sinkhorn, Online Sinkhorn, Compressed Online Sinkhorn (with Fourier compression) for two multivariate GMMs.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 19607,
     "status": "ok",
     "timestamp": 1694663320832,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "Zes9WhQ1rcDp",
    "outputId": "67f66ca0-170d-4452-fb35-544ae1bcfe53"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Mounted at /content/drive\n"
     ]
    }
   ],
   "source": [
    "from google.colab import drive\n",
    "drive.mount(\"/content/drive\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 7472,
     "status": "ok",
     "timestamp": 1694663328297,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "XevkJMUlYF3z",
    "outputId": "7a1e1871-fd6f-4f6b-d479-06210edaaa17"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Collecting celer\n",
      "  Downloading celer-0.7.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl (4.5 MB)\n",
      "\u001b[2K     \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m4.5/4.5 MB\u001b[0m \u001b[31m5.7 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
      "\u001b[?25hRequirement already satisfied: seaborn>=0.7 in /usr/local/lib/python3.10/dist-packages (from celer) (0.12.2)\n",
      "Requirement already satisfied: matplotlib>=2.0.0 in /usr/local/lib/python3.10/dist-packages (from celer) (3.7.1)\n",
      "Collecting libsvmdata>=0.3 (from celer)\n",
      "  Downloading libsvmdata-0.4.1-py3-none-any.whl (7.0 kB)\n",
      "Requirement already satisfied: scikit-learn>=1.0 in /usr/local/lib/python3.10/dist-packages (from celer) (1.2.2)\n",
      "Requirement already satisfied: xarray in /usr/local/lib/python3.10/dist-packages (from celer) (2023.7.0)\n",
      "Collecting download (from celer)\n",
      "  Downloading download-0.3.5-py3-none-any.whl (8.8 kB)\n",
      "Requirement already satisfied: tqdm in /usr/local/lib/python3.10/dist-packages (from celer) (4.66.1)\n",
      "Requirement already satisfied: numpy>=1.12 in /usr/local/lib/python3.10/dist-packages (from libsvmdata>=0.3->celer) (1.23.5)\n",
      "Requirement already satisfied: scipy in /usr/local/lib/python3.10/dist-packages (from libsvmdata>=0.3->celer) (1.11.2)\n",
      "Requirement already satisfied: contourpy>=1.0.1 in /usr/local/lib/python3.10/dist-packages (from matplotlib>=2.0.0->celer) (1.1.0)\n",
      "Requirement already satisfied: cycler>=0.10 in /usr/local/lib/python3.10/dist-packages (from matplotlib>=2.0.0->celer) (0.11.0)\n",
      "Requirement already satisfied: fonttools>=4.22.0 in /usr/local/lib/python3.10/dist-packages (from matplotlib>=2.0.0->celer) (4.42.1)\n",
      "Requirement already satisfied: kiwisolver>=1.0.1 in /usr/local/lib/python3.10/dist-packages (from matplotlib>=2.0.0->celer) (1.4.5)\n",
      "Requirement already satisfied: packaging>=20.0 in /usr/local/lib/python3.10/dist-packages (from matplotlib>=2.0.0->celer) (23.1)\n",
      "Requirement already satisfied: pillow>=6.2.0 in /usr/local/lib/python3.10/dist-packages (from matplotlib>=2.0.0->celer) (9.4.0)\n",
      "Requirement already satisfied: pyparsing>=2.3.1 in /usr/local/lib/python3.10/dist-packages (from matplotlib>=2.0.0->celer) (3.1.1)\n",
      "Requirement already satisfied: python-dateutil>=2.7 in /usr/local/lib/python3.10/dist-packages (from matplotlib>=2.0.0->celer) (2.8.2)\n",
      "Requirement already satisfied: joblib>=1.1.1 in /usr/local/lib/python3.10/dist-packages (from scikit-learn>=1.0->celer) (1.3.2)\n",
      "Requirement already satisfied: threadpoolctl>=2.0.0 in /usr/local/lib/python3.10/dist-packages (from scikit-learn>=1.0->celer) (3.2.0)\n",
      "Requirement already satisfied: pandas>=0.25 in /usr/local/lib/python3.10/dist-packages (from seaborn>=0.7->celer) (1.5.3)\n",
      "Requirement already satisfied: six in /usr/local/lib/python3.10/dist-packages (from download->celer) (1.16.0)\n",
      "Requirement already satisfied: requests in /usr/local/lib/python3.10/dist-packages (from download->celer) (2.31.0)\n",
      "Requirement already satisfied: pytz>=2020.1 in /usr/local/lib/python3.10/dist-packages (from pandas>=0.25->seaborn>=0.7->celer) (2023.3.post1)\n",
      "Requirement already satisfied: charset-normalizer<4,>=2 in /usr/local/lib/python3.10/dist-packages (from requests->download->celer) (3.2.0)\n",
      "Requirement already satisfied: idna<4,>=2.5 in /usr/local/lib/python3.10/dist-packages (from requests->download->celer) (3.4)\n",
      "Requirement already satisfied: urllib3<3,>=1.21.1 in /usr/local/lib/python3.10/dist-packages (from requests->download->celer) (2.0.4)\n",
      "Requirement already satisfied: certifi>=2017.4.17 in /usr/local/lib/python3.10/dist-packages (from requests->download->celer) (2023.7.22)\n",
      "Installing collected packages: download, libsvmdata, celer\n",
      "Successfully installed celer-0.7.3 download-0.3.5 libsvmdata-0.4.1\n"
     ]
    }
   ],
   "source": [
    "!pip install -U celer"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 13453,
     "status": "ok",
     "timestamp": 1694663341739,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "ybc4BzbN62xN",
    "outputId": "104774b1-e692-4ec8-f852-a050b29040d0"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Collecting pykeops\n",
      "  Downloading pykeops-2.1.2.tar.gz (88 kB)\n",
      "\u001b[2K     \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m88.9/88.9 kB\u001b[0m \u001b[31m2.4 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
      "\u001b[?25h  Preparing metadata (setup.py) ... \u001b[?25l\u001b[?25hdone\n",
      "Requirement already satisfied: numpy in /usr/local/lib/python3.10/dist-packages (from pykeops) (1.23.5)\n",
      "Collecting pybind11 (from pykeops)\n",
      "  Downloading pybind11-2.11.1-py3-none-any.whl (227 kB)\n",
      "\u001b[2K     \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m227.7/227.7 kB\u001b[0m \u001b[31m8.5 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
      "\u001b[?25hCollecting keopscore==2.1.2 (from pykeops)\n",
      "  Downloading keopscore-2.1.2.tar.gz (84 kB)\n",
      "\u001b[2K     \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m84.5/84.5 kB\u001b[0m \u001b[31m8.8 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
      "\u001b[?25h  Preparing metadata (setup.py) ... \u001b[?25l\u001b[?25hdone\n",
      "Building wheels for collected packages: pykeops, keopscore\n",
      "  Building wheel for pykeops (setup.py) ... \u001b[?25l\u001b[?25hdone\n",
      "  Created wheel for pykeops: filename=pykeops-2.1.2-py3-none-any.whl size=114071 sha256=6eb26c45e9a1ca0c1f5361afbb462f4eb337c6b84fe36c5e9d5be11898e3dd03\n",
      "  Stored in directory: /root/.cache/pip/wheels/93/91/9e/279e56403818cf05d868c2d90a13bde97572bcd11673d6e8ef\n",
      "  Building wheel for keopscore (setup.py) ... \u001b[?25l\u001b[?25hdone\n",
      "  Created wheel for keopscore: filename=keopscore-2.1.2-py3-none-any.whl size=146448 sha256=ef944cb37d7510dc7699dae2a370e8b6771f2e5350b477023b8f2f8ecb01a963\n",
      "  Stored in directory: /root/.cache/pip/wheels/63/ac/b7/75fb4d24be97d9a930905eddf822cb765ca204b83df7aeaaa9\n",
      "Successfully built pykeops keopscore\n",
      "Installing collected packages: pybind11, keopscore, pykeops\n",
      "Successfully installed keopscore-2.1.2 pybind11-2.11.1 pykeops-2.1.2\n"
     ]
    }
   ],
   "source": [
    "import locale\n",
    "locale.getpreferredencoding = lambda: \"UTF-8\"\n",
    "!pip install pykeops"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 697,
     "status": "ok",
     "timestamp": 1694663342427,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "-Ftp4rX2rio_",
    "outputId": "430ed02e-a4d1-4c22-8285-ff31aa3d929e"
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "['helper.py',\n",
       " 'COS.ipynb',\n",
       " '.git',\n",
       " '__pycache__',\n",
       " 'gq.py',\n",
       " 'recombination.py',\n",
       " 'recombination2.py',\n",
       " 'closed_forms.py',\n",
       " 'example_cts_20d.ipynb',\n",
       " 'algorithms.py',\n",
       " 'compression.py',\n",
       " 'example_gmm_20d.ipynb',\n",
       " 'myImagePDF.pgf',\n",
       " 'myImagePDF.pdf',\n",
       " 'OS_theoretical_cts_1d.pkl',\n",
       " 'New folder',\n",
       " 'example_gmm_2d.pkl',\n",
       " 'example_gmm_5d.pkl',\n",
       " 'OS_theoretical_cts_2d.pkl',\n",
       " 'OS_theortical_cts_2d.ipynb',\n",
       " 'OS_theortical_cts_1d.ipynb',\n",
       " 'OS_theoretical_cts_5d_.pkl',\n",
       " 'OS_theortical_cts_5d_ .ipynb',\n",
       " 'example_cts_1d.pkl',\n",
       " 'OS_theortical_cts_5d.ipynb',\n",
       " 'example_cts_1d_easy.ipynb',\n",
       " 'example_gmm_2d.ipynb',\n",
       " 'example_gmm_5d.ipynb',\n",
       " 'untitled0.py',\n",
       " 'repeat_better_data',\n",
       " 'OS_theoretical_cts_1d_test.pkl',\n",
       " 'OS_theortical_cts_1d_test.ipynb',\n",
       " 'example_gmm_2d_am.ipynb']"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import os\n",
    "path=\"/content/drive/My Drive/Colab Notebooks/compressedOT\"\n",
    "os.chdir(path)\n",
    "os.listdir(path)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {
    "executionInfo": {
     "elapsed": 265,
     "status": "ok",
     "timestamp": 1694663342686,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "tfRHebKAdox5"
   },
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 26745,
     "status": "ok",
     "timestamp": 1694663369424,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "6f4hsauPrjGD",
    "outputId": "0bd794d1-d42b-40fe-aa80-8b935a6c19af"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "update closed_form\n",
      "[KeOps] Compiling cuda jit compiler engine ... OK\n",
      "[pyKeOps] Compiling nvrtc binder for python ... OK\n",
      "Import recombination2\n",
      "2\n",
      "Importing algorithms.py\n"
     ]
    }
   ],
   "source": [
    "import sys\n",
    "import math\n",
    "import timeit\n",
    "import random\n",
    "import pickle\n",
    "\n",
    "import numpy as np\n",
    "\n",
    "import matplotlib.pyplot as pl\n",
    "\n",
    "# exact solution of Gaussian OT\n",
    "from closed_forms import closed_form\n",
    "# for Sinkhorn, Online Sinkhorn, Compressed\n",
    "import algorithms"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 53,
   "metadata": {
    "executionInfo": {
     "elapsed": 268,
     "status": "ok",
     "timestamp": 1694664210588,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "eutcKoFrrjN5"
   },
   "outputs": [],
   "source": [
    "# generate data and parameters\n",
    "size = 3000 # problem dimension (for Sinkhorn)\n",
    "d = 2 # sample dimension\n",
    "\n",
    "##\n",
    "if d == 1:\n",
    "    mean1, cov1 = 1., 2.\n",
    "    mean2, cov2 = 3., 2.5\n",
    "else:\n",
    "    mean1a = 10*np.random.rand(d)\n",
    "    mean2a = 5*np.random.rand(d)\n",
    "    cov1a = 3*algorithms.random_cov_matrix(d)\n",
    "    cov2a = 4*algorithms.random_cov_matrix(d)\n",
    "    mean1b = -mean1a\n",
    "    mean2b = -mean2a\n",
    "    cov1b = 3*algorithms.random_cov_matrix(d)\n",
    "    cov2b = 4*algorithms.random_cov_matrix(d)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 54,
   "metadata": {
    "executionInfo": {
     "elapsed": 8,
     "status": "ok",
     "timestamp": 1694664210810,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "Cam41gGGR206"
   },
   "outputs": [],
   "source": [
    "# test distribution\n",
    "\n",
    "def samp1(n,d):\n",
    "    x = np.random.multivariate_normal(mean1a, cov1a, n)\n",
    "    x2 = np.random.multivariate_normal(mean1b, cov1b, n)\n",
    "    r = np.random.rand(n, 1) > 0.5\n",
    "    x = r*x + (1-r)*x2\n",
    "    return x\n",
    "def samp2(n,d):\n",
    "    x = np.random.multivariate_normal(mean2a, cov2a, n)\n",
    "    x2 = np.random.multivariate_normal(mean2b, cov2b, n)\n",
    "    r = np.random.rand(n, 1) > 0.5\n",
    "    x = r*x + (1-r)*x2\n",
    "    return x\n",
    "\n",
    "# test distribution\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 70,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 227,
     "status": "ok",
     "timestamp": 1694664815379,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "vL9MUt27399P",
    "outputId": "d49e0156-c80a-4d49-8bc6-79efea462f9c"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "epsilon is 0.5\n",
      "a= 1.2 ; b= -0.6\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/usr/local/lib/python3.10/dist-packages/scipy/stats/_qmc.py:804: UserWarning: The balance properties of Sobol' points require n to be a power of 2.\n",
      "  sample = self._random(n, workers=workers)\n"
     ]
    }
   ],
   "source": [
    "size_var=10\n",
    "size_obj=3000\n",
    "params = {'b': -0.6,# b<a-1 with b=0 corresponding to Sinkhorn\n",
    "          'eta_decay_const':35, # eta=(1+(t/eta_decay_const)^b)\n",
    "          'init_batch':1000,\n",
    "          'batch_const':1., # batch_size=init_batch+batch_const*pow(i, 2*a))\n",
    "          'a': 1.2, # n_t=t^{2a} for a>1+b (number of samples upto iteration t)\n",
    "          'epsilon': 0.5, # regularisation parametr\n",
    "          \"dim\": d,\n",
    "          'maxits': 35,#\n",
    "          'no_initial_sinkhorn_its':5,\n",
    "          'compress':True,\n",
    "          'min_compress':1000,\n",
    "          'get_samples1':samp1,\n",
    "          'get_samples2':samp2,\n",
    "          'zeta':0.9, # compression error rate\n",
    "          'compression_skip':1,# number of steps between compression\n",
    "          'compression_const':0.3, #\n",
    "          't_sampling':'qmc_gau',\n",
    "          'test_samples1':samp1(size,d),# Sinkhorn\n",
    "          'test_samples2':samp2(size,d),\n",
    "          'test_samples1_obj':algorithms.get_qmc_samples(mean1a,cov1a,size_obj ),\n",
    "          'test_samples2_obj':algorithms.get_qmc_samples(mean2a,cov2a,size_obj ),\n",
    "          'test_samples1_var':algorithms.get_qmc_samples(mean1a,cov1a,size_var ),\n",
    "          'test_samples2_var':algorithms.get_qmc_samples(mean2a,cov2a,size_var )\n",
    "          }\n",
    "#\n",
    "#\n",
    "if (params['a']-params['b']<1):\n",
    "    print('Params violate a-b>1 condition')\n",
    "    exit()\n",
    "print('epsilon is', params['epsilon'])\n",
    "print('a=', params['a'], '; b=', params['b'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 71,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 12,
     "status": "ok",
     "timestamp": 1694664815634,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "KW2va0Ju4Qll",
    "outputId": "3d0459a5-51bc-4b81-cf4c-f5f77a380c71"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1.63835065 1.21356411] [-1.63835065 -1.21356411] [3.46092909 4.7524875 ] [-3.46092909 -4.7524875 ]\n",
      "[[1.33786463 0.57411337]\n",
      " [0.57411337 3.18933726]] [[2.0819209  2.10245171]\n",
      " [2.10245171 3.54939093]] [[4.48298752 2.1527307 ]\n",
      " [2.1527307  1.24903086]] [[0.15803812 0.66634029]\n",
      " [0.66634029 2.81554148]]\n"
     ]
    }
   ],
   "source": [
    "print(mean1a,mean1b,mean2a,mean2b)\n",
    "print(cov1a,cov1b,  cov2a,cov2b)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 72,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 6007,
     "status": "ok",
     "timestamp": 1694664821633,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "3Nyyy4BQWHhc",
    "outputId": "7473f74f-33e8-4f05-87e5-fe1b313f59ca"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "running sinkhorn...\n",
      "objective is 6.503325953185333 ; error is 73.79749535192033\n",
      "objective is 8.126124604456777 ; error is 9.73887189144115\n",
      "objective is 9.38286734246044 ; error is 7.878755515131181\n",
      "objective is 10.418833363343113 ; error is 6.697355638941767\n",
      "objective is 11.293567812283015 ; error is 6.054963899041065\n",
      "objective is 12.039295554725562 ; error is 5.684407777875339\n",
      "objective is 12.677665226802851 ; error is 5.412194867533226\n",
      "objective is 13.225107400189458 ; error is 5.125533627834965\n",
      "objective is 13.694863059665169 ; error is 4.83422309538984\n",
      "objective is 14.09794674953412 ; error is 4.547306773333208\n",
      "objective is 14.443691432613882 ; error is 4.26994367142777\n",
      "objective is 14.74009686290804 ; error is 4.013189410343561\n",
      "objective is 14.994064494330793 ; error is 3.7679849605782056\n",
      "objective is 15.211559779014893 ; error is 3.5320830807621126\n",
      "objective is 15.397736669775725 ; error is 3.3041060895883234\n",
      "objective is 15.557044045342128 ; error is 3.082860265354263\n",
      "objective is 15.693318759190337 ; error is 2.8685044817421606\n",
      "objective is 15.809865754547527 ; error is 2.667348178052258\n",
      "objective is 15.90952733141342 ; error is 2.4876728459538917\n",
      "objective is 15.994744194646028 ; error is 2.3231481985803484\n",
      "objective is 16.067609735962918 ; error is 2.1691132238090773\n",
      "objective is 16.129918045506496 ; error is 2.025201607895795\n",
      "objective is 16.183205860288083 ; error is 1.8906366574927738\n",
      "objective is 16.228788818391855 ; error is 1.7649024880999669\n",
      "objective is 16.26779265033318 ; error is 1.647545638132904\n",
      "objective is 16.301180030201017 ; error is 1.5379767989450244\n",
      "objective is 16.329773729723488 ; error is 1.4356581061598526\n",
      "objective is 16.354276593760122 ; error is 1.3400751231747776\n",
      "objective is 16.37528875879979 ; error is 1.2507312024454116\n",
      "objective is 16.39332247277587 ; error is 1.1671588023138995\n",
      "The value of the objective function for Sinkhorn is 16.39332247277587  elapsed time 5.9286828570000125\n"
     ]
    }
   ],
   "source": [
    "print('running sinkhorn...')\n",
    "x = params[\"test_samples1\"]\n",
    "y = params[\"test_samples2\"]\n",
    "start = timeit.default_timer()\n",
    "(ft_s, gt_s, obj_s) = algorithms.sinkhorn(x,y, 30, params['epsilon'])\n",
    "run_time = timeit.default_timer()-start\n",
    "data_s={\"f\":ft_s,\n",
    "  \"g\":gt_s,\n",
    "  \"obj\":obj_s,\n",
    "  \"rt\":run_time}\n",
    "print(\"The value of the objective function for Sinkhorn is\", obj_s[-1],\" elapsed time\",run_time)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 73,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 207017,
     "status": "ok",
     "timestamp": 1694665028633,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "mqMUVJaSrjVl",
    "outputId": "dca2961f-92ed-4c1b-bbf5-8ee42191cd54"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "running compressed online sinkhorn with Fourier..\n",
      "Online Sinkhorn with Compression= True\n",
      "Compression method:  Fourier\n",
      "Fourier solver= nnls\n",
      "Running 5 Sinkhorn iterations to start-up.\n",
      "step is 0 obj is 15.411244327241008 err is 3.9788731381169673 ; eta= 1.0 OS step time 0.04\n",
      "Compressing with Fourier from  1005  to  15  points.\n",
      "Compress from  2010  to  15  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.1564388333260851\n",
      "Compressed to  24  points\n",
      "Compress from  2010  to  15  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 2.932441189914064e-16\n",
      "Compressed to  28  points\n",
      "Compression time 0.03\n",
      "OS step time 0.24 (as fraction of last compression 8.02)\n",
      "step is 1 obj is 16.242476772102982 err is 10.222698535030819 ; eta= 0.9832395200760898\n",
      "Compressing with Fourier from  4632  to  23  points.\n",
      "Compress from  38326  to  23  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 4.2444689469238127e-16\n",
      "Compressed to  44  points\n",
      "Compress from  9264  to  23  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 5.862439292079899e-16\n",
      "Compressed to  44  points\n",
      "Compression time 0.13\n",
      "OS step time 0.15 (as fraction of last compression 1.11)\n",
      "step is 2 obj is 16.836683433521426 err is 10.161625522943254 ; eta= 0.967207804357326\n",
      "Compressing with Fourier from  4714  to  32  points.\n",
      "Compress from  39992  to  32  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.04510104729726541\n",
      "Compressed to  60  points\n",
      "Compress from  9428  to  32  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 4.690437735334492e-16\n",
      "Compressed to  64  points\n",
      "Compression time 0.14\n",
      "OS step time 0.15 (as fraction of last compression 1.12)\n",
      "step is 3 obj is 17.099962505779395 err is 2.5417901325178995 ; eta= 0.9518547218427332\n",
      "Compressing with Fourier from  4787  to  43  points.\n",
      "Compress from  40828  to  43  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.01606945007828395\n",
      "Compressed to  78  points\n",
      "Compress from  9574  to  43  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 5.906589683652791e-16\n",
      "Compressed to  84  points\n",
      "Compression time 0.15\n",
      "OS step time 0.15 (as fraction of last compression 1.02)\n",
      "step is 4 obj is 17.25928007774696 err is 1.905798055534356 ; eta= 0.9371347933586296\n",
      "Compressing with Fourier from  4835  to  56  points.\n",
      "Compress from  41738  to  56  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.028299874014348067\n",
      "Compressed to  92  points\n",
      "Compress from  9670  to  56  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.0738488032669498e-15\n",
      "Compressed to  112  points\n",
      "Compression time 0.27\n",
      "OS step time 0.17 (as fraction of last compression 0.63)\n",
      "step is 5 obj is 17.396321116038422 err is 2.69231563538414 ; eta= 0.9230066538429028\n",
      "Compressing with Fourier from  5179  to  70  points.\n",
      "Compress from  43018  to  70  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.011728416639694414\n",
      "Compressed to  128  points\n",
      "Compress from  10358  to  70  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 5.729619087587796e-16\n",
      "Compressed to  140  points\n",
      "Compression time 0.32\n",
      "OS step time 0.18 (as fraction of last compression 0.56)\n",
      "step is 6 obj is 17.460881272089875 err is 3.5829291721891345 ; eta= 0.9094325884438981\n",
      "Compressing with Fourier from  5432  to  85  points.\n",
      "Compress from  44672  to  85  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.011229957514851768\n",
      "Compressed to  140  points\n",
      "Compress from  10864  to  85  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0004978326661761441\n",
      "Compressed to  166  points\n",
      "Compression time 0.38\n",
      "OS step time 0.19 (as fraction of last compression 0.49)\n",
      "step is 7 obj is 17.53994605745545 err is 1.900721549013717 ; eta= 0.8963781307771418\n",
      "Compressing with Fourier from  5468  to  101  points.\n",
      "Compress from  46466  to  101  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.014236124990466963\n",
      "Compressed to  148  points\n",
      "Compress from  10936  to  101  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0025378864153481266\n",
      "Compressed to  188  points\n",
      "Compression time 0.47\n",
      "OS step time 0.20 (as fraction of last compression 0.42)\n",
      "step is 8 obj is 17.592718365769343 err is 1.6107306600913036 ; eta= 0.8838117137533721\n",
      "Compressing with Fourier from  5873  to  118  points.\n",
      "Compress from  49014  to  118  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.007212469184697314\n",
      "Compressed to  178  points\n",
      "Compress from  11746  to  118  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0012251287662044302\n",
      "Compressed to  222  points\n",
      "Compression time 0.52\n",
      "OS step time 0.20 (as fraction of last compression 0.39)\n",
      "step is 9 obj is 17.64683359081593 err is 1.1337084725463273 ; eta= 0.8717043650550409\n",
      "Compressing with Fourier from  6054  to  136  points.\n",
      "Compress from  51362  to  136  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.004549564103501676\n",
      "Compressed to  190  points\n",
      "Compress from  12108  to  136  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.002468615799087679\n",
      "Compressed to  236  points\n",
      "Compression time 0.67\n",
      "OS step time 0.22 (as fraction of last compression 0.32)\n",
      "step is 10 obj is 17.67778378903649 err is 1.0018317775201004 ; eta= 0.8600294406827096\n",
      "Compressing with Fourier from  6479  to  156  points.\n",
      "Compress from  54232  to  156  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0029380294467855465\n",
      "Compressed to  206  points\n",
      "Compress from  12958  to  156  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0030938722931140735\n",
      "Compressed to  262  points\n",
      "Compression time 0.72\n",
      "OS step time 0.21 (as fraction of last compression 0.30)\n",
      "step is 11 obj is 17.69141883572719 err is 0.8438162469402695 ; eta= 0.8487623910845714\n",
      "Compressing with Fourier from  6883  to  176  points.\n",
      "Compress from  57194  to  176  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0045986413802664595\n",
      "Compressed to  224  points\n",
      "Compress from  13766  to  176  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0026877611138330627\n",
      "Compressed to  266  points\n",
      "Compression time 0.59\n",
      "OS step time 0.23 (as fraction of last compression 0.39)\n",
      "step is 12 obj is 17.69699644088127 err is 0.9260017953303064 ; eta= 0.8378805552734693\n",
      "Compressing with Fourier from  7306  to  197  points.\n",
      "Compress from  61238  to  197  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0025018636952727773\n",
      "Compressed to  244  points\n",
      "Compress from  14612  to  197  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0012112157489670451\n",
      "Compressed to  298  points\n",
      "Compression time 0.70\n",
      "OS step time 0.28 (as fraction of last compression 0.40)\n",
      "step is 13 obj is 17.710596469288873 err is 0.8091215853925813 ; eta= 0.8273629790665655\n",
      "Compressing with Fourier from  7913  to  220  points.\n",
      "Compress from  65144  to  220  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.001834000408080584\n",
      "Compressed to  248  points\n",
      "Compress from  15826  to  220  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0008849250934854778\n",
      "Compressed to  336  points\n",
      "Compression time 0.98\n",
      "OS step time 0.35 (as fraction of last compression 0.35)\n",
      "step is 14 obj is 17.71894963239089 err is 0.6919461726351424 ; eta= 0.8171902541848022\n",
      "Compressing with Fourier from  8280  to  243  points.\n",
      "Compress from  69274  to  243  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.004005379843818049\n",
      "Compressed to  256  points\n",
      "Compress from  16560  to  243  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0023302263915081925\n",
      "Compressed to  332  points\n",
      "Compression time 1.15\n",
      "OS step time 0.37 (as fraction of last compression 0.32)\n",
      "step is 15 obj is 17.72594506833903 err is 0.7264223351190733 ; eta= 0.8073443754472972\n",
      "Compressing with Fourier from  9196  to  268  points.\n",
      "Compress from  73876  to  268  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0018252093836873854\n",
      "Compressed to  284  points\n",
      "Compress from  18392  to  268  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0016026895966093585\n",
      "Compressed to  354  points\n",
      "Compression time 1.39\n",
      "OS step time 0.40 (as fraction of last compression 0.29)\n",
      "step is 16 obj is 17.728990171053095 err is 0.6469813712556052 ; eta= 0.7978086137095132\n",
      "Compressing with Fourier from  9663  to  293  points.\n",
      "Compress from  79088  to  293  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0010406785748819504\n",
      "Compressed to  298  points\n",
      "Compress from  19326  to  293  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0011996673763242285\n",
      "Compressed to  366  points\n",
      "Compression time 1.68\n",
      "OS step time 0.43 (as fraction of last compression 0.25)\n",
      "step is 17 obj is 17.72659142090619 err is 0.611173515062001 ; eta= 0.7885674025390315\n",
      "Compressing with Fourier from  10227  to  319  points.\n",
      "Compress from  85100  to  319  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0012958133045920985\n",
      "Compressed to  302  points\n",
      "Compress from  20454  to  319  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0006768093810582779\n",
      "Compressed to  406  points\n",
      "Compression time 2.68\n",
      "OS step time 0.60 (as fraction of last compression 0.23)\n",
      "step is 18 obj is 17.727917411540485 err is 0.509336077667661 ; eta= 0.7796062369115665\n",
      "Compressing with Fourier from  10750  to  347  points.\n",
      "Compress from  90276  to  347  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0012943244944004933\n",
      "Compressed to  316  points\n",
      "Compress from  21500  to  347  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0007277900828990549\n",
      "Compressed to  410  points\n",
      "Compression time 3.22\n",
      "OS step time 0.66 (as fraction of last compression 0.20)\n",
      "step is 19 obj is 17.730317447646854 err is 0.5310341320594052 ; eta= 0.7709115824524779\n",
      "Compressing with Fourier from  11521  to  375  points.\n",
      "Compress from  97172  to  375  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.00167377174889913\n",
      "Compressed to  324  points\n",
      "Compress from  23042  to  375  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.000766725840923249\n",
      "Compressed to  416  points\n",
      "Compression time 2.87\n",
      "OS step time 0.70 (as fraction of last compression 0.24)\n",
      "step is 20 obj is 17.72836488052907 err is 0.46983208903862295 ; eta= 0.7624707939535903\n",
      "Compressing with Fourier from  12889  to  404  points.\n",
      "Compress from  103610  to  404  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0007209757070718854\n",
      "Compressed to  340  points\n",
      "Compress from  25778  to  404  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0011217476117349667\n",
      "Compressed to  450  points\n",
      "Compression time 3.41\n",
      "OS step time 0.73 (as fraction of last compression 0.21)\n",
      "step is 21 obj is 17.729067164833037 err is 0.3980730404477062 ; eta= 0.7542720420681454\n",
      "Compressing with Fourier from  13298  to  434  points.\n",
      "Compress from  111278  to  434  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.00045717676717790205\n",
      "Compressed to  358  points\n",
      "Compress from  26596  to  434  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0007997051394240237\n",
      "Compressed to  450  points\n",
      "Compression time 5.67\n",
      "OS step time 0.77 (as fraction of last compression 0.14)\n",
      "step is 22 obj is 17.724358060876504 err is 0.4417722923144769 ; eta= 0.7463042472335311\n",
      "Compressing with Fourier from  14291  to  466  points.\n",
      "Compress from  119204  to  466  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0003627169800701102\n",
      "Compressed to  376  points\n",
      "Compress from  28582  to  466  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0005295487612384657\n",
      "Compressed to  494  points\n",
      "Compression time 4.64\n",
      "OS step time 0.79 (as fraction of last compression 0.17)\n",
      "step is 23 obj is 17.72058741354237 err is 0.3590195328234014 ; eta= 0.7385570199964042\n",
      "Compressing with Fourier from  15398  to  498  points.\n",
      "Compress from  127136  to  498  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0001844654873466698\n",
      "Compressed to  384  points\n",
      "Compress from  30796  to  498  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0006830572229172904\n",
      "Compressed to  492  points\n",
      "Compression time 7.58\n",
      "OS step time 1.03 (as fraction of last compression 0.14)\n",
      "step is 24 obj is 17.72185807903101 err is 0.32792830504153603 ; eta= 0.7310206070215147\n",
      "Compressing with Fourier from  16503  to  531  points.\n",
      "Compress from  136018  to  531  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.00039265053624114397\n",
      "Compressed to  380  points\n",
      "Compress from  33006  to  531  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0003739315475118652\n",
      "Compressed to  502  points\n",
      "Compression time 6.63\n",
      "OS step time 1.08 (as fraction of last compression 0.16)\n",
      "step is 25 obj is 17.71192513711018 err is 0.33762298889638487 ; eta= 0.7236858421568871\n",
      "Compressing with Fourier from  17453  to  565  points.\n",
      "Compress from  145830  to  565  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.00039568579427396903\n",
      "Compressed to  390  points\n",
      "Compress from  34906  to  565  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.000548990126687297\n",
      "Compressed to  500  points\n",
      "Compression time 9.70\n",
      "OS step time 1.16 (as fraction of last compression 0.12)\n",
      "step is 26 obj is 17.704496118108946 err is 0.4309357762021291 ; eta= 0.7165441020064369\n",
      "Compressing with Fourier from  18430  to  600  points.\n",
      "Compress from  154758  to  600  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.00035558022408173286\n",
      "Compressed to  410  points\n",
      "Compress from  36860  to  600  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.00040870099052886776\n",
      "Compressed to  542  points\n",
      "Compression time 10.14\n",
      "OS step time 1.22 (as fraction of last compression 0.12)\n",
      "step is 27 obj is 17.703340164071623 err is 0.26678820532680425 ; eta= 0.7095872655286124\n",
      "Compressing with Fourier from  19757  to  636  points.\n",
      "Compress from  165826  to  636  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0006103114934480992\n",
      "Compressed to  418  points\n",
      "Compress from  39514  to  636  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0005823919743965304\n",
      "Compressed to  542  points\n",
      "Compression time 11.73\n",
      "OS step time 1.28 (as fraction of last compression 0.11)\n",
      "step is 28 obj is 17.697309961321533 err is 0.2787449438153944 ; eta= 0.7028076772379166\n",
      "Compressing with Fourier from  20955  to  673  points.\n",
      "Compress from  175856  to  673  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0004160645828053784\n",
      "Compressed to  424  points\n",
      "Compress from  41910  to  673  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0003549639945428775\n",
      "Compressed to  558  points\n",
      "Compression time 13.36\n",
      "OS step time 1.35 (as fraction of last compression 0.10)\n",
      "step is 29 obj is 17.696844374869343 err is 0.22872488032956984 ; eta= 0.696198113636572\n",
      "Compressing with Fourier from  22281  to  711  points.\n",
      "Compress from  186772  to  711  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0003607452263898887\n",
      "Compressed to  462  points\n",
      "Compress from  44562  to  711  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0005289833356707251\n",
      "Compressed to  574  points\n",
      "Compression time 15.93\n",
      "OS step time 1.68 (as fraction of last compression 0.11)\n",
      "step is 30 obj is 17.695584268459776 err is 0.2257548788784831 ; eta= 0.6897517525473114\n",
      "Compressing with Fourier from  23584  to  750  points.\n",
      "Compress from  199068  to  750  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.00024307954251999607\n",
      "Compressed to  442  points\n",
      "Compress from  47168  to  750  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.00039478867072845485\n",
      "Compressed to  596  points\n",
      "Compression time 20.04\n",
      "OS step time 1.77 (as fraction of last compression 0.09)\n",
      "step is 31 obj is 17.692589248917546 err is 0.19171658983772866 ; eta= 0.6834621450562824\n",
      "Compressing with Fourier from  25327  to  789  points.\n",
      "Compress from  210260  to  789  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0002483421928738504\n",
      "Compressed to  468  points\n",
      "Compress from  50654  to  789  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.00029579476243717713\n",
      "Compressed to  610  points\n",
      "Compression time 22.16\n",
      "OS step time 1.89 (as fraction of last compression 0.09)\n",
      "step is 32 obj is 17.68322780596609 err is 0.29869046000827915 ; eta= 0.6773231898081657\n",
      "Compressing with Fourier from  27048  to  830  points.\n",
      "Compress from  223132  to  830  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.00023063229537530005\n",
      "Compressed to  464  points\n",
      "Compress from  54096  to  830  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.0003228383616737935\n",
      "Compressed to  622  points\n",
      "Compression time 25.65\n",
      "OS step time 1.96 (as fraction of last compression 0.08)\n",
      "step is 33 obj is 17.67749189576265 err is 0.22238400857886642 ; eta= 0.6713291094245157\n",
      "OS step time 4.93 (as fraction of last compression 0.19)\n",
      "step is 34 obj is 17.673752983245564 err is 0.1835673267792588 ; eta= 0.6654744288416317\n",
      "Time usage for OS with Fourier is 206.847137298\n"
     ]
    }
   ],
   "source": [
    "print(\"running compressed online sinkhorn with Fourier..\")\n",
    "params[\"compress\"] = True\n",
    "params[\"method\"] =\"Fourier\"\n",
    "params[\"fourier_solver\"] =\"nnls\" # \"tl\" \"nnls\" \"sklearn\"\n",
    "start = timeit.default_timer()\n",
    "(ft_f, gt_f, x_f, y_f, obj_f, err_f) = algorithms.online_sinkhorn(params)\n",
    "run_time = timeit.default_timer()-start\n",
    "data_cos_f={\"obj\":obj_f,\"err\":err_f,\"rt\":run_time,\"solver\":params[\"fourier_solver\"]}\n",
    "print(\"Time usage for OS with Fourier is\", run_time)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 74,
   "metadata": {
    "executionInfo": {
     "elapsed": 10,
     "status": "ok",
     "timestamp": 1694665028634,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "pvDHL3-HyyRa"
   },
   "outputs": [],
   "source": [
    "if (False): # leave as False unless you want to compare run times of fourier_solver\n",
    "  print(\"running compressed online sinkhorn with Fourier..\")\n",
    "  params[\"compress\"] = True\n",
    "  params[\"fourier_solver\"] =\"celer\"\n",
    "  start = timeit.default_timer()\n",
    "  (ft_f2, gt_f2, x_f2, y_f2, obj_f2, err_f2) = algorithms.online_sinkhorn(params)\n",
    "  run_time = timeit.default_timer()-start\n",
    "  data_cos_f2={\"obj\":obj_f2,\"err\":err_f2,\"rt\":run_time,\"solver\":params[\"fourier_solver\"]}\n",
    "  print(\"Time usage for OS with Fourier is\", run_time)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 75,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 392685,
     "status": "ok",
     "timestamp": 1694665421310,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "qLDKFx6_ZTR5",
    "outputId": "d61ce224-8dc5-4e84-af00-06bf7c2b0972"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "running compressed online sinkhorn with Fourier..\n",
      "Online Sinkhorn with Compression= True\n",
      "Compression method:  Fourier\n",
      "Fourier solver= tl\n",
      "Running 5 Sinkhorn iterations to start-up.\n",
      "step is 0 obj is 15.241857468976345 err is 3.7418828361991254 ; eta= 1.0 OS step time 0.02\n",
      "Compressing with Fourier from  1005  to  15  points.\n",
      "Compress from  2010  to  15  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  30  points\n",
      "Compress from  2010  to  15  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  30  points\n",
      "Compression time 0.04\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/content/drive/My Drive/Colab Notebooks/compressedOT/recombination2.py:138: RuntimeWarning: invalid value encountered in divide\n",
      "  alpha = mu/Phi[:,0]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "OS step time 0.14 (as fraction of last compression 3.72)\n",
      "step is 1 obj is 16.04822008964202 err is 51.533136123259126 ; eta= 0.9832395200760898\n",
      "Compressing with Fourier from  4798  to  23  points.\n",
      "Compress from  39474  to  23  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  46  points\n",
      "Compress from  9596  to  23  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  46  points\n",
      "Compression time 0.17\n",
      "OS step time 0.15 (as fraction of last compression 0.89)\n",
      "step is 2 obj is 16.888097476474403 err is 6.760060012150262 ; eta= 0.967207804357326\n",
      "Compressing with Fourier from  4848  to  32  points.\n",
      "Compress from  40006  to  32  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  66  points\n",
      "Compress from  9696  to  32  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  66  points\n",
      "Compression time 0.20\n",
      "OS step time 0.16 (as fraction of last compression 0.77)\n",
      "step is 3 obj is 16.973702142680423 err is 6.645953477600299 ; eta= 0.9518547218427332\n",
      "Compressing with Fourier from  4819  to  43  points.\n",
      "Compress from  40920  to  43  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  86  points\n",
      "Compress from  9638  to  43  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  86  points\n",
      "Compression time 0.25\n",
      "OS step time 0.16 (as fraction of last compression 0.64)\n",
      "step is 4 obj is 17.274258940856743 err is 4.368798358214597 ; eta= 0.9371347933586296\n",
      "Compressing with Fourier from  4997  to  56  points.\n",
      "Compress from  41926  to  56  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  114  points\n",
      "Compress from  9994  to  56  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  114  points\n",
      "Compression time 0.35\n",
      "OS step time 0.16 (as fraction of last compression 0.46)\n",
      "step is 5 obj is 17.46150286197656 err is 2.243918441492792 ; eta= 0.9230066538429028\n",
      "Compressing with Fourier from  5192  to  70  points.\n",
      "Compress from  43234  to  70  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  142  points\n",
      "Compress from  10384  to  70  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  142  points\n",
      "Compression time 0.49\n",
      "OS step time 0.17 (as fraction of last compression 0.34)\n",
      "step is 6 obj is 17.60390154473772 err is 1.3721086462135212 ; eta= 0.9094325884438981\n",
      "Compressing with Fourier from  5345  to  85  points.\n",
      "Compress from  44648  to  85  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  170  points\n",
      "Compress from  10690  to  85  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  170  points\n",
      "Compression time 0.57\n",
      "OS step time 0.18 (as fraction of last compression 0.31)\n",
      "step is 7 obj is 17.809498980441187 err is 2.9119333301964514 ; eta= 0.8963781307771418\n",
      "Compressing with Fourier from  5663  to  101  points.\n",
      "Compress from  46844  to  101  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  202  points\n",
      "Compress from  11326  to  101  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  202  points\n",
      "Compression time 0.60\n",
      "OS step time 0.18 (as fraction of last compression 0.31)\n",
      "step is 8 obj is 17.969017552888186 err is 4.558855997541683 ; eta= 0.8838117137533721\n",
      "Compressing with Fourier from  5958  to  118  points.\n",
      "Compress from  48492  to  118  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  238  points\n",
      "Compress from  11916  to  118  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  238  points\n",
      "Compression time 0.75\n",
      "OS step time 0.19 (as fraction of last compression 0.25)\n",
      "step is 9 obj is 18.056347917886143 err is 1.713754055767584 ; eta= 0.8717043650550409\n",
      "Compressing with Fourier from  6167  to  136  points.\n",
      "Compress from  51358  to  136  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  274  points\n",
      "Compress from  12334  to  136  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  274  points\n",
      "Compression time 0.91\n",
      "OS step time 0.20 (as fraction of last compression 0.22)\n",
      "step is 10 obj is 18.11204810367589 err is 1.6391415088199528 ; eta= 0.8600294406827096\n",
      "Compressing with Fourier from  6592  to  156  points.\n",
      "Compress from  54114  to  156  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  314  points\n",
      "Compress from  13184  to  156  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  314  points\n",
      "Compression time 1.24\n",
      "OS step time 0.22 (as fraction of last compression 0.18)\n",
      "step is 11 obj is 18.104278422718995 err is 1.0565101398286496 ; eta= 0.8487623910845714\n",
      "Compressing with Fourier from  6894  to  176  points.\n",
      "Compress from  57608  to  176  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  354  points\n",
      "Compress from  13788  to  176  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  354  points\n",
      "Compression time 2.54\n",
      "OS step time 0.37 (as fraction of last compression 0.15)\n",
      "step is 12 obj is 18.126583711024438 err is 0.7571070180549084 ; eta= 0.8378805552734693\n",
      "Compressing with Fourier from  7304  to  197  points.\n",
      "Compress from  61490  to  197  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  394  points\n",
      "Compress from  14608  to  197  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  394  points\n",
      "Compression time 2.51\n",
      "OS step time 0.42 (as fraction of last compression 0.17)\n",
      "step is 13 obj is 18.117630149757577 err is 1.407058681185116 ; eta= 0.8273629790665655\n",
      "Compressing with Fourier from  7827  to  220  points.\n",
      "Compress from  65360  to  220  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  442  points\n",
      "Compress from  15654  to  220  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  442  points\n",
      "Compression time 2.14\n",
      "OS step time 0.45 (as fraction of last compression 0.21)\n",
      "step is 14 obj is 18.143969922303743 err is 0.7855849433008126 ; eta= 0.8171902541848022\n",
      "Compressing with Fourier from  8352  to  243  points.\n",
      "Compress from  69564  to  243  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  486  points\n",
      "Compress from  16704  to  243  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  486  points\n",
      "Compression time 2.53\n",
      "OS step time 0.52 (as fraction of last compression 0.21)\n",
      "step is 15 obj is 18.140313335756957 err is 1.3391287598406656 ; eta= 0.8073443754472972\n",
      "Compressing with Fourier from  9118  to  268  points.\n",
      "Compress from  74410  to  268  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  538  points\n",
      "Compress from  18236  to  268  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  538  points\n",
      "Compression time 3.36\n",
      "OS step time 0.60 (as fraction of last compression 0.18)\n",
      "step is 16 obj is 18.138810170113537 err is 1.8362727721809384 ; eta= 0.7978086137095132\n",
      "Compressing with Fourier from  9692  to  293  points.\n",
      "Compress from  79548  to  293  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  586  points\n",
      "Compress from  19384  to  293  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  586  points\n",
      "Compression time 5.21\n",
      "OS step time 0.58 (as fraction of last compression 0.11)\n",
      "step is 17 obj is 18.144335617159296 err is 0.9748721973094516 ; eta= 0.7885674025390315\n",
      "Compressing with Fourier from  10286  to  319  points.\n",
      "Compress from  85382  to  319  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  638  points\n",
      "Compress from  20572  to  319  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  638  points\n",
      "Compression time 4.37\n",
      "OS step time 0.61 (as fraction of last compression 0.14)\n",
      "step is 18 obj is 18.138684639751183 err is 0.7412147097160062 ; eta= 0.7796062369115665\n",
      "Compressing with Fourier from  10994  to  347  points.\n",
      "Compress from  91320  to  347  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  694  points\n",
      "Compress from  21988  to  347  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  694  points\n",
      "Compression time 6.46\n",
      "OS step time 0.69 (as fraction of last compression 0.11)\n",
      "step is 19 obj is 18.138802522711224 err is 1.0399439228746363 ; eta= 0.7709115824524779\n",
      "Compressing with Fourier from  11785  to  375  points.\n",
      "Compress from  97598  to  375  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  750  points\n",
      "Compress from  23570  to  375  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  750  points\n",
      "Compression time 6.22\n",
      "OS step time 0.71 (as fraction of last compression 0.11)\n",
      "step is 20 obj is 18.122732161010063 err is 0.6195380876052896 ; eta= 0.7624707939535903\n",
      "Compressing with Fourier from  12645  to  404  points.\n",
      "Compress from  105208  to  404  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  810  points\n",
      "Compress from  25290  to  404  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  810  points\n",
      "Compression time 9.18\n",
      "OS step time 0.73 (as fraction of last compression 0.08)\n",
      "step is 21 obj is 18.126198533111342 err is 0.9970335979018987 ; eta= 0.7542720420681454\n",
      "Compressing with Fourier from  13698  to  434  points.\n",
      "Compress from  112054  to  434  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  870  points\n",
      "Compress from  27396  to  434  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  870  points\n",
      "Compression time 8.31\n",
      "OS step time 0.78 (as fraction of last compression 0.09)\n",
      "step is 22 obj is 18.135671458716363 err is 0.4890637433259881 ; eta= 0.7463042472335311\n",
      "Compressing with Fourier from  14531  to  466  points.\n",
      "Compress from  119680  to  466  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  934  points\n",
      "Compress from  29062  to  466  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  934  points\n",
      "Compression time 11.22\n",
      "OS step time 0.81 (as fraction of last compression 0.07)\n",
      "step is 23 obj is 18.12145901520058 err is 0.2969111829508275 ; eta= 0.7385570199964042\n",
      "Compressing with Fourier from  15551  to  498  points.\n",
      "Compress from  128366  to  498  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  998  points\n",
      "Compress from  31102  to  498  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  998  points\n",
      "Compression time 13.17\n",
      "OS step time 1.04 (as fraction of last compression 0.08)\n",
      "step is 24 obj is 18.11629226045323 err is 0.41091755130343444 ; eta= 0.7310206070215147\n",
      "Compressing with Fourier from  16408  to  531  points.\n",
      "Compress from  136924  to  531  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1062  points\n",
      "Compress from  32816  to  531  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1062  points\n",
      "Compression time 15.30\n",
      "OS step time 1.09 (as fraction of last compression 0.07)\n",
      "step is 25 obj is 18.107468958664995 err is 0.5805601066111756 ; eta= 0.7236858421568871\n",
      "Compressing with Fourier from  17872  to  565  points.\n",
      "Compress from  145724  to  565  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1130  points\n",
      "Compress from  35744  to  565  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1130  points\n",
      "Compression time 19.14\n",
      "OS step time 1.16 (as fraction of last compression 0.06)\n",
      "step is 26 obj is 18.098385610475507 err is 0.46501723967746544 ; eta= 0.7165441020064369\n",
      "Compressing with Fourier from  18908  to  600  points.\n",
      "Compress from  155634  to  600  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1202  points\n",
      "Compress from  37816  to  600  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1202  points\n",
      "Compression time 21.16\n",
      "OS step time 1.24 (as fraction of last compression 0.06)\n",
      "step is 27 obj is 18.079216737930935 err is 0.4056550179697558 ; eta= 0.7095872655286124\n",
      "Compressing with Fourier from  20410  to  636  points.\n",
      "Compress from  166260  to  636  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1274  points\n",
      "Compress from  40820  to  636  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1274  points\n",
      "Compression time 25.12\n",
      "OS step time 1.30 (as fraction of last compression 0.05)\n",
      "step is 28 obj is 18.07483253982919 err is 0.546448626814346 ; eta= 0.7028076772379166\n",
      "Compressing with Fourier from  21442  to  673  points.\n",
      "Compress from  176444  to  673  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1346  points\n",
      "Compress from  42884  to  673  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1346  points\n",
      "Compression time 28.66\n",
      "OS step time 1.36 (as fraction of last compression 0.05)\n",
      "step is 29 obj is 18.078544699614245 err is 0.3336793458385827 ; eta= 0.696198113636572\n",
      "Compressing with Fourier from  22606  to  711  points.\n",
      "Compress from  187506  to  711  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1422  points\n",
      "Compress from  45212  to  711  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1422  points\n",
      "Compression time 33.61\n",
      "OS step time 1.70 (as fraction of last compression 0.05)\n",
      "step is 30 obj is 18.07348571617789 err is 0.797356976168544 ; eta= 0.6897517525473114\n",
      "Compressing with Fourier from  24379  to  750  points.\n",
      "Compress from  199678  to  750  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1502  points\n",
      "Compress from  48758  to  750  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1502  points\n",
      "Compression time 40.08\n",
      "OS step time 1.80 (as fraction of last compression 0.04)\n",
      "step is 31 obj is 18.05054356145979 err is 0.496562874599662 ; eta= 0.6834621450562824\n",
      "Compressing with Fourier from  25688  to  789  points.\n",
      "Compress from  212246  to  789  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1578  points\n",
      "Compress from  51376  to  789  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1578  points\n",
      "Compression time 43.75\n",
      "OS step time 1.92 (as fraction of last compression 0.04)\n",
      "step is 32 obj is 18.03255363178876 err is 0.46069823809711763 ; eta= 0.6773231898081657\n",
      "Compressing with Fourier from  27393  to  830  points.\n",
      "Compress from  224624  to  830  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1662  points\n",
      "Compress from  54786  to  830  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1662  points\n",
      "Compression time 50.87\n",
      "OS step time 2.03 (as fraction of last compression 0.04)\n",
      "step is 33 obj is 18.028224781334778 err is 0.5128733170828879 ; eta= 0.6713291094245157\n",
      "OS step time 4.96 (as fraction of last compression 0.10)\n",
      "step is 34 obj is 18.016890467143085 err is 0.15520532411373544 ; eta= 0.6654744288416317\n",
      "Time usage for OS with Fourier is 392.57528376899995\n"
     ]
    }
   ],
   "source": [
    "if (True): # leave as False unless you want to compare run times of fourier_solver\n",
    "  print(\"running compressed online sinkhorn with Fourier..\")\n",
    "  params[\"compress\"] = True\n",
    "  params[\"fourier_solver\"] =\"tl\"\n",
    "  start = timeit.default_timer()\n",
    "  (ft_f3, gt_f3, x_f3, y_f3, obj_f3, err_f3) = algorithms.online_sinkhorn(params)\n",
    "  run_time = timeit.default_timer()-start\n",
    "  data_cos_f3={\"obj\":obj_f3,\"err\":err_f3,\"rt\":run_time,\"solver\":params[\"fourier_solver\"]}\n",
    "  print(\"Time usage for OS with Fourier is\", run_time)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 76,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 432974,
     "status": "ok",
     "timestamp": 1694665854276,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "2L38LtCe4hQT",
    "outputId": "e0fe449f-066e-44b2-bc02-404588efb103"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "running online sinkhorn...\n",
      "Online Sinkhorn with Compression= False\n",
      "Running 5 Sinkhorn iterations to start-up.\n",
      "step is 0 obj is 15.132260409444953 err is 3.9978170676446587 ; eta= 1.0 OS step time 0.02\n",
      "step is 1 obj is 15.593420330650853 err is 3.4999680859307154 ; eta= 0.9832395200760898 OS step time 0.16\n",
      "step is 2 obj is 15.968655450009365 err is 3.2039119175443243 ; eta= 0.967207804357326 OS step time 0.34\n",
      "step is 3 obj is 16.268760576308253 err is 2.796245003436347 ; eta= 0.9518547218427332 OS step time 0.52\n",
      "step is 4 obj is 16.51211863266982 err is 2.5621156500625974 ; eta= 0.9371347933586296 OS step time 0.71\n",
      "step is 5 obj is 16.72343919010906 err is 2.2785029137060766 ; eta= 0.9230066538429028 OS step time 0.90\n",
      "step is 6 obj is 16.905702719195585 err is 2.1088687872499436 ; eta= 0.9094325884438981 OS step time 1.09\n",
      "step is 7 obj is 17.052224931583318 err is 1.9007371476732553 ; eta= 0.8963781307771418 OS step time 1.38\n",
      "step is 8 obj is 17.17519055742226 err is 1.8197425161178797 ; eta= 0.8838117137533721 OS step time 1.61\n",
      "step is 9 obj is 17.28276002950893 err is 1.6091610178302553 ; eta= 0.8717043650550409 OS step time 1.86\n",
      "step is 10 obj is 17.378278545490502 err is 1.6161033453588862 ; eta= 0.8600294406827096 OS step time 2.11\n",
      "step is 11 obj is 17.445276083248466 err is 1.2934416846426386 ; eta= 0.8487623910845714 OS step time 2.38\n",
      "step is 12 obj is 17.51542797989582 err is 1.3064893020437678 ; eta= 0.8378805552734693 OS step time 2.67\n",
      "step is 13 obj is 17.562983812265845 err is 1.1021765920080222 ; eta= 0.8273629790665655 OS step time 3.67\n",
      "step is 14 obj is 17.61378136266468 err is 1.198992267404969 ; eta= 0.8171902541848022 OS step time 4.40\n",
      "step is 15 obj is 17.651411204983912 err is 1.038215081861626 ; eta= 0.8073443754472972 OS step time 4.85\n",
      "step is 16 obj is 17.689212671196877 err is 0.9473298008597881 ; eta= 0.7978086137095132 OS step time 5.59\n",
      "step is 17 obj is 17.713029070789084 err is 0.9077789863935184 ; eta= 0.7885674025390315 OS step time 6.16\n",
      "step is 18 obj is 17.734153182409123 err is 0.8502774777719475 ; eta= 0.7796062369115665 OS step time 6.75\n",
      "step is 19 obj is 17.756279455407842 err is 0.766518777781835 ; eta= 0.7709115824524779 OS step time 7.71\n",
      "step is 20 obj is 17.766434064509337 err is 0.7687137478612094 ; eta= 0.7624707939535903 OS step time 8.44\n",
      "step is 21 obj is 17.78197705432742 err is 0.7199505938440467 ; eta= 0.7542720420681454 OS step time 9.58\n",
      "step is 22 obj is 17.79615967365025 err is 0.6516994459781902 ; eta= 0.7463042472335311 OS step time 10.43\n",
      "step is 23 obj is 17.797967838108324 err is 0.6028529669192775 ; eta= 0.7385570199964042 OS step time 11.81\n",
      "step is 24 obj is 17.805869670535298 err is 0.532366233720202 ; eta= 0.7310206070215147 OS step time 15.04\n",
      "step is 25 obj is 17.816944853103223 err is 0.49574711305066366 ; eta= 0.7236858421568871 OS step time 16.85\n",
      "step is 26 obj is 17.81945078204527 err is 0.5282940535266576 ; eta= 0.7165441020064369 OS step time 18.85\n",
      "step is 27 obj is 17.817985842421145 err is 0.4672026016426134 ; eta= 0.7095872655286124 OS step time 20.39\n",
      "step is 28 obj is 17.82336157578204 err is 0.39512266595692846 ; eta= 0.7028076772379166 OS step time 22.72\n",
      "step is 29 obj is 17.825080387298193 err is 0.4171432744281445 ; eta= 0.696198113636572 OS step time 25.26\n",
      "step is 30 obj is 17.82697133276009 err is 0.38504347978284326 ; eta= 0.6897517525473114 OS step time 30.81\n",
      "step is 31 obj is 17.8240647541817 err is 0.37294784711899087 ; eta= 0.6834621450562824 OS step time 34.04\n",
      "step is 32 obj is 17.823839077971275 err is 0.36738448370174126 ; eta= 0.6773231898081657 OS step time 37.55\n",
      "step is 33 obj is 17.82084245100528 err is 0.31468687603714596 ; eta= 0.6713291094245157 OS step time 41.37\n",
      "step is 34 obj is 17.817423078879724 err is 0.2666574872185281 ; eta= 0.6654744288416317 OS step time 45.49\n",
      "Time usage for OS is 432.7737860739999\n"
     ]
    }
   ],
   "source": [
    "print(\"running online sinkhorn...\")\n",
    "params[\"compress\"] = False\n",
    "start = timeit.default_timer()\n",
    "(ft_os, gt_os, x_os, y_os, obj_os, err_os) = algorithms.online_sinkhorn(params)\n",
    "stop3 = timeit.default_timer()\n",
    "run_time = timeit.default_timer()-start\n",
    "data_os={\"obj\":obj_os,\"err\":err_os,\"rt\":run_time}\n",
    "print(\"Time usage for OS is\", run_time)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 77,
   "metadata": {
    "executionInfo": {
     "elapsed": 6,
     "status": "ok",
     "timestamp": 1694665854277,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "u0mQPDwUrjMt"
   },
   "outputs": [],
   "source": [
    "# plotting the objective values\n",
    "a = params['a']\n",
    "b = params['b']\n",
    "maxits = params['maxits']\n",
    "batch = [2*math.ceil(pow(i, 2*a)) for i in range(1, maxits+2)]\n",
    "N = np.cumsum(batch)[:len(obj_os)]\n",
    "ref_value=obj_os[-1]\n",
    "#\n",
    "\n",
    "save_data={\"N\":N,\n",
    "          \"ref_value\":ref_value,\n",
    "          \"a\":a,\"b\":b,    \"d\":d,\n",
    "          \"os\":data_os,\n",
    "          \"cosf\":data_cos_f,\n",
    "          \"s\":data_s,\n",
    "           }\n",
    "if \"data_f2\" in locals():\n",
    "  save_data[\"data_f2\"]=data_f2\n",
    "if \"data_f3\" in locals():\n",
    "  save_data[\"data_f3\"]=data_f3\n",
    "if True:\n",
    "  filehandler = open('example_gmm_2d.pkl', 'wb')\n",
    "  pickle.dump(save_data, filehandler)\n",
    "  filehandler.close()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 5,
     "status": "ok",
     "timestamp": 1694665854277,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "-sM2ApdUG83Z",
    "outputId": "46898903-abe8-4f83-b06c-c9c13bbe45fd"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[ 90100  99576 109734]\n"
     ]
    }
   ],
   "source": [
    "import pickle\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as pl\n",
    "filehandler = open('example_gmm_2d.pkl','rb')\n",
    "in_data= pickle.load(filehandler)\n",
    "d=in_data[\"d\"]\n",
    "\n",
    "N=in_data[\"N\"]\n",
    "print(N[-3:])\n",
    "ref_value=in_data[\"ref_value\"]\n",
    "obj_os=in_data[\"os\"][\"obj\"]\n",
    "obj_f=in_data[\"cosf\"][\"obj\"]\n",
    "obj_s=in_data[\"s\"][\"obj\"]\n",
    "err_os=in_data[\"os\"][\"err\"]\n",
    "err_rec=in_data[\"cosf\"][\"err\"]\n",
    "a=in_data[\"a\"]\n",
    "run_time3=in_data[\"os\"][\"rt\"]\n",
    "run_time1=in_data[\"cosf\"][\"rt\"]\n",
    "#"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 526
    },
    "executionInfo": {
     "elapsed": 778,
     "status": "ok",
     "timestamp": 1694666074346,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "_V773368GUhH",
    "outputId": "605e5dd6-f69e-4bd4-efa0-b2f64fcab0fc"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "reference value 17.817423078879724\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "OS= 17.817423078879724 ; Fourier OS= 17.673752983245564 ; S= 16.39332247277587\n"
     ]
    }
   ],
   "source": [
    "print(\"reference value\",ref_value)\n",
    "stop=-18\n",
    "pl.loglog(N[3:stop], abs(ref_value-obj_os[3:stop])/ref_value, label='OS',linewidth=5)\n",
    "pl.loglog(N[3:stop], abs(ref_value-obj_f[3:stop])/ref_value,'--', label='Fourier COS',linewidth=5)\n",
    "if d==1:\n",
    "   pl.loglog(N[1:], abs(ref_value-obj_gq[1:])/ref_value,label='GQ COS',linewidth=5)\n",
    "#pl.legend(bbox_to_anchor=(1.05, 1), loc=\"upper right\", borderaxespad=0.,fontsize=\"20\" )\n",
    "\n",
    "#pl.title('Obj error to Sinkhorn')\n",
    "pl.xlabel('Samples',fontsize=20)\n",
    "pl.ylabel('Relative error (objective)',fontsize=20)\n",
    "pl.xticks(fontsize=20)\n",
    "pl.yticks(fontsize=20)\n",
    "\n",
    "pl.show()\n",
    "print(\"OS=\",obj_os[-1],'; Fourier OS=',obj_f[-1],\"; S=\",obj_s[-1])\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 514
    },
    "executionInfo": {
     "elapsed": 793,
     "status": "ok",
     "timestamp": 1694665855696,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "k2DHeq2N-CqH",
    "outputId": "9ebf516d-1ae1-406f-9b95-4086e884e853"
   },
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "dimension: 2 ; Convergence rate: theory=-0.35; Fourier OS=  -0.46;  OS=  -0.37 \n",
      "Execution time: OS 432.77 and Fourier 206.85\n"
     ]
    }
   ],
   "source": [
    "# plotting the error at every step\n",
    "pl.loglog(N[1:], err_os[1:], label='OS',linewidth=5)\n",
    "pl.loglog(N[1:], err_rec[1:], '--',label='Fourier COS',linewidth=5)\n",
    "if d==1:\n",
    "   pl.loglog(N[1:], err_gq[1:],label='GQ COS',linewidth=5)\n",
    "pl.loglog(N[1:],12*N[1:]**(-a/(2*a+1)),'k:',label='Theoretical',linewidth=5)\n",
    "#pl.legend(loc=\"upper right\", borderaxespad=0.,fontsize=\"20\" )\n",
    "pl.xlabel('Samples',fontsize=20)\n",
    "pl.ylabel('Error (variational)',fontsize=20)\n",
    "pl.xticks(fontsize=20)\n",
    "pl.yticks(fontsize=20)\n",
    "pl.show()\n",
    "i=5\n",
    "z_rec = np.polyfit(np.log(N[i:]), np.log(err_rec[i:]), 1)\n",
    "z_os = np.polyfit(np.log(N[i:]), np.log(err_os[i:]), 1)\n",
    "\n",
    "print('dimension:',d,'; Convergence rate: theory={:2.2}; Fourier OS=  {:2.2};  OS=  {:2.2} '.format(-a/(2*a+1),z_rec[0],z_os[0]))\n",
    "print('Execution time: OS {:.2f} and Fourier {:.2f}'.format(run_time3,run_time1))"
   ]
  }
 ],
 "metadata": {
  "accelerator": "GPU",
  "colab": {
   "provenance": []
  },
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.8.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 1
}
