{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "GRdoE3U-SybK"
   },
   "source": [
    "## Gaussian mixture model in 5d\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": 30,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 1796,
     "status": "ok",
     "timestamp": 1694665572462,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "Zes9WhQ1rcDp",
    "outputId": "42063c9d-61d5-4272-8546-a2f605374eaf"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Drive already mounted at /content/drive; to attempt to forcibly remount, call drive.mount(\"/content/drive\", force_remount=True).\n"
     ]
    }
   ],
   "source": [
    "from google.colab import drive\n",
    "drive.mount(\"/content/drive\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 4949,
     "status": "ok",
     "timestamp": 1694665577402,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "XevkJMUlYF3z",
    "outputId": "b9419c4d-5622-43d1-e155-65b167e66213"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Requirement already satisfied: celer in /usr/local/lib/python3.10/dist-packages (0.7.3)\n",
      "Requirement 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",
      "Requirement already satisfied: libsvmdata>=0.3 in /usr/local/lib/python3.10/dist-packages (from celer) (0.4.1)\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",
      "Requirement already satisfied: download in /usr/local/lib/python3.10/dist-packages (from celer) (0.3.5)\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"
     ]
    }
   ],
   "source": [
    "!pip install -U celer"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 7162,
     "status": "ok",
     "timestamp": 1694665584560,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "ybc4BzbN62xN",
    "outputId": "1d5f31ee-334b-4dc1-eacb-e26efbff917b"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Requirement already satisfied: pykeops in /usr/local/lib/python3.10/dist-packages (2.1.2)\n",
      "Requirement already satisfied: numpy in /usr/local/lib/python3.10/dist-packages (from pykeops) (1.23.5)\n",
      "Requirement already satisfied: pybind11 in /usr/local/lib/python3.10/dist-packages (from pykeops) (2.11.1)\n",
      "Requirement already satisfied: keopscore==2.1.2 in /usr/local/lib/python3.10/dist-packages (from pykeops) (2.1.2)\n"
     ]
    }
   ],
   "source": [
    "import locale\n",
    "locale.getpreferredencoding = lambda: \"UTF-8\"\n",
    "!pip install pykeops"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 18,
     "status": "ok",
     "timestamp": 1694665584561,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "-Ftp4rX2rio_",
    "outputId": "bcfc86fb-50da-4be2-b178-2ef97d4f6785"
   },
   "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.pkl',\n",
       " 'example_gmm_5d_am.ipynb',\n",
       " 'example_gmm_2d_am.ipynb',\n",
       " 'example_gmm_5d_am.pkl']"
      ]
     },
     "execution_count": 33,
     "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": 34,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 350,
     "status": "ok",
     "timestamp": 1694665584900,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "tfRHebKAdox5",
    "outputId": "bf48c1eb-76fd-4b50-aaa5-69880298ee87"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The autoreload extension is already loaded. To reload it, use:\n",
      "  %reload_ext autoreload\n"
     ]
    }
   ],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "metadata": {
    "executionInfo": {
     "elapsed": 20,
     "status": "ok",
     "timestamp": 1694665584901,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "6f4hsauPrjGD"
   },
   "outputs": [],
   "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": 36,
   "metadata": {
    "executionInfo": {
     "elapsed": 20,
     "status": "ok",
     "timestamp": 1694665584902,
     "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 = 5 # 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 = algorithms.random_cov_matrix(d)\n",
    "    cov2a = 0.4*algorithms.random_cov_matrix(d)\n",
    "    mean1b = -mean1a\n",
    "    mean2b = -mean2a\n",
    "    cov1b = algorithms.random_cov_matrix(d)\n",
    "    cov2b = 0.4*algorithms.random_cov_matrix(d)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "metadata": {
    "executionInfo": {
     "elapsed": 19,
     "status": "ok",
     "timestamp": 1694665584902,
     "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": 42,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 197,
     "status": "ok",
     "timestamp": 1694665694743,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "vL9MUt27399P",
    "outputId": "574568e8-51c4-4318-a94d-fa3088ec754d"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "epsilon is 0.5\n",
      "a= 1.2 ; b= -0.6\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': 25,\n",
    "          'no_initial_sinkhorn_its':30,\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",
    "#exact=algorithms.get_gaussian_potentials(cov1,cov2,mean1,mean2,params['epsilon'])\n",
    "#params['exact_f']=lambda tt: exact[0](tt)\n",
    "#params['exact_g']=lambda tt: exact[1](tt)\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": 43,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 10,
     "status": "ok",
     "timestamp": 1694665694963,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "KW2va0Ju4Qll",
    "outputId": "caa854e8-9374-4a1c-87f1-45dcae17c008"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2.90561758 9.303283   7.86439261 0.81915809 6.86172769] [-2.90561758 -9.303283   -7.86439261 -0.81915809 -6.86172769] [0.49751701 3.16172263 0.70485026 0.73329157 4.41995192] [-0.49751701 -3.16172263 -0.70485026 -0.73329157 -4.41995192]\n",
      "[[2.85080513 1.98975242 1.62712582 2.47792634 1.36818609]\n",
      " [1.98975242 1.5398667  0.92824525 1.66084293 0.88633813]\n",
      " [1.62712582 0.92824525 1.63035366 1.81919533 1.07023588]\n",
      " [2.47792634 1.66084293 1.81919533 2.5098658  1.487408  ]\n",
      " [1.36818609 0.88633813 1.07023588 1.487408   0.98961289]] [[2.33251199 1.61318015 1.46598467 2.33957456 0.89078604]\n",
      " [1.61318015 1.77549406 1.83130615 2.18892767 1.09421407]\n",
      " [1.46598467 1.83130615 1.95876584 2.23923977 1.15181284]\n",
      " [2.33957456 2.18892767 2.23923977 3.04796315 1.3269476 ]\n",
      " [0.89078604 1.09421407 1.15181284 1.3269476  0.78841516]] [[0.51931738 0.57514491 0.35329095 0.31193489 0.52033886]\n",
      " [0.57514491 1.07772809 0.56550882 0.65428001 0.60293038]\n",
      " [0.35329095 0.56550882 0.3920482  0.28986276 0.34321685]\n",
      " [0.31193489 0.65428001 0.28986276 0.48883375 0.35859226]\n",
      " [0.52033886 0.60293038 0.34321685 0.35859226 0.53579249]] [[0.43056943 0.41563702 0.33176118 0.59337423 0.32480966]\n",
      " [0.41563702 0.68838812 0.3945669  0.64965821 0.52107666]\n",
      " [0.33176118 0.3945669  0.53664589 0.7132859  0.36210201]\n",
      " [0.59337423 0.64965821 0.7132859  1.15106931 0.5964255 ]\n",
      " [0.32480966 0.52107666 0.36210201 0.5964255  0.43345727]]\n"
     ]
    }
   ],
   "source": [
    "print(mean1a,mean1b,mean2a,mean2b)\n",
    "print(cov1a,cov1b,  cov2a,cov2b)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 15213,
     "status": "ok",
     "timestamp": 1694665710170,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "3Nyyy4BQWHhc",
    "outputId": "a3575613-7b54-4a6b-aec5-3031d22b855c"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "running sinkhorn...\n",
      "objective is 65.70879102972901 ; error is 206.1337711303491\n",
      "objective is 69.74248277992703 ; error is 12.923821696353896\n",
      "objective is 73.17399577273032 ; error is 12.357938400808791\n",
      "objective is 76.2627471213496 ; error is 11.868820823177085\n",
      "objective is 79.06270515863704 ; error is 11.437032799781441\n",
      "objective is 81.59931707171964 ; error is 11.02873549491322\n",
      "objective is 83.89298464889032 ; error is 10.62623765908332\n",
      "objective is 85.96401902315898 ; error is 10.227716572914446\n",
      "objective is 87.83205234343586 ; error is 9.838183871892145\n",
      "objective is 89.51499839847466 ; error is 9.46609553095303\n",
      "objective is 91.02939973100382 ; error is 9.107749381335985\n",
      "objective is 92.39095618124422 ; error is 8.751946827751713\n",
      "objective is 93.61427697092272 ; error is 8.394708862916435\n",
      "objective is 94.71272946204287 ; error is 8.042183938985364\n",
      "objective is 95.69856514084742 ; error is 7.702287178855872\n",
      "objective is 96.58302862753492 ; error is 7.379348648400235\n",
      "objective is 97.37637396965076 ; error is 7.072216999448775\n",
      "objective is 98.0878791411995 ; error is 6.776923963574836\n",
      "objective is 98.72590456043976 ; error is 6.526273268460187\n",
      "objective is 99.29796745143098 ; error is 6.294717570966171\n",
      "objective is 99.81081063333733 ; error is 6.064685213244434\n",
      "objective is 100.27047792902235 ; error is 5.8338477805762645\n",
      "objective is 100.68240264914584 ; error is 5.597723464457778\n",
      "objective is 101.05149044909246 ; error is 5.354915386789207\n",
      "objective is 101.3821760648604 ; error is 5.109932913137059\n",
      "objective is 101.6784605359561 ; error is 4.870174950641239\n",
      "objective is 101.94394347896007 ; error is 4.641483607412434\n",
      "objective is 102.18185520358054 ; error is 4.426224787930121\n",
      "objective is 102.39508890691819 ; error is 4.224099743836538\n",
      "objective is 102.5862325229796 ; error is 4.03367782102472\n",
      "The value of the objective function for Sinkhorn is 102.5862325229796  elapsed time 15.036937285000022\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": 45,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 92309,
     "status": "ok",
     "timestamp": 1694665802466,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "mqMUVJaSrjVl",
    "outputId": "b5b38bec-e6bc-4d11-9d33-4426f0ad8b46"
   },
   "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 30 Sinkhorn iterations to start-up.\n",
      "step is 0 obj is 104.52605341940986 err is 1.3280357093862811 ; eta= 1.0 OS step time 0.02\n",
      "Compressing with Fourier from  1005  to  15  points.\n",
      "Compress from  5025  to  15  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 9.370738617859665e-16\n",
      "Compressed to  70  points\n",
      "Compress from  5025  to  15  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 6.082973080507841e-16\n",
      "Compressed to  70  points\n",
      "Compression time 0.02\n",
      "OS step time 0.35 (as fraction of last compression 14.61)\n",
      "step is 1 obj is 104.13373769879593 err is 4.5011306754158875 ; eta= 0.9832395200760898\n",
      "Compressing with Fourier from  43321  to  23  points.\n",
      "Compress from  34795  to  23  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 5.621267092910754e-16\n",
      "Compressed to  110  points\n",
      "Compress from  216605  to  23  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 5.973202994970058e-16\n",
      "Compressed to  110  points\n",
      "Compression time 0.35\n",
      "OS step time 0.34 (as fraction of last compression 1.00)\n",
      "step is 2 obj is 102.84418744168704 err is 4.55124320624784 ; eta= 0.967207804357326\n",
      "Compressing with Fourier from  44157  to  32  points.\n",
      "Compress from  34850  to  32  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.1254843123812563\n",
      "Compressed to  150  points\n",
      "Compress from  220785  to  32  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 8.116790880179818e-16\n",
      "Compressed to  160  points\n",
      "Compression time 0.38\n",
      "OS step time 0.35 (as fraction of last compression 0.94)\n",
      "step is 3 obj is 103.78847656740476 err is 5.619610196419014 ; eta= 0.9518547218427332\n",
      "Compressing with Fourier from  44925  to  43  points.\n",
      "Compress from  35825  to  43  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 6.605498247137933e-16\n",
      "Compressed to  210  points\n",
      "Compress from  224625  to  43  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.2348560521984865e-15\n",
      "Compressed to  210  points\n",
      "Compression time 0.41\n",
      "OS step time 0.36 (as fraction of last compression 0.90)\n",
      "step is 4 obj is 104.21316944012926 err is 2.3797321184941223 ; eta= 0.9371347933586296\n",
      "Compressing with Fourier from  45989  to  56  points.\n",
      "Compress from  36425  to  56  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.00014500710801949382\n",
      "Compressed to  275  points\n",
      "Compress from  229945  to  56  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 6.199207445419121e-16\n",
      "Compressed to  280  points\n",
      "Compression time 0.48\n",
      "OS step time 0.40 (as fraction of last compression 0.84)\n",
      "step is 5 obj is 104.67797007565613 err is 1.7504945526807023 ; eta= 0.9230066538429028\n",
      "Compressing with Fourier from  47437  to  70  points.\n",
      "Compress from  37900  to  70  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 0.03843887907541005\n",
      "Compressed to  335  points\n",
      "Compress from  237185  to  70  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 8.270539111385029e-16\n",
      "Compressed to  350  points\n",
      "Compression time 0.54\n",
      "OS step time 0.41 (as fraction of last compression 0.77)\n",
      "step is 6 obj is 104.88792292989969 err is 1.4102016029204565 ; eta= 0.9094325884438981\n",
      "Compressing with Fourier from  49286  to  85  points.\n",
      "Compress from  39160  to  85  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 7.24781313069649e-16\n",
      "Compressed to  420  points\n",
      "Compress from  246430  to  85  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 9.334911935486775e-16\n",
      "Compressed to  420  points\n",
      "Compression time 0.73\n",
      "OS step time 0.43 (as fraction of last compression 0.59)\n",
      "step is 7 obj is 104.95746529451722 err is 0.5886093842809337 ; eta= 0.8963781307771418\n",
      "Compressing with Fourier from  51262  to  101  points.\n",
      "Compress from  40995  to  101  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 8.172867686424911e-16\n",
      "Compressed to  500  points\n",
      "Compress from  256310  to  101  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 9.210772920251206e-16\n",
      "Compressed to  500  points\n",
      "Compression time 1.13\n",
      "OS step time 0.46 (as fraction of last compression 0.41)\n",
      "step is 8 obj is 105.05912406595056 err is 0.8796401595001706 ; eta= 0.8838117137533721\n",
      "Compressing with Fourier from  53787  to  118  points.\n",
      "Compress from  42835  to  118  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 9.819939364538857e-16\n",
      "Compressed to  590  points\n",
      "Compress from  268935  to  118  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 8.996895880263504e-16\n",
      "Compressed to  590  points\n",
      "Compression time 1.36\n",
      "OS step time 0.51 (as fraction of last compression 0.38)\n",
      "step is 9 obj is 105.14399464526002 err is 0.8916501949716609 ; eta= 0.8717043650550409\n",
      "Compressing with Fourier from  56411  to  136  points.\n",
      "Compress from  45260  to  136  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.0282245500209515e-15\n",
      "Compressed to  680  points\n",
      "Compress from  282055  to  136  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.0282889686155177e-15\n",
      "Compressed to  680  points\n",
      "Compression time 1.18\n",
      "OS step time 0.54 (as fraction of last compression 0.46)\n",
      "step is 10 obj is 105.19384080489192 err is 0.6717421764781375 ; eta= 0.8600294406827096\n",
      "Compressing with Fourier from  59582  to  156  points.\n",
      "Compress from  47820  to  156  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.2056192779662188e-15\n",
      "Compressed to  780  points\n",
      "Compress from  297910  to  156  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.1356280785993465e-15\n",
      "Compressed to  780  points\n",
      "Compression time 1.24\n",
      "OS step time 0.60 (as fraction of last compression 0.49)\n",
      "step is 11 obj is 105.2539443861339 err is 0.6101477849306489 ; eta= 0.8487623910845714\n",
      "Compressing with Fourier from  63231  to  176  points.\n",
      "Compress from  50925  to  176  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.0764120541135425e-15\n",
      "Compressed to  880  points\n",
      "Compress from  316155  to  176  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.1243752045208264e-15\n",
      "Compressed to  880  points\n",
      "Compression time 1.47\n",
      "OS step time 0.63 (as fraction of last compression 0.43)\n",
      "step is 12 obj is 105.26743716052869 err is 0.7418028631121416 ; eta= 0.8378805552734693\n",
      "Compressing with Fourier from  66962  to  197  points.\n",
      "Compress from  53445  to  197  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 9.388241927509648e-16\n",
      "Compressed to  980  points\n",
      "Compress from  334810  to  197  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.200288769199169e-15\n",
      "Compressed to  980  points\n",
      "Compression time 1.64\n",
      "OS step time 0.73 (as fraction of last compression 0.44)\n",
      "step is 13 obj is 105.3326133652609 err is 0.4599477644980361 ; eta= 0.8273629790665655\n",
      "Compressing with Fourier from  71434  to  220  points.\n",
      "Compress from  57915  to  220  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.4011186949589292e-15\n",
      "Compressed to  1100  points\n",
      "Compress from  357170  to  220  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.181225632561035e-15\n",
      "Compressed to  1100  points\n",
      "Compression time 1.93\n",
      "OS step time 0.78 (as fraction of last compression 0.40)\n",
      "step is 14 obj is 105.3717351520776 err is 0.5119649191381574 ; eta= 0.8171902541848022\n",
      "Compressing with Fourier from  76416  to  243  points.\n",
      "Compress from  61575  to  243  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 8.826912497627492e-16\n",
      "Compressed to  1210  points\n",
      "Compress from  382080  to  243  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.3500891482696703e-15\n",
      "Compressed to  1210  points\n",
      "Compression time 3.18\n",
      "OS step time 1.03 (as fraction of last compression 0.33)\n",
      "step is 15 obj is 105.38064275454946 err is 0.5510138158222508 ; eta= 0.8073443754472972\n",
      "Compressing with Fourier from  81575  to  268  points.\n",
      "Compress from  64980  to  268  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.0266650414783213e-15\n",
      "Compressed to  1340  points\n",
      "Compress from  407875  to  268  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.0858164930267652e-15\n",
      "Compressed to  1340  points\n",
      "Compression time 2.51\n",
      "OS step time 0.97 (as fraction of last compression 0.39)\n",
      "step is 16 obj is 105.40701589827569 err is 0.35141040501308307 ; eta= 0.7978086137095132\n",
      "Compressing with Fourier from  87241  to  293  points.\n",
      "Compress from  70525  to  293  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.3854671422479504e-15\n",
      "Compressed to  1460  points\n",
      "Compress from  436205  to  293  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.4996807960363696e-15\n",
      "Compressed to  1460  points\n",
      "Compression time 3.52\n",
      "OS step time 1.26 (as fraction of last compression 0.36)\n",
      "step is 17 obj is 105.4450690255974 err is 0.42479008566323184 ; eta= 0.7885674025390315\n",
      "Compressing with Fourier from  93394  to  319  points.\n",
      "Compress from  75080  to  319  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.0611038428133753e-15\n",
      "Compressed to  1590  points\n",
      "Compress from  466970  to  319  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.3414468351394666e-15\n",
      "Compressed to  1590  points\n",
      "Compression time 5.31\n",
      "OS step time 1.32 (as fraction of last compression 0.25)\n",
      "step is 18 obj is 105.44747245461187 err is 0.2102538255702342 ; eta= 0.7796062369115665\n",
      "Compressing with Fourier from  99881  to  347  points.\n",
      "Compress from  80455  to  347  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.5085537391468463e-15\n",
      "Compressed to  1730  points\n",
      "Compress from  499405  to  347  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.5591571508213574e-15\n",
      "Compressed to  1730  points\n",
      "Compression time 5.55\n",
      "OS step time 1.47 (as fraction of last compression 0.26)\n",
      "step is 19 obj is 105.46386304423926 err is 0.45204213879934496 ; eta= 0.7709115824524779\n",
      "Compressing with Fourier from  107025  to  375  points.\n",
      "Compress from  86735  to  375  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.577152651674947e-15\n",
      "Compressed to  1870  points\n",
      "Compress from  535125  to  375  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.186954257416979e-15\n",
      "Compressed to  1870  points\n",
      "Compression time 6.39\n",
      "OS step time 1.63 (as fraction of last compression 0.26)\n",
      "step is 20 obj is 105.48297595585287 err is 0.2600818596498282 ; eta= 0.7624707939535903\n",
      "Compressing with Fourier from  114603  to  404  points.\n",
      "Compress from  92180  to  404  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.4212520167453968e-15\n",
      "Compressed to  2020  points\n",
      "Compress from  573015  to  404  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.44586004856971e-15\n",
      "Compressed to  2020  points\n",
      "Compression time 5.48\n",
      "OS step time 1.77 (as fraction of last compression 0.32)\n",
      "step is 21 obj is 105.49514442020484 err is 0.29748153688140855 ; eta= 0.7542720420681454\n",
      "Compressing with Fourier from  122709  to  434  points.\n",
      "Compress from  98915  to  434  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 9.814026296237528e-16\n",
      "Compressed to  2170  points\n",
      "Compress from  613545  to  434  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.4247497912313018e-15\n",
      "Compressed to  2165  points\n",
      "Compression time 6.77\n",
      "OS step time 2.01 (as fraction of last compression 0.30)\n",
      "step is 22 obj is 105.5164390260305 err is 0.26756812753267756 ; eta= 0.7463042472335311\n",
      "Compressing with Fourier from  131155  to  466  points.\n",
      "Compress from  105165  to  466  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.5540405502444161e-15\n",
      "Compressed to  2330  points\n",
      "Compress from  655775  to  466  points ( nnls )\n",
      "Reduced compression method\n",
      "Relative residual= 1.4484338582956631e-15\n",
      "Compressed to  2330  points\n",
      "Compression time 9.27\n",
      "OS step time 2.20 (as fraction of last compression 0.24)\n",
      "step is 23 obj is 105.52927853346412 err is 0.21537674600344348 ; eta= 0.7385570199964042\n",
      "OS step time 6.14 (as fraction of last compression 0.66)\n",
      "step is 24 obj is 105.54057997377498 err is 0.2584818176125854 ; eta= 0.7310206070215147\n",
      "Time usage for OS with Fourier is 92.2608235319999\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)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "metadata": {
    "executionInfo": {
     "elapsed": 13,
     "status": "ok",
     "timestamp": 1694665802468,
     "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)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 140260,
     "status": "ok",
     "timestamp": 1694665942719,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "qLDKFx6_ZTR5",
    "outputId": "33f0ad1e-ecf9-4aa5-b0cd-7eef8b08c8c2"
   },
   "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 30 Sinkhorn iterations to start-up.\n",
      "step is 0 obj is 104.8732109994207 err is 1.307495694296354 ; eta= 1.0 OS step time 0.02\n",
      "Compressing with Fourier from  1005  to  15  points.\n",
      "Compress from  5025  to  15  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  75  points\n",
      "Compress from  5025  to  15  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  75  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.35 (as fraction of last compression 8.22)\n",
      "step is 1 obj is 103.82738160557423 err is 5.612231050141901 ; eta= 0.9832395200760898\n",
      "Compressing with Fourier from  43552  to  23  points.\n",
      "Compress from  34365  to  23  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  115  points\n",
      "Compress from  217760  to  23  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  115  points\n",
      "Compression time 0.55\n",
      "OS step time 0.36 (as fraction of last compression 0.65)\n",
      "step is 2 obj is 104.03297467026019 err is 2.219258903640462 ; eta= 0.967207804357326\n",
      "Compressing with Fourier from  44113  to  32  points.\n",
      "Compress from  34745  to  32  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  165  points\n",
      "Compress from  220565  to  32  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  165  points\n",
      "Compression time 0.64\n",
      "OS step time 0.37 (as fraction of last compression 0.57)\n",
      "step is 3 obj is 104.20694890341628 err is 4.145447717898071 ; eta= 0.9518547218427332\n",
      "Compressing with Fourier from  44881  to  43  points.\n",
      "Compress from  35805  to  43  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  215  points\n",
      "Compress from  224405  to  43  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  215  points\n",
      "Compression time 0.66\n",
      "OS step time 0.38 (as fraction of last compression 0.58)\n",
      "step is 4 obj is 104.53309293177745 err is 6.134485897800751 ; eta= 0.9371347933586296\n",
      "Compressing with Fourier from  46162  to  56  points.\n",
      "Compress from  36860  to  56  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  285  points\n",
      "Compress from  230810  to  56  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  285  points\n",
      "Compression time 0.85\n",
      "OS step time 0.41 (as fraction of last compression 0.48)\n",
      "step is 5 obj is 104.7173237043001 err is 1.374916805806413 ; eta= 0.9230066538429028\n",
      "Compressing with Fourier from  47365  to  70  points.\n",
      "Compress from  37460  to  70  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  355  points\n",
      "Compress from  236825  to  70  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  355  points\n",
      "Compression time 0.82\n",
      "OS step time 0.42 (as fraction of last compression 0.52)\n",
      "step is 6 obj is 104.85354063691494 err is 1.5480572959259078 ; eta= 0.9094325884438981\n",
      "Compressing with Fourier from  49364  to  85  points.\n",
      "Compress from  39315  to  85  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  425  points\n",
      "Compress from  246820  to  85  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  425  points\n",
      "Compression time 1.13\n",
      "OS step time 0.44 (as fraction of last compression 0.39)\n",
      "step is 7 obj is 104.9561024899752 err is 0.5811226497642699 ; eta= 0.8963781307771418\n",
      "Compressing with Fourier from  51233  to  101  points.\n",
      "Compress from  41020  to  101  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  505  points\n",
      "Compress from  256165  to  101  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  505  points\n",
      "Compression time 1.23\n",
      "OS step time 0.46 (as fraction of last compression 0.37)\n",
      "step is 8 obj is 104.90923076466774 err is 1.08544947854415 ; eta= 0.8838117137533721\n",
      "Compressing with Fourier from  53804  to  118  points.\n",
      "Compress from  42965  to  118  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  595  points\n",
      "Compress from  269020  to  118  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  595  points\n",
      "Compression time 1.53\n",
      "OS step time 0.51 (as fraction of last compression 0.33)\n",
      "step is 9 obj is 104.9757470588965 err is 0.7567210524005361 ; eta= 0.8717043650550409\n",
      "Compressing with Fourier from  56436  to  136  points.\n",
      "Compress from  45040  to  136  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  685  points\n",
      "Compress from  282180  to  136  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  685  points\n",
      "Compression time 1.72\n",
      "OS step time 0.54 (as fraction of last compression 0.31)\n",
      "step is 10 obj is 104.97791193654082 err is 0.4134914705439665 ; eta= 0.8600294406827096\n",
      "Compressing with Fourier from  59520  to  156  points.\n",
      "Compress from  47845  to  156  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  785  points\n",
      "Compress from  297600  to  156  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  785  points\n",
      "Compression time 3.33\n",
      "OS step time 0.79 (as fraction of last compression 0.24)\n",
      "step is 11 obj is 105.11134791552952 err is 0.8359099306161264 ; eta= 0.8487623910845714\n",
      "Compressing with Fourier from  63062  to  176  points.\n",
      "Compress from  50400  to  176  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  885  points\n",
      "Compress from  315310  to  176  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  885  points\n",
      "Compression time 2.42\n",
      "OS step time 0.64 (as fraction of last compression 0.27)\n",
      "step is 12 obj is 105.15108182797728 err is 0.6900947708942491 ; eta= 0.8378805552734693\n",
      "Compressing with Fourier from  67119  to  197  points.\n",
      "Compress from  54020  to  197  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  985  points\n",
      "Compress from  335595  to  197  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  985  points\n",
      "Compression time 2.71\n",
      "OS step time 0.87 (as fraction of last compression 0.32)\n",
      "step is 13 obj is 105.21209202591291 err is 0.7070631090423531 ; eta= 0.8273629790665655\n",
      "Compressing with Fourier from  71563  to  220  points.\n",
      "Compress from  57475  to  220  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1105  points\n",
      "Compress from  357815  to  220  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1105  points\n",
      "Compression time 3.32\n",
      "OS step time 0.95 (as fraction of last compression 0.29)\n",
      "step is 14 obj is 105.26422262295594 err is 0.7790155501040119 ; eta= 0.8171902541848022\n",
      "Compressing with Fourier from  76300  to  243  points.\n",
      "Compress from  61165  to  243  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1215  points\n",
      "Compress from  381500  to  243  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1215  points\n",
      "Compression time 4.90\n",
      "OS step time 1.04 (as fraction of last compression 0.21)\n",
      "step is 15 obj is 105.31105433459176 err is 0.8279883025401187 ; eta= 0.8073443754472972\n",
      "Compressing with Fourier from  81648  to  268  points.\n",
      "Compress from  65465  to  268  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1345  points\n",
      "Compress from  408240  to  268  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1345  points\n",
      "Compression time 4.70\n",
      "OS step time 1.11 (as fraction of last compression 0.24)\n",
      "step is 16 obj is 105.34891670829111 err is 0.2863878577845336 ; eta= 0.7978086137095132\n",
      "Compressing with Fourier from  87393  to  293  points.\n",
      "Compress from  70370  to  293  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1465  points\n",
      "Compress from  436965  to  293  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1465  points\n",
      "Compression time 7.45\n",
      "OS step time 1.26 (as fraction of last compression 0.17)\n",
      "step is 17 obj is 105.38375702337132 err is 0.4649527187337823 ; eta= 0.7885674025390315\n",
      "Compressing with Fourier from  93295  to  319  points.\n",
      "Compress from  75760  to  319  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1595  points\n",
      "Compress from  466475  to  319  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1595  points\n",
      "Compression time 6.58\n",
      "OS step time 1.33 (as fraction of last compression 0.20)\n",
      "step is 18 obj is 105.4062213725656 err is 0.3732705248230843 ; eta= 0.7796062369115665\n",
      "Compressing with Fourier from  99952  to  347  points.\n",
      "Compress from  80310  to  347  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1735  points\n",
      "Compress from  499760  to  347  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1735  points\n",
      "Compression time 8.44\n",
      "OS step time 1.46 (as fraction of last compression 0.17)\n",
      "step is 19 obj is 105.43021375829284 err is 0.2003385622252729 ; eta= 0.7709115824524779\n",
      "Compressing with Fourier from  107115  to  375  points.\n",
      "Compress from  86605  to  375  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1875  points\n",
      "Compress from  535575  to  375  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  1875  points\n",
      "Compression time 11.09\n",
      "OS step time 1.59 (as fraction of last compression 0.14)\n",
      "step is 20 obj is 105.42395706743662 err is 0.25136568207643606 ; eta= 0.7624707939535903\n",
      "Compressing with Fourier from  114702  to  404  points.\n",
      "Compress from  92190  to  404  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  2025  points\n",
      "Compress from  573510  to  404  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  2025  points\n",
      "Compression time 12.88\n",
      "OS step time 1.77 (as fraction of last compression 0.14)\n",
      "step is 21 obj is 105.45395720122838 err is 0.6568481040640015 ; eta= 0.7542720420681454\n",
      "Compressing with Fourier from  122858  to  434  points.\n",
      "Compress from  98995  to  434  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  2175  points\n",
      "Compress from  614290  to  434  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  2175  points\n",
      "Compression time 14.82\n",
      "OS step time 2.00 (as fraction of last compression 0.14)\n",
      "step is 22 obj is 105.47692397776112 err is 0.44125522568736697 ; eta= 0.7463042472335311\n",
      "Compressing with Fourier from  131245  to  466  points.\n",
      "Compress from  105820  to  466  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  2335  points\n",
      "Compress from  656225  to  466  points ( tl )\n",
      "Full compression method\n",
      "Compressed to  2335  points\n",
      "Compression time 16.60\n",
      "OS step time 2.21 (as fraction of last compression 0.13)\n",
      "step is 23 obj is 105.50860613733553 err is 0.5263965743106733 ; eta= 0.7385570199964042\n",
      "OS step time 6.11 (as fraction of last compression 0.37)\n",
      "step is 24 obj is 105.51267841187385 err is 0.21741960192392806 ; eta= 0.7310206070215147\n",
      "Time usage for OS with Fourier is 140.17263671800004\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)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 386158,
     "status": "ok",
     "timestamp": 1694666328870,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "2L38LtCe4hQT",
    "outputId": "7704ac3b-6e94-4abf-d906-a9657806cd53"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "running online sinkhorn...\n",
      "Online Sinkhorn with Compression= False\n",
      "Running 30 Sinkhorn iterations to start-up.\n",
      "step is 0 obj is 104.99975505333177 err is 1.2589776783999471 ; eta= 1.0 OS step time 0.02\n",
      "step is 1 obj is 105.06728494641732 err is 1.5050398251903205 ; eta= 0.9832395200760898 OS step time 0.38\n",
      "step is 2 obj is 105.14007579044147 err is 0.9812309078041572 ; eta= 0.967207804357326 OS step time 0.87\n",
      "step is 3 obj is 105.18936434020688 err is 0.852536928856586 ; eta= 0.9518547218427332 OS step time 1.37\n",
      "step is 4 obj is 105.2367407490429 err is 0.7244704514064613 ; eta= 0.9371347933586296 OS step time 1.88\n",
      "step is 5 obj is 105.28152751953725 err is 0.6910019445498743 ; eta= 0.9230066538429028 OS step time 2.51\n",
      "step is 6 obj is 105.31709324184379 err is 0.659228364752285 ; eta= 0.9094325884438981 OS step time 3.42\n",
      "step is 7 obj is 105.34672184424878 err is 0.5863235002085809 ; eta= 0.8963781307771418 OS step time 4.53\n",
      "step is 8 obj is 105.37382693596021 err is 0.5656309357593718 ; eta= 0.8838117137533721 OS step time 5.31\n",
      "step is 9 obj is 105.399611314234 err is 0.5195855784859194 ; eta= 0.8717043650550409 OS step time 6.35\n",
      "step is 10 obj is 105.4234447149889 err is 0.43681058764835434 ; eta= 0.8600294406827096 OS step time 7.23\n",
      "step is 11 obj is 105.4403985017968 err is 0.49700636845810786 ; eta= 0.8487623910845714 OS step time 8.44\n",
      "step is 12 obj is 105.4570359826214 err is 0.375637304828202 ; eta= 0.8378805552734693 OS step time 9.49\n",
      "step is 13 obj is 105.47526034811719 err is 0.39570516530898026 ; eta= 0.8273629790665655 OS step time 10.93\n",
      "step is 14 obj is 105.48878562417926 err is 0.37704300706646343 ; eta= 0.8171902541848022 OS step time 12.14\n",
      "step is 15 obj is 105.50381417674599 err is 0.34391623803448024 ; eta= 0.8073443754472972 OS step time 13.86\n",
      "step is 16 obj is 105.51489184638254 err is 0.35461663749843986 ; eta= 0.7978086137095132 OS step time 15.75\n",
      "step is 17 obj is 105.52742371712688 err is 0.30174568746239494 ; eta= 0.7885674025390315 OS step time 17.86\n",
      "step is 18 obj is 105.53738006537489 err is 0.3458750253990548 ; eta= 0.7796062369115665 OS step time 22.85\n",
      "step is 19 obj is 105.5471240046325 err is 0.26548160698817824 ; eta= 0.7709115824524779 OS step time 25.72\n",
      "step is 20 obj is 105.55597726918323 err is 0.27166084978298954 ; eta= 0.7624707939535903 OS step time 28.74\n",
      "step is 21 obj is 105.56375906003888 err is 0.24534259600632957 ; eta= 0.7542720420681454 OS step time 32.13\n",
      "step is 22 obj is 105.57260819301996 err is 0.255174877920032 ; eta= 0.7463042472335311 OS step time 36.50\n",
      "step is 23 obj is 105.57922894701213 err is 0.20153933884234565 ; eta= 0.7385570199964042 OS step time 40.62\n",
      "step is 24 obj is 105.58412050704807 err is 0.2520154758281299 ; eta= 0.7310206070215147 OS step time 50.60\n",
      "Time usage for OS is 386.0907620060001\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",
    "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": 49,
   "metadata": {
    "executionInfo": {
     "elapsed": 12,
     "status": "ok",
     "timestamp": 1694666328871,
     "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 \"cosf2\" in locals():\n",
    "  save_data[\"data_f2\"]=data_f2\n",
    "if \"cosf3\" in locals():\n",
    "  save_data[\"data_f3\"]=data_f3\n",
    "if (True):\n",
    "  filehandler = open('example_gmm_5d.pkl', 'wb')\n",
    "  pickle.dump(save_data, filehandler)\n",
    "  filehandler.close()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 50,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 8,
     "status": "ok",
     "timestamp": 1694666328871,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "-sM2ApdUG83Z",
    "outputId": "145ab383-156d-43de-a398-937cbf060fc5"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[26996 31104 35634]\n"
     ]
    }
   ],
   "source": [
    "import pickle\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as pl\n",
    "filehandler = open('example_gmm_5d.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\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 54,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 514
    },
    "executionInfo": {
     "elapsed": 986,
     "status": "ok",
     "timestamp": 1694666351180,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "_V773368GUhH",
    "outputId": "c7ddd17e-cee0-499e-cc71-dfe47ebf4f02"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "reference value 105.58412050704807\n"
     ]
    },
    {
     "data": {
      "image/png": "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YHr16nXN66tWreKGG24gICCAM2fOlOpFpCIlxWKxkJWVRYUKFa55PSsri5tuuonly5ezYMEC+vfvX0IRipQNV65coXPnznh7e9OoUSO++uorlydv+ieYuMydd955XeIG0KNHD/r06cPFixfZuXNnCUQmUvp5eHhcl7gBeHl5MXjwYAAOHTrk6rBEypw33niD3bt3M3XqVDw9PUskBiVv5URcXBwLFizgpZde4tZbb6VatWq5c/Vjx44tUlvHjx9n4sSJNG3aFH9/f6pWrUqHDh145513SE1NLVZ83t7egPWDRqS0cefny2Kx8MsvvwAQFRVV5PoiJc2dnq8tW7bwxhtv8PLLLxMZGVnM38gBTCkXgHy/xowZY3M78+fPNytXrpxvW40bNzYPHjxYpNiOHz9u+vj4mDVr1jSzsrKK+JuJlDx3er4yMjLMl19+2XzppZfMRx55xGzatKkJmOPGjbPztxQpGe7yfKWnp5vNmzc327dvn/tZNWbMGBMw161bZ++vWSQaeSuHwsPD6devX5Hrbd26leHDh5OUlERAQABvvPEGa9euZenSpTz44IMAHDhwgP79+5OcnGxTm1euXOHee+8lIyODt99+u8SGoEUcpaSfr8zMTF599VVee+01oqOj2b9/P08//TSTJ08u9u8k4i5K8vl66aWXOHjwINOmTSv5zyqXpopSYl566SXzp59+Ms+ePWuapmkePXq0yP9y6dGjhwmYXl5e5tq1a697/1//+ldumy+//HKh7WVnZ5v33HOPCZgPPvhgUX4dEbfirs/XyZMnzY8//tgMCgoyu3XrZl66dKkov5aIW3CH52vt2rWmh4eH+dprr13zekmNvCl5K6eK+pd/w4YNueXHjx+fZ5ns7GyzWbNmJmAGBQWZmZmZ+baXnZ2d+5d+1KhRZnZ2dnF/FRG3U9LP15999913JmA+88wzNtcRcVeufr6uXLliNmrUyGzduvV1z52mTcWtzZs3L/f7cePG5VnGw8OD0aNHA5CYmMiyZcvyLGexWBg3bhwzZsxgxIgRTJ8+XWdPSbnmyOcrLznTTMuXLy92jCKllb3P1+XLlzl48CDbtm2jQoUKuZslDMNgxowZAHTp0gXDMK7py5m0tU9ssnr1agD8/f1p165dvuWuPgpkzZo1161NyEncZs6cyfDhw/nyyy9Lfu2ASAlz1POVn9On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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "OS= 105.58412050704807 ; Fourier OS= 105.54057997377498 ; S= 102.5862325229796\n"
     ]
    }
   ],
   "source": [
    "print(\"reference value\",ref_value)\n",
    "stop=-7\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])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 56,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 514
    },
    "executionInfo": {
     "elapsed": 828,
     "status": "ok",
     "timestamp": 1694666366072,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "k2DHeq2N-CqH",
    "outputId": "ba79e9ec-b8a5-4f90-ff26-18df39e57d3a"
   },
   "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: 5 ; Convergence rate: theory=-0.35; Fourier OS=  -0.43;  OS=  -0.23 \n",
      "Execution time: OS 386.09 and Fourier 92.26\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:],8*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",
    "\n",
    "pl.show()\n",
    "i=1\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))\n"
   ]
  }
 ],
 "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
}
