{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "GRdoE3U-SybK"
   },
   "source": [
    "## Compare convergence rates 2d for Online Sinkhorn\n",
    "\n",
    "This notebook compares the theoretical convergence rates with the old paper.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 3974,
     "status": "ok",
     "timestamp": 1694678697445,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "Zes9WhQ1rcDp",
    "outputId": "c41b0fb3-eccd-47f3-80fe-f39f06dd1a12"
   },
   "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"
     ]
    },
    {
     "data": {
      "text/plain": [
       "['OS_theortical_cts_2d.ipynb',\n",
       " 'closed_forms.py',\n",
       " 'helper.py',\n",
       " 'gq.py',\n",
       " 'recombination.py',\n",
       " 'recombination2.py',\n",
       " 'README.md',\n",
       " 'compression.py',\n",
       " 'algorithms.py',\n",
       " 'OS_theoretical_cts_2d.pkl',\n",
       " 'OS_theoretical_cts_5d.pkl',\n",
       " 'example_gmm_5d.pkl',\n",
       " 'example_gmm_2d.pkl',\n",
       " 'example_cts_1d.ipynb',\n",
       " 'example_cts_1d.pkl',\n",
       " 'example_gmm_5d.ipynb',\n",
       " 'example_gmm_2d.ipynb',\n",
       " 'OS_theortical_cts_5d.ipynb',\n",
       " '__pycache__',\n",
       " 'OS_theoretical_cts_1d_test.pkl',\n",
       " 'OS_theoretical_cts_1d.pkl',\n",
       " 'OS_theortical_cts_1d.ipynb']"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from google.colab import drive\n",
    "import os\n",
    "drive.mount(\"/content/drive\")\n",
    "path=\"/content/drive/My Drive/Colab Notebooks/COT-gcopy\"\n",
    "os.chdir(path)\n",
    "os.listdir(path)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 5857,
     "status": "ok",
     "timestamp": 1694678703296,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "FQYGMCvpsMZN",
    "outputId": "4538be48-0caf-457d-afef-82962bb367c4"
   },
   "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[?25l     \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m0.0/4.5 MB\u001b[0m \u001b[31m?\u001b[0m eta \u001b[36m-:--:--\u001b[0m\r",
      "\u001b[2K     \u001b[91m╸\u001b[0m\u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m0.1/4.5 MB\u001b[0m \u001b[31m1.8 MB/s\u001b[0m eta \u001b[36m0:00:03\u001b[0m\r",
      "\u001b[2K     \u001b[91m━━━━━━━━━━━━━━━━━━━━━\u001b[0m\u001b[91m╸\u001b[0m\u001b[90m━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m2.4/4.5 MB\u001b[0m \u001b[31m35.3 MB/s\u001b[0m eta \u001b[36m0:00:01\u001b[0m\r",
      "\u001b[2K     \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m4.5/4.5 MB\u001b[0m \u001b[31m43.4 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 celer"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 7986,
     "status": "ok",
     "timestamp": 1694678711276,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "ybc4BzbN62xN",
    "outputId": "44e0fe94-d335-4363-de17-ff55e7bbd4ba"
   },
   "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[31m1.2 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[31m10.6 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[31m9.1 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=61043ae6f9ec962eb6904af2ac70ad86f87ad70af24aa074dd725a481cc78099\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=7909b212d7e6db8384faecc61aaedb44488917dc441178b5257bd4193515446e\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": 6,
   "metadata": {
    "executionInfo": {
     "elapsed": 13,
     "status": "ok",
     "timestamp": 1694678711277,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "tfRHebKAdox5"
   },
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 29423,
     "status": "ok",
     "timestamp": 1694678740688,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "6f4hsauPrjGD",
    "outputId": "3760876c-f8f1-4305-b50a-a60e6dc3ca66"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[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 math\n",
    "import timeit\n",
    "import random\n",
    "import pickle\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as pl\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": 8,
   "metadata": {
    "executionInfo": {
     "elapsed": 15,
     "status": "ok",
     "timestamp": 1694678740689,
     "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",
    "if d == 1:\n",
    "    mean1, cov1 = 1., 2.\n",
    "    mean2, cov2 = 3., 2.5\n",
    "else:\n",
    "    mean1 = 10*np.random.rand(d)\n",
    "    mean2 = 5*np.random.rand(d)\n",
    "    cov1 = algorithms.random_cov_matrix(d)\n",
    "    cov2 = algorithms.random_cov_matrix(d)\n",
    "## objective function\n",
    "def loss(f,g):\n",
    "  deg=20\n",
    "  return algorithms.GaussHermiteIntegrate(mean1,cov1,f,mean2,cov2,g,deg)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {
    "executionInfo": {
     "elapsed": 15,
     "status": "ok",
     "timestamp": 1694678741336,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "Cam41gGGR206"
   },
   "outputs": [],
   "source": [
    "# test distribution\n",
    "def samp1(n, d):\n",
    "    # generate samples from test distribution 2\n",
    "    if d == 1:\n",
    "      x = np.random.normal(mean1,cov1,n)\n",
    "    else:\n",
    "      x = np.random.multivariate_normal(mean1, cov1, n)\n",
    "    return x\n",
    "# test distribution\n",
    "def samp2(n, d):\n",
    "    # generate samples from test distribution 2\n",
    "    if d == 1:\n",
    "      x = np.random.normal(mean2,cov2,n)\n",
    "    else:\n",
    "      x = np.random.multivariate_normal(mean2, cov2, n)\n",
    "    return x"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 14,
     "status": "ok",
     "timestamp": 1694678741336,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "vL9MUt27399P",
    "outputId": "a75e3cc8-2110-4394-b43d-10e0fbcbeaa2"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Exact form\n",
      "A mean= [1.56909848 7.962906  ] covariance= [[0.09869516 0.20145461]\n",
      " [0.20145461 0.53981043]]\n",
      "B mean= [1.09556344 0.39069921] covariance= [[0.13311458 0.23233201]\n",
      " [0.23233201 0.93537485]]\n",
      "epsilon is 0.3\n",
      "a= 1.7 ; 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=1000\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.7, # n_t=t^{2a} for a>1+b (number of samples upto iteration t)\n",
    "          'epsilon': 0.3, # regularisation parametr\n",
    "          \"dim\": d,\n",
    "          'maxits': 20,\n",
    "          'no_initial_sinkhorn_its':30,\n",
    "          'compress':True,\n",
    "          'min_compress':1000,\n",
    "          'get_samples1':samp1,\n",
    "          'get_samples2':samp2,\n",
    "          'zeta':0.95, # compression error rate\n",
    "          'compression_skip':3,# number of steps between compression\n",
    "          'compression_const':1, # multiply constant for working out compression\n",
    "          't_sampling':'qmc_gau',\n",
    "          'test_samples1':samp1(size,d),\n",
    "          'test_samples2':samp2(size,d),\n",
    "          'test_samples1_obj':algorithms.get_qmc_samples(mean1,cov1,size_obj ),\n",
    "          'test_samples2_obj':algorithms.get_qmc_samples(mean2,cov2,size_obj ),\n",
    "          'test_samples1_var':algorithms.get_qmc_samples(mean1,cov1,size_var ),\n",
    "          'test_samples2_var':algorithms.get_qmc_samples(mean2,cov2,size_var )}\n",
    "#\n",
    "if d==1:\n",
    "  params[\"measure\"]=loss\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",
    "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": 11,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 11,
     "status": "ok",
     "timestamp": 1694678741336,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "KW2va0Ju4Qll",
    "outputId": "895c7b9d-da0d-4eb5-8ddb-697fca00cc16"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Objective (theoretical)  58.10246395000952\n"
     ]
    }
   ],
   "source": [
    "# Theoretical results for sqeuclidean cost\n",
    "sigma=(params[\"epsilon\"]/2)**0.5\n",
    "if d == 1:\n",
    "  theoretical = closed_form(np.array([[cov1]]), np.array([[cov2]]), sigma, mean1, mean2)\n",
    "else:\n",
    "  theoretical = closed_form(cov1, cov2, sigma, mean1, mean2)\n",
    "#theoretical-=2*params['epsilon']\n",
    "print(\"Objective (theoretical) \",exact[2])\n",
    "#print(\"Objective (Gauss-Hermite quadrature)\",loss(params['exact_f'],params['exact_g']))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 140752,
     "status": "ok",
     "timestamp": 1694678882080,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "2L38LtCe4hQT",
    "outputId": "8fd3c6e9-e9cf-4fd2-bdf0-0d1c32117cc2"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "running online sinkhorn...\n",
      "Online Sinkhorn with Compression= False\n",
      "[KeOps] Generating code for formula Max_SumShiftExpWeight_Reduction(Concat((Var(0,1,1)-Sum((Var(1,2,1)-Var(2,2,0))**2))/Var(3,1,2),1),0) ... OK\n",
      "Running 30 Sinkhorn iterations to start-up.\n",
      "step is 0 obj is 58.121838896800206 err is 0.2060908785935549 ; eta= 1.0 OS step time 0.02\n",
      "step is 1 obj is 58.12181831942887 err is 0.24990695943739638 ; eta= 0.9832395200760898 OS step time 0.08\n",
      "step is 2 obj is 58.12264165563583 err is 0.1832238405065656 ; eta= 0.967207804357326 OS step time 0.15\n",
      "step is 3 obj is 58.12257691775367 err is 0.06913849702369035 ; eta= 0.9518547218427332 OS step time 0.23\n",
      "step is 4 obj is 58.1224494027392 err is 0.18291973103716064 ; eta= 0.9371347933586296 OS step time 0.33\n",
      "step is 5 obj is 58.12267626716084 err is 0.14646967818356948 ; eta= 0.9230066538429028 OS step time 0.47\n",
      "step is 6 obj is 58.122767287792456 err is 0.051381141005693465 ; eta= 0.9094325884438981 OS step time 0.62\n",
      "step is 7 obj is 58.12240732717292 err is 0.10281226721353143 ; eta= 0.8963781307771418 OS step time 0.84\n",
      "step is 8 obj is 58.12258714739906 err is 0.07561258295830164 ; eta= 0.8838117137533721 OS step time 1.12\n",
      "step is 9 obj is 58.12262024340876 err is 0.09865261355324506 ; eta= 0.8717043650550409 OS step time 1.50\n",
      "step is 10 obj is 58.12258499814503 err is 0.04566483278263256 ; eta= 0.8600294406827096 OS step time 1.98\n",
      "step is 11 obj is 58.12278816584219 err is 0.056471157310496345 ; eta= 0.8487623910845714 OS step time 2.67\n",
      "step is 12 obj is 58.12274257061421 err is 0.03439118507802519 ; eta= 0.8378805552734693 OS step time 3.61\n",
      "step is 13 obj is 58.1228584663839 err is 0.0761894316369407 ; eta= 0.8273629790665655 OS step time 4.85\n",
      "step is 14 obj is 58.12286763532522 err is 0.034042188780784244 ; eta= 0.8171902541848022 OS step time 7.72\n",
      "step is 15 obj is 58.12284224182298 err is 0.023605958970048135 ; eta= 0.8073443754472972 OS step time 10.27\n",
      "step is 16 obj is 58.12285180980652 err is 0.04218077286132882 ; eta= 0.7978086137095132 OS step time 13.42\n",
      "step is 17 obj is 58.12281129335151 err is 0.03013196714332622 ; eta= 0.7885674025390315 OS step time 17.57\n",
      "step is 18 obj is 58.12280603528367 err is 0.015369769358168028 ; eta= 0.7796062369115665 OS step time 26.50\n",
      "step is 19 obj is 58.122815059904525 err is 0.012589447660319664 ; eta= 0.7709115824524779 OS step time 34.23\n",
      "Time usage for OS is 140.685799083\n",
      "Objective (theoretical)  58.10246395000952\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={\"rt\":run_time,\"obj\":obj_os,\"err\":err_os}\n",
    "print(\"Time usage for OS is\", run_time)\n",
    "print(\"Objective (theoretical) \",exact[2])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {
    "executionInfo": {
     "elapsed": 11,
     "status": "ok",
     "timestamp": 1694678882081,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "t4y--YKor6ga"
   },
   "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=theoretical"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {
    "executionInfo": {
     "elapsed": 1455,
     "status": "ok",
     "timestamp": 1694678883528,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "EWaY4L1Br6ga"
   },
   "outputs": [],
   "source": [
    "save_data={\"N\":N,\"ref_value\":ref_value,\n",
    "            \"os\":data_os,\n",
    "           \"a\":a,\"b\":b,\"d\":d,\"theoretical\":theoretical\n",
    "           }\n",
    "if True:\n",
    "  filehandler = open('OS_theoretical_cts_2d.pkl', 'wb')\n",
    "  pickle.dump(save_data, filehandler)\n",
    "  filehandler.close()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "executionInfo": {
     "elapsed": 4,
     "status": "ok",
     "timestamp": 1694678883528,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "A5iwINx0r6ga"
   },
   "outputs": [],
   "source": [
    "import pickle\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as pl\n",
    "filehandler = open('OS_theoretical_cts_2d.pkl','rb')\n",
    "in_data= pickle.load(filehandler)\n",
    "\n",
    "N=in_data[\"N\"]\n",
    "ref_value=in_data[\"ref_value\"]\n",
    "obj_os=in_data[\"os\"][\"obj\"]\n",
    "err_os=in_data[\"os\"][\"err\"]\n",
    "a=in_data[\"a\"]\n",
    "b=in_data[\"b\"]\n",
    "d=in_data[\"d\"]\n",
    "theoretical=in_data[\"theoretical\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "executionInfo": {
     "elapsed": 572,
     "status": "ok",
     "timestamp": 1694678884743,
     "user": {
      "displayName": "Tony Shardlow",
      "userId": "01347584962082294341"
     },
     "user_tz": -60
    },
    "id": "k2DHeq2N-CqH",
    "outputId": "c5ad4ca9-1430-47a7-c8fa-f5cde67228a7"
   },
   "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.39; old theory=-0.14; OS=  -0.37 \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:],2*N[1:]**(-a/(2*a+1)),'k:',label='Theoretical',linewidth=4)\n",
    "pl.loglog(N[1:],0.5*N[1:]**(b/(2*a+1)),'k--',label='Old Theoretical',linewidth=4)\n",
    "#pl.legend(loc=\"upper right\", borderaxespad=0.,fontsize=\"18\")\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_os = np.polyfit(np.log(N[i:]), np.log(err_os[i:]), 1)\n",
    "print('; dimension:',d,'; Convergence rate: theory={:2.2}; old theory={:2.2}; OS=  {:2.2} '.format(-a/(2*a+1),b/(2*a+1),z_os[0]))"
   ]
  }
 ],
 "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
}
