{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {
    "colab_type": "text",
    "id": "view-in-github"
   },
   "source": [
    "<a href=\"https://colab.research.google.com/github/gle-bellier/flow-matching/blob/main/Flow_Matching.ipynb\" target=\"_parent\"><img src=\"https://colab.research.google.com/assets/colab-badge.svg\" alt=\"Open In Colab\"/></a>"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "Ln6lNwXtIl_G"
   },
   "source": [
    "# Flow Matching for Generative Modeling\n",
    "_notebook by Georges Le Bellier_  - [Twitter](https://twitter.com/_lebellig), [Website](https://gle-bellier.github.io)\n",
    "\n",
    "\n",
    "⚠️ This is not the official implementation of the original paper.\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "Uhh45Iolt-Ky"
   },
   "source": [
    "This notebook centers around the **Flow Matching for Generative Modeling** article [1] and proposes an implementation of _Flow Matching_ in the case of _Optimal Transport conditional Vector Fields_. The implementation proposed in [2] was consulted and it inspired the use of the _Zuko_ [3] package for ODE solving and sampling. Moreover, this notebook adopts the notations of the original article and thus the numbers of the equations are the same as in the paper.\n",
    "\n",
    "## References:\n",
    "\n",
    "📄 [1] **Flow Matching for Generative Modeling** by Yaron Lipman, Ricky T. Q. Chen, Heli Ben-Hamu, Maximilian Nickel, Matt Le - [Article](https://arxiv.org/abs/2210.02747)\n",
    "\n",
    "🐍 [2] **Flow Matching in 100 LOC** by François Rozet - [Implementation](https://gist.github.com/francois-rozet/fd6a820e052157f8ac6e2aa39e16c1aa)\n",
    "\n",
    "🐍 [3] **Zuko** package - [Website](https://zuko.readthedocs.io/en/stable/index.html)\n",
    "\n",
    "📄 [4] **Score-Based Generative Modeling through Stochastic Differential Equations** by Yang Song, Jascha Sohl-Dickstein, Diederik P. Kingma, Abhishek Kumar, Stefano Ermon, Ben Poole - [Article](https://arxiv.org/abs/2011.13456)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "iQNuNue1mdN7"
   },
   "source": [
    "## Introduction\n",
    "\n",
    "The article [1] reminds the reader with the following notations:\n",
    "- $\\mathbb{R}^d$ denote the adata space with data points $x = (x^1, ..., x^d) \\in \\mathbb{R}^d$. \n",
    "- $v: [0, 1] \\times \\mathbb{R}^d \\rightarrow \\mathbb{R}^d$ is the _time-dependant vector field_. \n",
    "- $\\phi: [0, 1] \\times \\mathbb{R}^d \\rightarrow \\mathbb{R}^d$ is a _flow_. A flow is associated to a vector field $v_t$ thanks to the following equations:\n",
    "$$\n",
    "\\begin{cases}\n",
    "  \\frac{d}{dt}\\phi_t(x) = v_t(\\phi_t(x))\\\\ \n",
    "  \\phi_0(x) = x \\tag{1, 2}\\\\\n",
    "\\end{cases}\n",
    "$$ \n",
    "\n",
    "The main idea is then to use a neural network to model the vector field $v_t$, which implies that $\\phi_t$ will also be a parametric model and we call it a  _Continuous Normalizing Flow_ (CNF). In the context of generative modeling, the CNF allows us to transform a simple prior distribution (_e.g._ gaussian) $p_0$ into a more complex one $p_1$ (which we want close to the data distribution $q$ we need to model) by using the _push-forward equation_:\n",
    "$$\n",
    " p_t = [\\phi_t]_*p_0 \\tag{3}\n",
    "$$\n",
    "Where:\n",
    "$$\n",
    " \\forall x \\in \\mathbb{R}^2, [\\phi_t]_*p_0(x) = p_0(\\phi^{-1}(x))\\det\\left[\\frac{\\partial\\phi_t^{-1}}{\\partial x}(x)\\right]\n",
    "$$\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "Thus the _Flow Matching_ aims at learning the true vector field $u_t$:\n",
    "\n",
    "$$\n",
    "\\mathcal{L}_{FM}(\\theta) = \\mathbb{E}_{t, p_t(x)} \\lVert v_t(x) - u_t(x) \\rVert^2 \\tag{5}\n",
    "$$\n",
    "\n",
    "However this objectif is intractable, so the article [1] introduces the _Conditional Flow Matching_ (CFM) objective where $x_1 \\sim q(x_1) \\approx p_1(x_1)$ :\n",
    "\n",
    "$$\n",
    "\\mathcal{L}_{CFM}(\\theta) = \\mathbb{E}_{t, q(x_1), p_t(x|x_1)} \\lVert v_t(x) - u_t(x|x_1) \\rVert^2 \\tag{9}\n",
    "$$ \n",
    "\n",
    "This objective can be declined in several ways depending on the choice of the form of the conditional probability paths. In this notebook we will focus on the case of _Optimal Transport conditional Vector Fields_ and will leave the _Diffusion conditional Vector Fields_ for future works.\n",
    "\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "cellView": "form",
    "id": "fOTJ1mwJDR8y"
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "UsageError: Line magic function `%%capture` not found.\n"
     ]
    }
   ],
   "source": [
    "#@title ⚠️ TODO: imports\n",
    "%%capture \n",
    "!pip install zuko\n",
    "!pip install torchdiffeq\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "\n",
    "from sklearn.datasets import make_moons, make_swiss_roll\n",
    "from sklearn.preprocessing import StandardScaler\n",
    "from torch import Tensor\n",
    "from torch.distributions import Normal\n",
    "from torch.utils.data import TensorDataset, DataLoader\n",
    "from zuko.utils import odeint\n",
    "from tqdm import tqdm\n",
    "from typing import *\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "\n",
    "\n",
    "device = torch.device(\"cuda:0\" if torch.cuda.is_available() else \"cpu\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "ruFBhLGxuxfA"
   },
   "source": [
    "## Optimal Transport conditional Vector Fields:\n",
    "\n",
    "In this section, we will propose an implementation of the _Optimal Transport conditional Vector Fields_. Please refer to the article for more details about the motivations behind this concept.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "li2ynn-5vhzV"
   },
   "outputs": [],
   "source": [
    "class OTFlowMatching:\n",
    "  \n",
    "  def __init__(self, sig_min: float = 0.001) -> None:\n",
    "    super().__init__()\n",
    "    self.sig_min = sig_min"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "L2737wxMvsyP"
   },
   "source": [
    "Let $x_1 \\sim q(x_1)$, _i.e._ a point from the dataset, and we can define the conditional flow $\\psi_t$ (conditional version of $\\phi_t$) to be the Optimal Transport _displacement map_ between the two Gaussians $p_0(x|x_1)$ and $p_1(x|x_1)$ (we can proove that $\\psi_t$ satisfies Eq.3)\n",
    "$$\n",
    "\\psi_t(x) = (1 - (1 - \\sigma_{min})t)x + tx_1 \\tag{22}\n",
    "$$\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "AarEQZ2tvwj0"
   },
   "outputs": [],
   "source": [
    "  def psi_t(self, x: torch.Tensor, x_1: torch.Tensor, t: torch.Tensor) -> torch.Tensor:\n",
    "    \"\"\" Conditional Flow\n",
    "    \"\"\"\n",
    "    t = t[:, None].expand(x.shape)\n",
    "    return (1 - (1 - self.sig_min) * t) * x + t * x_1"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "tuZBpMQ2q48w"
   },
   "source": [
    "In the case of _Optimal Transport conditional VFs_ the conditional flow matching objective loss can be written as follow:\n",
    "$$\n",
    "\\mathcal{L}_{CFM}(\\theta) = \\mathbb{E}_{t, q(x_1), p(x_0)} \\big\\lVert v_t(\\psi_t(x_0)) - \\frac{d}{dt}\\psi_t(x_0) \\big\\rVert^2 \\tag{14}\n",
    "$$\n",
    "Given (22), we have:\n",
    "$$\n",
    "\\frac{d}{dt}\\psi_t(x_0) = (x_1 - (1 - \\sigma_{min})x_0)\n",
    "$$\n",
    "Thus, we obtain the final form of the _Conditional Flow Matching_ loss:\n",
    "$$\n",
    "\\mathcal{L}_{CFM}(\\theta) = \\mathbb{E}_{t, q(x_1), p(x_0)} \\big\\lVert v_t(\\psi_t(x_0)) - (x_1 - (1 - \\sigma_{min})x_0) \\big\\rVert^2 \\tag{23}\n",
    "$$\n",
    "We sample $x_1 \\sim q(x_1)$ (samples from the dataset), $x_0 \\sim p(x_0)$ (samples from the _prior distribution_) and $t \\sim \\text{Unif}([0, 1])$. It is worth noticing that we apply a little trick for time sampling since we batch covers the range $[0, 1]$ uniformly with a random offset. Then we compute an approximation of the _CFM Loss_ on the batch: \n",
    "$$\n",
    "\\mathcal{L}_{\\text{batch}}(\\theta) = \\frac{1}{N}\\sum_{i=0}^{N-1} \\big\\lVert v_t\\big(\\psi_t\\big(x_0^{(i)}\\big)\\big) - \\big(x_{1}^{(i)} - \\big(1 - \\sigma_{min}\\big)x_0^{(i)}\\big) \\big\\rVert^2\n",
    "$$\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "zAFdnQ7Ov4T3"
   },
   "outputs": [],
   "source": [
    "  def loss(self, v_t: nn.Module, x_1: torch.Tensor) -> torch.Tensor:\n",
    "    \"\"\" Compute loss\n",
    "    \"\"\"\n",
    "    # t ~ Unif([0, 1])\n",
    "    t = (torch.rand(1, device=x_1.device) + torch.arange(len(x_1), device=x_1.device) / len(x_1)) % (1 - self.eps)\n",
    "    t = t[:, None].expand(x_1.shape)\n",
    "    # x ~ p_t(x_0)\n",
    "    x_0 = torch.randn_like(x_1) \n",
    "    v_psi = v_t(t[:,0], self.psi_t(x_0, x_1, t))\n",
    "    d_psi = x_1 - (1 - self.sig_min) * x_0\n",
    "    return torch.mean((v_psi - d_psi) ** 2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "cellView": "form",
    "id": "rhMwA-BJIrd3"
   },
   "outputs": [],
   "source": [
    "#@title ⏳ Summary: please run this cell which contains the ```OTFlowMatching``` class\n",
    "class OTFlowMatching:\n",
    "  \n",
    "  def __init__(self, sig_min: float = 0.001) -> None:\n",
    "    super().__init__()\n",
    "    self.sig_min = sig_min\n",
    "    self.eps = 1e-5\n",
    "\n",
    "  def psi_t(self, x: torch.Tensor, x_1: torch.Tensor, t: torch.Tensor) -> torch.Tensor:\n",
    "    \"\"\" Conditional Flow\n",
    "    \"\"\"\n",
    "    return (1 - (1 - self.sig_min) * t) * x + t * x_1\n",
    "\n",
    "  def loss(self, v_t: nn.Module, x_1: torch.Tensor) -> torch.Tensor:\n",
    "    \"\"\" Compute loss\n",
    "    \"\"\"\n",
    "    # t ~ Unif([0, 1])\n",
    "    t = (torch.rand(1, device=x_1.device) + torch.arange(len(x_1), device=x_1.device) / len(x_1)) % (1 - self.eps)\n",
    "    t = t[:, None].expand(x_1.shape)\n",
    "    # x ~ p_t(x_0)\n",
    "    x_0 = torch.randn_like(x_1) \n",
    "    v_psi = v_t(t[:,0], self.psi_t(x_0, x_1, t))\n",
    "    d_psi = x_1 - (1 - self.sig_min) * x_0\n",
    "    return torch.mean((v_psi - d_psi) ** 2)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "Ugd9xR1ugsWp"
   },
   "source": [
    "## Diffusion conditional Vector Fields:\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "jo5mwRlDhS-A"
   },
   "source": [
    "### Variance Preserving diffusion:\n",
    "In this section, we propose an implementation of the _Conditional Vector Fields_ in the case of _Variance Preserving_ (VP) _Diffusion_."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "y2kYl26xkdls"
   },
   "outputs": [],
   "source": [
    "class VPDiffusionFlowMatching:\n",
    "\n",
    "  def __init__(self) -> None:\n",
    "    super().__init__()\n",
    "    self.beta_min = 0.1\n",
    "    self.beta_max = 20.0\n",
    "    self.eps = 1e-5"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "XXkDq9XQWqFS"
   },
   "source": [
    "We need to implement the $\\mu_t$ and $\\sigma_t$ functions which define the following flow:\n",
    "$$\n",
    " \\psi_t(x) = \\sigma_t(x_1)x + \\mu_t(x_1)\n",
    "$$\n",
    "These functions rely on the $\\alpha_t$ and $\\beta_t$ schedules [4]."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "EL-BuPNgkh5M"
   },
   "outputs": [],
   "source": [
    "  def T(self, s: torch.Tensor) -> torch.Tensor:\n",
    "    return self.beta_min * s + 0.5 * (s ** 2) * (self.beta_max - self.beta_min)\n",
    "\n",
    "  def beta(self, t: torch.Tensor) -> torch.Tensor:\n",
    "    return self.beta_min + t*(self.beta_max - self.beta_min)\n",
    "\n",
    "  def alpha(self, t: torch.Tensor) -> torch.Tensor:\n",
    "    return torch.exp(-0.5 * self.T(t))\n",
    "\n",
    "  def mu_t(self, t: torch.Tensor, x_1: torch.Tensor) -> torch.Tensor:\n",
    "    return self.alpha(1. - t) * x_1\n",
    "\n",
    "  def sigma_t(self, t: torch.Tensor, x_1: torch.Tensor) -> torch.Tensor:\n",
    "    return torch.sqrt(1. - self.alpha(1. - t) ** 2)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "ckoNCAd_0iSV"
   },
   "source": [
    "In the case of _Variance Preserving_ we have a closed-form equation for the _conditional Vector Field_:\n",
    "$$\n",
    " u_t(x|x_1) = - \\frac{T'(1-t)}{2}\\left[ \\frac{e^{-T(1-t)}x - e^{-\\frac{1}{2}T(1-t)}x_1}{1 - e^{-T(1-t)}}\\right] \\tag{19}\n",
    "$$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 182,
   "metadata": {
    "id": "QnuyFe4NydZT"
   },
   "outputs": [],
   "source": [
    "  def u_t(self, t: torch.Tensor, x: torch.Tensor, x_1: torch.Tensor) -> torch.Tensor:\n",
    "    num = torch.exp(-self.T(1. - t)) * x - torch.exp(-0.5 * self.T(1.-t))* x_1\n",
    "    denum = 1. - torch.exp(- self.T(1. - t))\n",
    "    return - 0.5 * self.beta(1. - t) * (num/denum)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "QIOww1YH42Gk"
   },
   "source": [
    "Hence, we can compute the regression loss on the conditional vector fields.\n",
    "$$\n",
    "\\mathcal{L}_{CFM}(\\theta) = \\mathbb{E}_{t, q(x_1), p_t(x|x_1)} \\lVert v_t(x) - u_t(x|x_1) \\rVert^2 \\tag{9}\n",
    "$$ \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 183,
   "metadata": {
    "id": "stJnvihOkndc"
   },
   "outputs": [],
   "source": [
    "  def loss(self, v_t: nn.Module, x_1: torch.Tensor) -> torch.Tensor:\n",
    "    \"\"\" Compute loss\n",
    "    \"\"\" \n",
    "    # t ~ Unif([0, 1])\n",
    "    t = (torch.rand(1, device=x_1.device) + torch.arange(len(x_1), device=x_1.device) / len(x_1)) % (1 - self.eps)\n",
    "    t = t[:, None].expand(x_1.shape)\n",
    "    # x ~ p_t(x|x_1)\n",
    "    x = self.mu_t(t, x_1) + self.sigma_t(t, x_1) * torch.randn_like(x_1)\n",
    "\n",
    "    return torch.mean((v_t(t[:,0], x) - self.u_t(t, x, x_1)) ** 2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 184,
   "metadata": {
    "cellView": "form",
    "id": "XZP48U0N4PRq"
   },
   "outputs": [],
   "source": [
    "#@title ⏳ Summary: please run this cell which contains the ```VPDiffusionFlowMatching``` class\n",
    "\n",
    "class VPDiffusionFlowMatching:\n",
    "\n",
    "  def __init__(self) -> None:\n",
    "    super().__init__()\n",
    "    self.beta_min = 0.1\n",
    "    self.beta_max = 20.0\n",
    "    self.eps = 1e-5\n",
    "\n",
    "  def T(self, s: torch.Tensor) -> torch.Tensor:\n",
    "\n",
    "    return self.beta_min * s + 0.5 * (s ** 2) * (self.beta_max - self.beta_min)\n",
    "\n",
    "  def beta(self, t: torch.Tensor) -> torch.Tensor:\n",
    "    \n",
    "    return self.beta_min + t*(self.beta_max - self.beta_min)\n",
    "\n",
    "  def alpha(self, t: torch.Tensor) -> torch.Tensor:\n",
    "\n",
    "    return torch.exp(-0.5 * self.T(t))\n",
    "\n",
    "  def mu_t(self, t: torch.Tensor, x_1: torch.Tensor) -> torch.Tensor:\n",
    "\n",
    "    return self.alpha(1. - t) * x_1\n",
    "\n",
    "  def sigma_t(self, t: torch.Tensor, x_1: torch.Tensor) -> torch.Tensor:\n",
    "\n",
    "    return torch.sqrt(1. - self.alpha(1. - t) ** 2)\n",
    "\n",
    "  def u_t(self, t: torch.Tensor, x: torch.Tensor, x_1: torch.Tensor) -> torch.Tensor:\n",
    "\n",
    "    num = torch.exp(-self.T(1. - t)) * x - torch.exp(-0.5 * self.T(1.-t))* x_1\n",
    "    denum = 1. - torch.exp(- self.T(1. - t))\n",
    "    return - 0.5 * self.beta(1. - t) * (num/denum)\n",
    "\n",
    "\n",
    "  def loss(self, v_t: nn.Module, x_1: torch.Tensor) -> torch.Tensor:\n",
    "    \"\"\" Compute loss\n",
    "    \"\"\" \n",
    "    # t ~ Unif([0, 1])\n",
    "    t = (torch.rand(1, device=x_1.device) + torch.arange(len(x_1), device=x_1.device) / len(x_1)) % (1 - self.eps)\n",
    "    t = t[:, None].expand(x_1.shape)\n",
    "    # x ~ p_t(x|x_1)\n",
    "    x = self.mu_t(t, x_1) + self.sigma_t(t, x_1) * torch.randn_like(x_1)\n",
    "\n",
    "    return torch.mean((v_t(t[:,0], x) - self.u_t(t, x, x_1)) ** 2)\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "qVPUlqsvhK3h"
   },
   "source": [
    "### Variance Exploding Diffusion:\n",
    "\n",
    "In this section, we propose an implementation of the _Conditional Vector Fields_ in the case of _Variance Exploding_ (VE) _Diffusion_."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 185,
   "metadata": {
    "id": "avX9dBj6lueq"
   },
   "outputs": [],
   "source": [
    "class VEDiffusionFlowMatching:\n",
    "\n",
    "  def __init__(self) -> None:\n",
    "    super().__init__()\n",
    "    self.sigma_min = 0.01\n",
    "    self.sigma_max = 2. \n",
    "    self.eps = 1e-5"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "SZE3q7gG5OLj"
   },
   "source": [
    "In the case of _Variance Exploding_ we have a closed-form equation for the _conditional Vector Field_ that rely on $\\sigma_t$ and $\\sigma_t'$:\n",
    "$$\n",
    "u(x| x_1) = - \\frac{\\sigma_{1-t}'}{\\sigma_{1-t}}(x - x_1) \\tag{17}\n",
    "$$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 186,
   "metadata": {
    "id": "o0VhxVjQlyPE"
   },
   "outputs": [],
   "source": [
    "  def sigma_t(self, t: torch.Tensor) -> torch.Tensor:\n",
    "    return self.sigma_min * (self.sigma_max / self.sigma_min) ** t\n",
    "\n",
    "  def dsigma_dt(self, t: torch.Tensor) -> torch.Tensor:\n",
    "    return self.sigma_t(t) * torch.log(torch.tensor(self.sigma_max/self.sigma_min))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 187,
   "metadata": {
    "id": "jV48_XbV20uk"
   },
   "outputs": [],
   "source": [
    "  def u_t(self, t: torch.Tensor, x: torch.Tensor, x_1: torch.Tensor) -> torch.Tensor:\n",
    "    return -(self.dsigma_dt(1. - t) / self.sigma_t(1. - t)) * (x - x_1)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "eD41NIGy5K2u"
   },
   "source": [
    "As for VP Diffusion, we can directly compute the regression loss on the conditional vector fields.\n",
    "$$\n",
    "\\mathcal{L}_{CFM}(\\theta) = \\mathbb{E}_{t, q(x_1), p_t(x|x_1)} \\lVert v_t(x) - u_t(x|x_1) \\rVert^2 \\tag{9}\n",
    "$$ "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 188,
   "metadata": {
    "id": "PG4NRb1Nl2pE"
   },
   "outputs": [],
   "source": [
    "def loss(self, v_t: nn.Module, x_1: torch.Tensor) -> torch.Tensor:\n",
    "    \"\"\" Compute loss\n",
    "    \"\"\" \n",
    "    # t ~ Unif([0, 1])\n",
    "    t = (torch.rand(1, device=x_1.device) + torch.arange(len(x_1), device=x_1.device) / len(x_1)) % (1 - self.eps)\n",
    "    t = t[:, None].expand(x_1.shape)\n",
    "    # x ~ p_t(x|x_1)\n",
    "    x = x_1 + self.sigma_t(1. - t) * torch.randn_like(x_1)\n",
    "\n",
    "    return torch.mean((v_t(t[:,0], x) - self.u_t(t, x, x_1)) ** 2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 189,
   "metadata": {
    "cellView": "form",
    "id": "z9H9cihc63Jl"
   },
   "outputs": [],
   "source": [
    "#@title ⏳ Summary: please run this cell which contains the ```VEDiffusionFlowMatching``` class\n",
    "class VEDiffusionFlowMatching:\n",
    "\n",
    "  def __init__(self) -> None:\n",
    "    super().__init__()\n",
    "    self.sigma_min = 0.01\n",
    "    self.sigma_max = 2.\n",
    "    self.eps = 1e-5\n",
    "\n",
    "\n",
    "  def sigma_t(self, t: torch.Tensor) -> torch.Tensor:\n",
    "\n",
    "    return self.sigma_min * (self.sigma_max / self.sigma_min) ** t\n",
    "\n",
    "  def dsigma_dt(self, t: torch.Tensor) -> torch.Tensor:\n",
    "\n",
    "    return self.sigma_t(t) * torch.log(torch.tensor(self.sigma_max/self.sigma_min))\n",
    "\n",
    "  def u_t(self, t: torch.Tensor, x: torch.Tensor, x_1: torch.Tensor) -> torch.Tensor:\n",
    "\n",
    "    return -(self.dsigma_dt(1. - t) / self.sigma_t(1. - t)) * (x - x_1)\n",
    "\n",
    "  def loss(self, v_t: nn.Module, x_1: torch.Tensor) -> torch.Tensor:\n",
    "    \"\"\" Compute loss\n",
    "    \"\"\" \n",
    "    # t ~ Unif([0, 1])\n",
    "    t = (torch.rand(1, device=x_1.device) + torch.arange(len(x_1), device=x_1.device) / len(x_1)) % (1 - self.eps)\n",
    "    t = t[:, None].expand(x_1.shape)\n",
    "    # x ~ p_t(x|x_1)\n",
    "    x = x_1 + self.sigma_t(1. - t) * torch.randn_like(x_1)\n",
    "\n",
    "    return torch.mean((v_t(t[:,0], x) - self.u_t(t, x, x_1)) ** 2)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "qoZPnVR100VV"
   },
   "source": [
    "## Conditional Vector Field:\n",
    "\n",
    "We add the ```CondVF```  class which aims to model the _Optimal Transport Conditional Vector Field_ $v_t$. This approximation $v_{t}(\\;⋅\\;;\\theta)$ of the true $v_t$ will be optimized later in this notebook by tuning the parameters $\\theta$ of ```CondVF.net```."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 190,
   "metadata": {
    "id": "RHVJ1cNz1T2X"
   },
   "outputs": [],
   "source": [
    "class CondVF(nn.Module):\n",
    "  def __init__(self, net: nn.Module) -> None:\n",
    "    super().__init__()\n",
    "    self.net = net"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 191,
   "metadata": {
    "id": "sgsc7aPk1f7S"
   },
   "outputs": [],
   "source": [
    "  def forward(self, t: torch.Tensor, x: torch.Tensor) -> torch.Tensor:\n",
    "    return self.net(t, x)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "iR3EgmPhDjpN"
   },
   "source": [
    "Furthermore, we define the ```encode``` and ```decode``` functions:\n",
    "- ```encode``` maps the data distribtion $q(x_1)\\approx p_1(x_1)$ to the prior distribution $p(x_0)$.\n",
    "- ```decode``` maps the prior distribution $p(x_0)$ to the data distribtion $p_1(x_1) \\approx q(x_1)$.\n",
    "\n",
    "The article mentions that a flow $\\phi_t$ and its corresponding vector field $v_t$ verify the following equations:\n",
    "$$\n",
    "\\begin{cases}\n",
    "  \\frac{d}{dt}\\phi_t(x) = v_t(\\phi_t(x))\\\\ \n",
    "  \\phi_0(x) = x \\tag{1, 2}\\\\\n",
    "\\end{cases}\n",
    "$$\n",
    "\n",
    "Thus, to ```encode``` we integrate the first equation from $t=1$ to $t=0$ and the ```decode``` function is the same but going from $t=0$ to $t=1$. Following François Rozet's implementation and side notes [2], we use the function ```odeint``` from the _Zuko_ package [3]. We build a small ```wrapper``` which only aims at providing time tensor with the correct shapes to the neural network ```net``` when using ```odeint```."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 192,
   "metadata": {
    "id": "8Oz-xPWr1iJw"
   },
   "outputs": [],
   "source": [
    "  def wrapper(self, t: torch.Tensor, x: torch.Tensor):\n",
    "      t = t * torch.ones(len(x))\n",
    "      return self(t, x)\n",
    "\n",
    "  def encode(self, x_1: torch.Tensor) -> torch.Tensor:\n",
    "    return odeint(self.wrapper, x_1, 1., 0., self.parameters())\n",
    "\n",
    "  def decode(self, x_0: torch.Tensor) -> Tensor:\n",
    "    return odeint(self.wrapper, x_0, 0., 1., self.parameters())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 193,
   "metadata": {
    "cellView": "form",
    "id": "RqH-PUY6UCBR"
   },
   "outputs": [],
   "source": [
    "#@title ⏳ Summary: please run this cell which contains the ```CondVF``` class.\n",
    "class CondVF(nn.Module):\n",
    "  def __init__(self, net: nn.Module, n_steps: int = 100) -> None:\n",
    "    super().__init__()\n",
    "    self.net = net\n",
    "\n",
    "  def forward(self, t: torch.Tensor, x: torch.Tensor) -> torch.Tensor:\n",
    "    return self.net(t, x)\n",
    "    \n",
    "  def wrapper(self, t: torch.Tensor, x: torch.Tensor) -> torch.Tensor:\n",
    "      t = t * torch.ones(len(x), device=x.device)\n",
    "      return self(t, x)\n",
    "\n",
    "  def decode_t0_t1(self, x_0, t0, t1):\n",
    "    return odeint(self.wrapper, x_0, t0, t1, self.parameters())\n",
    "\n",
    "\n",
    "  def encode(self, x_1: torch.Tensor) -> torch.Tensor:\n",
    "    return odeint(self.wrapper, x_1, 1., 0., self.parameters())\n",
    "\n",
    "  def decode(self, x_0: torch.Tensor) -> torch.Tensor:\n",
    "    return odeint(self.wrapper, x_0, 0., 1., self.parameters())"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "NlDlHJya188e"
   },
   "source": [
    "## Neural Network for VF approximation:"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "MyIp8CfWbOoJ"
   },
   "source": [
    "We want to approximate $v_t$ with a parametric model $v_{t}(\\;\\cdot\\;;\\theta)$. Since we deal with point coordinates and want to keep the neural network simple for this example, we use a simple _Multilayer Perceptron_ (MLP)."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 194,
   "metadata": {
    "id": "8S1Pl8IH5OfY"
   },
   "outputs": [],
   "source": [
    "class Net(nn.Module):\n",
    "  def __init__(self, in_dim: int, out_dim: int, h_dims: List[int], n_frequencies: int) -> None:\n",
    "    super().__init__()\n",
    "\n",
    "    ins = [in_dim + 2 * n_frequencies] + h_dims\n",
    "    outs = h_dims + [out_dim]\n",
    "    self.n_frequencies = n_frequencies\n",
    "\n",
    "    self.layers = nn.ModuleList([\n",
    "        nn.Sequential(nn.Linear(in_d, out_d), nn.LeakyReLU()) for in_d, out_d in zip(ins, outs)\n",
    "    ])\n",
    "    self.top = nn.Sequential(nn.Linear(out_dim, out_dim))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "YPte-sZAdxCH"
   },
   "source": [
    " However the vector field $v$ is conditioned on time $t$. Thus, we create a _sinusoidal positional encoding_ of $t$ thanks to sine and cosine functions to have a time representation that make more sense than only its value between $0$ and $1$."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 195,
   "metadata": {
    "id": "1tca5p1H5R3x"
   },
   "outputs": [],
   "source": [
    "  def time_encoder(self, t: torch.Tensor) -> torch.Tensor:\n",
    "    freq = 2 * torch.arange(self.n_frequencies) * torch.pi\n",
    "    t = freq * t[..., None]\n",
    "    return torch.cat((t.cos(), t.sin()), dim=-1)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "gO6ZkwyTgGCu"
   },
   "source": [
    "When computing $v_t(x)$ we simply concatenate the _sinusoidale positional encoding_ of $t$ with $x$ (input coordinates), and then process the obtained tensor with the MLP's layers."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 196,
   "metadata": {
    "id": "yjML0fFn5UIm"
   },
   "outputs": [],
   "source": [
    "  def forward(self, t: torch.Tensor, x: torch.Tensor) -> torch.Tensor:\n",
    "    t = self.time_encoder(t)\n",
    "    x = torch.cat((x, t), dim=-1)\n",
    "    for l in self.layers:\n",
    "      x = l(x)\n",
    "    return self.top(x) "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 197,
   "metadata": {
    "id": "BA-LgtoEeN4a"
   },
   "outputs": [],
   "source": [
    "#@title ⏳ Summary: please run this cell which contains the ```Net``` class.\n",
    "class Net(nn.Module):\n",
    "  def __init__(self, in_dim: int, out_dim: int, h_dims: List[int], n_frequencies:int) -> None:\n",
    "    super().__init__()\n",
    "\n",
    "    ins = [in_dim + 2 * n_frequencies] + h_dims\n",
    "    outs = h_dims + [out_dim]\n",
    "    self.n_frequencies = n_frequencies\n",
    "\n",
    "    self.layers = nn.ModuleList([\n",
    "        nn.Sequential(nn.Linear(in_d, out_d), nn.LeakyReLU()) for in_d, out_d in zip(ins, outs)\n",
    "    ])\n",
    "    self.top = nn.Sequential(nn.Linear(out_dim, out_dim))\n",
    "  \n",
    "  def time_encoder(self, t: torch.Tensor) -> torch.Tensor:\n",
    "    freq = 2 * torch.arange(self.n_frequencies, device=t.device) * torch.pi\n",
    "    t = freq * t[..., None]\n",
    "    return torch.cat((t.cos(), t.sin()), dim=-1)\n",
    "    \n",
    "  def forward(self, t: torch.Tensor, x: torch.Tensor) -> torch.Tensor:\n",
    "    t = self.time_encoder(t)\n",
    "    x = torch.cat((x, t), dim=-1)\n",
    "    \n",
    "    for l in self.layers:\n",
    "      x = l(x)\n",
    "    return self.top(x)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "iyrny-eMIdJK"
   },
   "source": [
    "## Create dataset\n",
    "\n",
    "For this notebook we consider a 2D-dataset and we choose a standard dataset: the _half-moon dataset_. This defines the distribution $q(x_1)$ over the  $\\mathbb{R}^2$ data space, _i.e._ the data points are described by two coordinates $x = (a, b)$. We sample a large number ```n_points``` of data points from $q(x_1)$ to build the dataset and we plot them. In addition, in this work we consider a simple gaussian _prior distribution_:\n",
    "$$\n",
    "p(x) = \\mathcal{N}(x|0, I)\n",
    "$$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 198,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 265
    },
    "id": "bWzZjcEzHheU",
    "outputId": "23e52adb-04f2-40ce-93eb-c6f0cd24d77f"
   },
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "def get_data(dataset: str, n_points: int) -> np.ndarray:\n",
    "  if dataset == \"moons\":\n",
    "    data, _ = make_moons(n_points, noise=0.15)\n",
    "  elif dataset == \"swiss\":\n",
    "    data, _ = make_swiss_roll(n_points, noise=0.25)\n",
    "    data = data[:, [0, 2]] / 10.0\n",
    "  return StandardScaler().fit_transform(data)\n",
    "\n",
    "\n",
    "n_points = 10_000\n",
    "DATASET = \"swiss\"\n",
    "data = get_data(DATASET, n_points)\n",
    "\n",
    "%matplotlib inline\n",
    "plt.hist2d(data[:, 0], data[:, 1], bins=128)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "8dBLSaZ4rOKn"
   },
   "source": [
    "We then transform the data points into a ```pytorch.utils.data.DataLoader``` to facilitate data handling. We set the ```batch_size``` to $512$."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 199,
   "metadata": {
    "id": "EPCtsiJcHmFr"
   },
   "outputs": [],
   "source": [
    "batch_size = 2048\n",
    "dataset = torch.from_numpy(data).float()\n",
    "dataset = dataset.to(device)\n",
    "dataset = TensorDataset(dataset) \n",
    "dataloader = DataLoader(dataset, batch_size=batch_size)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "PzQXJmYJ6jIf"
   },
   "source": [
    "## Training:"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "dmWvMupS0DlH"
   },
   "source": [
    "We train the neural network using the ```AdamW``` optimizer (Adam optimizer with _weight decay_) with a _learning rate_ $lr = 10^{-3}$ during $N = 500$ epochs:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 200,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "rd83qQF8xodk",
    "outputId": "80b44a05-58a1-4884-bbb2-f54264fb50ce"
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████████| 5000/5000 [05:51<00:00, 14.22it/s]\n"
     ]
    }
   ],
   "source": [
    "def get_model(name: str):\n",
    "    if name == \"vp\":\n",
    "      return VPDiffusionFlowMatching()\n",
    "    elif name == \"ve\":\n",
    "      return VEDiffusionFlowMatching()\n",
    "    if name == \"ot\":\n",
    "      return OTFlowMatching()\n",
    "\n",
    "MODEL = \"ot\"\n",
    "model = get_model(MODEL)\n",
    "net = Net(2, 2, [512]*5, 10).to(device)\n",
    "v_t = CondVF(net)    \n",
    "\n",
    "losses = [] \n",
    "# configure optimizer\n",
    "optimizer = torch.optim.Adam(v_t.parameters(), lr=1e-3)\n",
    "n_epochs = 5000\n",
    "    \n",
    "for epoch in tqdm(range(n_epochs), ncols=88):\n",
    "    for batch in dataloader:\n",
    "      x_1 = batch[0]\n",
    "      # compute loss \n",
    "      loss = model.loss(v_t, x_1)\n",
    "      optimizer.zero_grad()\n",
    "      loss.backward()\n",
    "      optimizer.step()\n",
    "      losses += [loss.detach()]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "azqtOY0S8Cr9"
   },
   "source": [
    "## Sampling:"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "g7yvvByh33b4"
   },
   "source": [
    "To sample $\\hat{x}_{1} \\sim q(x_1)$ we first sample $x_0 \\sim p(x)$, _i.e._ we draw samples from the _prior distribution_, and then apply the ```decode``` function from the ```CondVF``` class. This step may take several minutes depending on the ```n_samples``` chosen."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 201,
   "metadata": {
    "id": "4VEyOyy91Wju"
   },
   "outputs": [],
   "source": [
    "# Sampling\n",
    "n_samples = 10_000\n",
    "with torch.no_grad():\n",
    "    x_0 = torch.randn(n_samples, 2, device=device)\n",
    "    x_1_hat = v_t.decode(x_0)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "5hsDf_TD6J_j"
   },
   "source": [
    "Last, we plot the samples $\\hat{x}_1$ distribution:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 202,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 266
    },
    "id": "l_NPXHeSNg8Y",
    "outputId": "b4e43284-8caf-46a9-855c-4f2829db4130"
   },
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "%matplotlib inline\n",
    "import matplotlib.pyplot as plt\n",
    "x_1_hat = x_1_hat.cpu().numpy()\n",
    "plt.hist2d(x_1_hat[:, 0], x_1_hat[:, 1], bins=164)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 203,
   "metadata": {
    "id": "3TW8GRXHRnji"
   },
   "outputs": [],
   "source": [
    "# Sampling\n",
    "N_SAMPLES = 10_000\n",
    "N_STEPS = 100\n",
    "t_steps = torch.linspace(0, 1, N_STEPS, device=device)\n",
    "with torch.no_grad():\n",
    "    x_t = [torch.randn(n_samples, 2, device=device)]\n",
    "    for t in range(len(t_steps)-1):\n",
    "      x_t += [v_t.decode_t0_t1(x_t[-1], t_steps[t], t_steps[t+1])]\n",
    "\n",
    "# pad predictions\n",
    "x_t = [x_t[0]]*10 + x_t + [x_t[-1]] * 10\n",
    "\n",
    "x_t_numpy = np.array([x.detach().cpu().numpy() for x in x_t])\n",
    "filename = f\"{DATASET}_{MODEL}_{N_SAMPLES}_{N_STEPS}.npy\"\n",
    "np.save(filename, x_t_numpy)\n"
   ]
  }
 ],
 "metadata": {
  "accelerator": "GPU",
  "colab": {
   "authorship_tag": "ABX9TyO8IoWngZmvSHlYJWKw+kch",
   "include_colab_link": true,
   "provenance": []
  },
  "gpuClass": "standard",
  "kernelspec": {
   "display_name": "Python (ctgan)",
   "language": "python",
   "name": "ctgan"
  },
  "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.9.21"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
