{
  "nbformat": 4,
  "nbformat_minor": 0,
  "metadata": {
    "colab": {
      "provenance": [],
      "gpuType": "T4"
    },
    "kernelspec": {
      "name": "python3",
      "display_name": "Python 3"
    },
    "language_info": {
      "name": "python"
    },
    "accelerator": "GPU"
  },
  "cells": [
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "uaI3Lx79Rhtp",
        "outputId": "756340a8-505d-41aa-ca12-513b1dd26812"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Collecting pot\n",
            "  Downloading POT-0.9.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl (789 kB)\n",
            "\u001b[?25l     \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m0.0/790.0 kB\u001b[0m \u001b[31m?\u001b[0m eta \u001b[36m-:--:--\u001b[0m\r\u001b[2K     \u001b[91m━━━━━━\u001b[0m\u001b[90m╺\u001b[0m\u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m122.9/790.0 kB\u001b[0m \u001b[31m3.5 MB/s\u001b[0m eta \u001b[36m0:00:01\u001b[0m\r\u001b[2K     \u001b[91m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m\u001b[91m╸\u001b[0m \u001b[32m788.5/790.0 kB\u001b[0m \u001b[31m13.0 MB/s\u001b[0m eta \u001b[36m0:00:01\u001b[0m\r\u001b[2K     \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m790.0/790.0 kB\u001b[0m \u001b[31m11.1 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
            "\u001b[?25hRequirement already satisfied: numpy>=1.16 in /usr/local/lib/python3.10/dist-packages (from pot) (1.23.5)\n",
            "Requirement already satisfied: scipy>=1.0 in /usr/local/lib/python3.10/dist-packages (from pot) (1.11.2)\n",
            "Installing collected packages: pot\n",
            "Successfully installed pot-0.9.1\n"
          ]
        }
      ],
      "source": [
        "!pip install pot"
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "import sys\n",
        "import torch\n",
        "import torch.nn as nn\n",
        "import torch.optim as optim\n",
        "from torch.nn import functional as F\n",
        "from torch.autograd import Variable\n",
        "import numpy as np;\n",
        "import matplotlib.pyplot as plt\n",
        "from matplotlib.pyplot import figure\n",
        "from matplotlib.colors import LinearSegmentedColormap\n",
        "import math\n",
        "import seaborn as sns\n",
        "import ot\n",
        "from mpl_toolkits.axes_grid1 import AxesGrid\n",
        "\n",
        "device = torch.device(\"cuda:0\" if torch.cuda.is_available() else \"cpu\")"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "hVI-XADgRyul",
        "outputId": "60caa4b6-ed03-467d-c0c0-778b24a55048"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "/usr/local/lib/python3.10/dist-packages/ot/backend.py:1368: UserWarning: Explicitly requested dtype <class 'jax.numpy.float64'> requested in array is not available, and will be truncated to dtype float32. To enable more dtypes, set the jax_enable_x64 configuration option or the JAX_ENABLE_X64 shell environment variable. See https://github.com/google/jax#current-gotchas for more.\n",
            "  jax.device_put(jnp.array(1, dtype=jnp.float64), d)\n",
            "/usr/local/lib/python3.10/dist-packages/ot/backend.py:2998: UserWarning: To use TensorflowBackend, you need to activate the tensorflow numpy API. You can activate it by running: \n",
            "from tensorflow.python.ops.numpy_ops import np_config\n",
            "np_config.enable_numpy_behavior()\n",
            "  register_backend(TensorflowBackend())\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# Create neural network architectures\n",
        "\n",
        "# Define width of neural networks\n",
        "m = 150\n",
        "m2 = 100\n",
        "\n",
        "# Create branch network\n",
        "class Branch(nn.Module):\n",
        "    def __init__(self):\n",
        "        super(Branch, self).__init__()\n",
        "        # Modify neural network input dimension based on mesh size n (currently 100)\n",
        "        self.hidden_U = nn.Linear(100,m, bias=True)\n",
        "        self.hidden_V = nn.Linear(100,m, bias=True)\n",
        "        self.hidden_H_1 = nn.Linear(100,m, bias=True); self.hidden_Z_1 = nn.Linear(m,m, bias=True)\n",
        "        self.hidden_Z_2 = nn.Linear(m,m, bias=True); self.hidden_Z_3 = nn.Linear(m,m, bias=True)\n",
        "        self.hidden_Z_4 = nn.Linear(m,m, bias=True); self.hidden_Z_5 = nn.Linear(m,m, bias=True)\n",
        "        self.hidden_Z_6 = nn.Linear(m,m, bias=True); self.hidden_Z_7 = nn.Linear(m,800, bias=True)\n",
        "\n",
        "\n",
        "    def forward(self, u):\n",
        "        U = F.tanh(self.hidden_U(u)); V = F.tanh(self.hidden_V(u)); H_1 = F.tanh(self.hidden_H_1(u));\n",
        "        Z_1 = F.tanh(self.hidden_Z_1(H_1)); H_2 = (1 - Z_1)*U + Z_1*V\n",
        "        Z_2 = F.tanh(self.hidden_Z_2(H_2)); H_3 = (1 - Z_2)*U + Z_2*V\n",
        "        Z_3 = F.tanh(self.hidden_Z_3(H_3)); H_4 = (1 - Z_3)*U + Z_3*V\n",
        "        Z_4 = F.tanh(self.hidden_Z_4(H_4)); H_5 = (1 - Z_4)*U + Z_4*V\n",
        "        Z_5 = F.tanh(self.hidden_Z_5(H_5)); H_6 = (1 - Z_5)*U + Z_5*V\n",
        "        Z_6 = F.tanh(self.hidden_Z_6(H_6)); H_7 = (1 - Z_6)*U + Z_6*V\n",
        "        Z_7 = self.hidden_Z_7(H_7)\n",
        "        output = Z_7\n",
        "        return output\n",
        "\n",
        "\n",
        "# Create trunk network\n",
        "class Trunk(nn.Module):\n",
        "    def __init__(self):\n",
        "        super(Trunk, self).__init__()\n",
        "        self.hidden_U = nn.Linear(2,m2, bias=True)\n",
        "        self.hidden_V = nn.Linear(2,m2, bias=True)\n",
        "        self.hidden_H_1 = nn.Linear(2,m2, bias=True); self.hidden_Z_1 = nn.Linear(m2,m2, bias=True)\n",
        "        self.hidden_Z_2 = nn.Linear(m2,m2, bias=True); self.hidden_Z_3 = nn.Linear(m2,m2, bias=True)\n",
        "        self.hidden_Z_4 = nn.Linear(m2,m2, bias=True); self.hidden_Z_5 = nn.Linear(m2,m2, bias=True)\n",
        "        self.hidden_Z_6 = nn.Linear(m2,m2, bias=True); self.hidden_Z_7 = nn.Linear(m2,800, bias=True)\n",
        "\n",
        "\n",
        "    def forward(self, x,t):\n",
        "        z = torch.cat((x,t),1)\n",
        "        U = F.tanh(self.hidden_U(z)); V = F.tanh(self.hidden_V(z)); H_1 = F.tanh(self.hidden_H_1(z));\n",
        "        Z_1 = F.tanh(self.hidden_Z_1(H_1)); H_2 = (1 - Z_1)*U + Z_1*V\n",
        "        Z_2 = F.tanh(self.hidden_Z_2(H_2)); H_3 = (1 - Z_2)*U + Z_2*V\n",
        "        Z_3 = F.tanh(self.hidden_Z_3(H_3)); H_4 = (1 - Z_3)*U + Z_3*V\n",
        "        Z_4 = F.tanh(self.hidden_Z_4(H_4)); H_5 = (1 - Z_4)*U + Z_4*V\n",
        "        Z_5 = F.tanh(self.hidden_Z_5(H_5)); H_6 = (1 - Z_5)*U + Z_5*V\n",
        "        Z_6 = F.tanh(self.hidden_Z_6(H_6)); H_7 = (1 - Z_6)*U + Z_6*V\n",
        "        Z_7 = self.hidden_Z_7(H_7)\n",
        "        output = Z_7\n",
        "        return output"
      ],
      "metadata": {
        "id": "iSXnmnkbRy2i"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# Create neural networks\n",
        "cty_branch0 = Branch().to(device)\n",
        "cty_branch1 = Branch().to(device)\n",
        "cty_trunk = Trunk().to(device)\n",
        "\n",
        "HJ_branch0 = Branch().to(device)\n",
        "HJ_branch1 = Branch().to(device)\n",
        "HJ_trunk = Trunk().to(device)"
      ],
      "metadata": {
        "id": "pWfPqhQJRy6U"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# Load in pre-trained neural networks from current directory\n",
        "\n",
        "File = \"Cty_branch0.pth\"\n",
        "cty_branch0.load_state_dict(torch.load(File))\n",
        "\n",
        "File = \"Cty_branch1.pth\"\n",
        "cty_branch1.load_state_dict(torch.load(File))\n",
        "\n",
        "File = \"Cty_trunk.pth\"\n",
        "cty_trunk.load_state_dict(torch.load(File))\n",
        "\n",
        "File = \"HJ_branch0.pth\"\n",
        "HJ_branch0.load_state_dict(torch.load(File))\n",
        "\n",
        "File = \"HJ_branch1.pth\"\n",
        "HJ_branch1.load_state_dict(torch.load(File))\n",
        "\n",
        "File = \"HJ_trunk.pth\"\n",
        "HJ_trunk.load_state_dict(torch.load(File))"
      ],
      "metadata": {
        "id": "VLxby2kGUsxZ"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# Define loss functions\n",
        "mse_cost_function = torch.nn.MSELoss()\n",
        "l1_cost_function = torch.nn.L1Loss()"
      ],
      "metadata": {
        "id": "BfylRabtRy-o"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# Create optimizer. Choose initial learning rate at 5e-4 and lower periodically\n",
        "optimizer = optim.Adam([\n",
        "    {'params': cty_branch0.parameters()},\n",
        "    {'params': cty_branch1.parameters()},\n",
        "    {'params': cty_trunk.parameters()},\n",
        "    {'params': HJ_branch0.parameters()},\n",
        "    {'params': HJ_branch1.parameters()},\n",
        "    {'params': HJ_trunk.parameters()},\n",
        "], lr=5e-4)"
      ],
      "metadata": {
        "id": "4d33w5I2RzCm"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# Declare univariate normal density value function\n",
        "def h(x, mu, sigma):\n",
        "  return (1/(sigma*(2*math.pi)**(1/2)))*(math.e)**((-1/2)*((x-mu)**2)/sigma**2)"
      ],
      "metadata": {
        "id": "iPqXCl5KRzGI"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "n = 100 # Mesh size\n",
        "N = 5000 # Number of initial conditions\n",
        "x_min = 0\n",
        "x_max = 10\n",
        "k = 12  # Number of mixtures in u0 and u1 combined\n",
        "c = 1/6 # Constant for each Gaussian in the mixtures\n",
        "\n",
        "X = torch.arange(x_min, x_max, (x_max-x_min)/n).to(device)\n",
        "X_vec = torch.unsqueeze(X,1).to(device)\n",
        "t0 = torch.zeros(size=(n,1)).to(device)\n",
        "t1 = torch.ones(size=(n,1)).to(device)\n",
        "\n",
        "# Create Gaussian mixture data\n",
        "u0 = torch.zeros(size=(N,n))\n",
        "u1 = torch.zeros(size=(N,n))\n",
        "for i in range(N):\n",
        "  Sigma = np.random.uniform(0.5,0.6,k)\n",
        "  means = np.random.uniform(2,8,k)\n",
        "  u0[i,:] = c*h(X, means[0], Sigma[0]) + c*h(X, means[1], Sigma[1]) + c*h(X, means[2], Sigma[2]) + \\\n",
        "            c*h(X, means[3], Sigma[3]) + c*h(X, means[4], Sigma[4]) + c*h(X, means[5], Sigma[5])\n",
        "  u1[i,:] = c*h(X, means[6], Sigma[6]) + c*h(X, means[7], Sigma[7]) + c*h(X, means[8], Sigma[8]) + \\\n",
        "            c*h(X, means[9], Sigma[9]) + c*h(X, means[10], Sigma[10]) + c*h(X, means[11], Sigma[11])\n",
        "\n",
        "\n",
        "u0 = u0.to(device)\n",
        "u1 = u1.to(device)"
      ],
      "metadata": {
        "id": "bw36fLfNRzKF"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# Plot Gaussian mixture sample\n",
        "I = np.random.randint(0,N,1).squeeze()\n",
        "plt.plot(u0[I,:].cpu().detach().numpy(), color='blue')\n",
        "plt.plot(u1[I,:].cpu().detach().numpy(), color='green')"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 447
        },
        "id": "lawJqz5LRzNe",
        "outputId": "91fc023c-4815-4407-e20b-1f8d596e3a22"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "[<matplotlib.lines.Line2D at 0x7c9e02e684f0>]"
            ]
          },
          "metadata": {},
          "execution_count": 16
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ],
            "image/png": "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\n"
          },
          "metadata": {}
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# Create physics-informed loss function for total loss minimization\n",
        "def physics_informed_loss(x, t, u0, u1, cty_branch0, cty_branch1, cty_trunk, HJ_branch0, HJ_branch1, HJ_trunk):\n",
        "\n",
        "  # Feed in data to continuity neural networks\n",
        "  cty_branch0_output = cty_branch0(u0)\n",
        "  cty_branch1_output = cty_branch1(u1)\n",
        "  cty_trunk_output = cty_trunk(x, t)\n",
        "\n",
        "  # Construct continuity DeepONet with its derivatives\n",
        "  cty = torch.sum( (cty_branch0_output * cty_branch1_output) * cty_trunk_output, dim=1)\n",
        "  cty_x = torch.autograd.grad(cty.sum(), x, create_graph=True)[0]\n",
        "  cty_t = torch.autograd.grad(cty.sum(), t, create_graph=True)[0]\n",
        "  cty = torch.unsqueeze(cty,1)\n",
        "\n",
        "  # Feed in data to Hamilton-Jacobi neural networks\n",
        "  HJ_branch0_output = HJ_branch0(u0)\n",
        "  HJ_branch1_output = HJ_branch1(u1)\n",
        "  HJ_trunk_output = HJ_trunk(x, t)\n",
        "\n",
        "  # Construct HJ DeepONet with its derivatives\n",
        "  HJ = torch.sum( (HJ_branch0_output * HJ_branch1_output) * HJ_trunk_output, dim=1)\n",
        "  HJ_x = torch.autograd.grad(HJ.sum(), x, create_graph=True)[0]\n",
        "  HJ_xx = torch.autograd.grad(HJ_x.sum(), x, create_graph=True)[0]\n",
        "  HJ_t = torch.autograd.grad(HJ.sum(), t, create_graph=True)[0]\n",
        "  HJ = torch.unsqueeze(HJ,1)\n",
        "\n",
        "  # Compute continuity and HJ residuals\n",
        "  cty_loss = (cty_t + cty_x*HJ_x + cty*HJ_xx)**2\n",
        "  HJ_loss = (0.5)*(HJ_t + (1/2)*(HJ_x**2))**2\n",
        "\n",
        "  # Return physics-informed loss\n",
        "  return(cty_loss + HJ_loss)"
      ],
      "metadata": {
        "id": "App1_8aKRzRS"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# Declare number of training iterations and batch size. Recommended to take batch size at least 1000\n",
        "num_iter = 100000\n",
        "batch_size = 2000\n",
        "\n",
        "# Declare a zero vector for evaluating loss terms\n",
        "T0 = torch.zeros(size=(batch_size,1)).to(device)\n",
        "\n",
        "# Training algorithm\n",
        "for i in range(num_iter):\n",
        "\n",
        "  # Indices for DeepONet evaluation\n",
        "  I = np.random.randint(0,N,batch_size)\n",
        "  J = np.random.randint(0,n,batch_size)\n",
        "\n",
        "  # Collocation procedure\n",
        "  x_unif = Variable(torch.from_numpy(np.random.uniform(x_min,x_max,size=(batch_size,1))).float(), requires_grad=True).to(device)\n",
        "  t_unif = Variable(torch.from_numpy(np.random.uniform(0,1,size=(batch_size,1))).float(), requires_grad=True).to(device)\n",
        "\n",
        "  # Evaluate continuity DeepONets for boundary conditions\n",
        "  cty_branch0_output = cty_branch0(u0[I,:])\n",
        "  cty_branch1_output = cty_branch1(u1[I,:])\n",
        "  cty_trunk_output_0 = cty_trunk(X_vec[J], t0[J])\n",
        "  cty_trunk_output_1 = cty_trunk(X_vec[J], t1[J])\n",
        "\n",
        "  # Boundary loss terms\n",
        "  inner_prod_0 = torch.sum( (cty_branch0_output * cty_branch1_output) * cty_trunk_output_0, dim=1)\n",
        "  inner_prod_1 = torch.sum( (cty_branch0_output * cty_branch1_output) * cty_trunk_output_1, dim=1)\n",
        "  boundary_loss_0 = mse_cost_function(inner_prod_0, u0[I,J])\n",
        "  boundary_loss_1 = mse_cost_function(inner_prod_1, u1[I,J])\n",
        "\n",
        "  # Physics-informed terms\n",
        "  physics_term = physics_informed_loss(x_unif, t_unif, u0[I,:], u1[I,:], cty_branch0, cty_branch1, cty_trunk, HJ_branch0, HJ_branch1, HJ_trunk)\n",
        "  physics_loss = l1_cost_function(physics_term, T0)\n",
        "\n",
        "  # Evaluate the loss\n",
        "  loss = 1*boundary_loss_0  + 1*boundary_loss_1  + 0.5*physics_loss\n",
        "\n",
        "\n",
        "  optimizer.zero_grad()\n",
        "  loss.backward()\n",
        "  optimizer.step()\n",
        "\n",
        "  if i % 100 == 0:\n",
        "    print(\"Training loss:\", '{:.4e}'.format(loss.data), \"- PINN loss:\", '{:.4e}'.format(physics_loss))"
      ],
      "metadata": {
        "id": "WZQCPBPoRzVo"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# Save newly-trained neural networks to current directory\n",
        "\n",
        "File = \"Cty_branch0.pth\"\n",
        "torch.save(cty_branch0.state_dict(), File)\n",
        "\n",
        "File = \"Cty_branch1.pth\"\n",
        "torch.save(cty_branch1.state_dict(), File)\n",
        "\n",
        "File = \"Cty_trunk.pth\"\n",
        "torch.save(cty_trunk.state_dict(), File)\n",
        "\n",
        "File = \"HJ_branch0.pth\"\n",
        "torch.save(HJ_branch0.state_dict(), File)\n",
        "\n",
        "File = \"HJ_branch1.pth\"\n",
        "torch.save(HJ_branch1.state_dict(), File)\n",
        "\n",
        "File = \"HJ_trunk.pth\"\n",
        "torch.save(HJ_trunk.state_dict(), File)"
      ],
      "metadata": {
        "id": "4O-4-vS5RzaT"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [],
      "metadata": {
        "id": "lHnDsnXpRzeS"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# Define a heat map plot\n",
        "def heatmap2d(data: np.ndarray):\n",
        "  cmap = LinearSegmentedColormap.from_list('Colormap', [ 'orangered', 'darkorange', 'aquamarine', 'dodgerblue', 'blue'])\n",
        "  fig, (ax) = plt.subplots(1,1, figsize=(4,4))\n",
        "  ax = sns.heatmap(data, ax=ax, cmap=cmap, square=True, cbar=False)\n",
        "\n",
        "  ax.set(xticklabels=[]); ax.set(yticklabels=[]); ax.tick_params(left=False, bottom=False)\n",
        "  plt.savefig(\"heatmap.png\", dpi=300)\n",
        "  plt.show()"
      ],
      "metadata": {
        "id": "aohFp2aYRziO"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# Create plot for test pairing\n",
        "\n",
        "u_test = torch.zeros(size=(n,n))\n",
        "I = np.random.randint(0,N,1)\n",
        "J = np.random.randint(0,N,1)\n",
        "\n",
        "# Compute GeONet geodesic\n",
        "for i in range(n):\n",
        "  cty_branch0_output_test = cty_branch0(u0[I,:])\n",
        "  cty_branch1_output_test = cty_branch1(u1[J,:])\n",
        "  T_test = 1*(i/(n-1))*torch.ones(size=(n,1)).to(device)\n",
        "  cty_trunk_output_test = cty_trunk(X_vec, T_test)\n",
        "  u_test[:,i] = torch.sum( (cty_branch0_output_test * cty_branch1_output_test) * cty_trunk_output_test, dim=1)\n",
        "\n",
        "heatmap2d(u_test.detach().numpy())"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 345
        },
        "id": "-hOI7bB9Rzmh",
        "outputId": "3c4f8e96-f76d-4cff-ea10-319ea57c97c1"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 400x400 with 1 Axes>"
            ],
            "image/png": "iVBORw0KGgoAAAANSUhEUgAAAUgAAAFICAYAAAAyFGczAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/bCgiHAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAWSklEQVR4nO3dy64rRxXG8ba9OdyTgLiGOxKCCaOMEYIJ4tV4JN4CAUJcpAwQgogoQkASzo7tZnCU7ep2f+W1alW1y97/38jpXV1VbvtU6uvqbm/GcRwHAMCZ7bU7AAC9YoAEAIEBEgAEBkgAEBggAUBggAQAgQESAAQGSAAQGCABQHgwl/zFpmE3OsO9RcB9+7XtHzkzSAAQGCABQLBH7N4RiwFUxgwSAAQGSAAQ7BGbCAvgmWEGCQACAyQACAyQACAwQAKAwAAJAMLtXSjeyy3hrOoDd48ZJAAIDJAAINgjdi/Rthccj3VxSgNXwAwSAAQGSAAQbm8VG88TpzRwBcwgAUBggAQAgQESAAQGSAAQGCABQGCABACBARIABAZIABAYIAFAYIAEAIEBEgAEBkgAEBggAUBggAQAgQESAASeBwnA5xk9m5MZJAAIDJAAIBCxgd48owjrtvKxYQYJAAIDJAAIRGw8P0TYmGsePyI2APSBARIABCI28oijferlc2nRj+i0rWKfmEECgMAACQACEXstvUQixPT4ObbqU2T65O1TyXuw7BM8NswgAUBggAQA4flG7B6jEmx6/OyIue3LROsqeH/MIAFAYIAEAOH2InaP8eo56vFz6DHmpnqPsNGYG2lDHeNcnaxiA8D1MEACgHD9iN1jVOtd78esdf/Wvle3VfnWK7LefXPH1VtX6+25v5XEdWdVAPDsMUACgMAACQCC/Rxk7+e91nBLx+DezgNG9219KUz07o1a59M2zjKW7dZ6LX1du09c5gMAbTBAAoBw/ct8aukx/q7Rp2vd5dFjzLWWqXX5Sy9xMd1eM+ZG+tG6ryXtcZkPANTDAAkAQruI3Uvk7T3mRvvXIhrf+4MNeoiONduLlNlmylyrf9Y7eojYAHA9DJAAIKxzoTgx11++9wceRB9scK2YW1KXt4z1veXibY02oses9z7Vai+DGSQACAyQACCURexWw2rrGN/jxc3Re0ivda+udcW41oXL0Vgc6VOtvuaOmYrbrdteI85GVrFLrkwQr8fNZrFIDjNIABAYIAFAsEfsXfK6xaptyT5rlPGu4taK0iVxVm2PrJZay0TiXIsot0bEbrWK3TpiN4qwqTTOHrc78Xp5+JmU2exmf3tYLifqTfdPy3x1udtnmEECgMAACQBCu4gdXZ29VrS17lOyKlhaTzTOto621nJrxuqacbbVimxgFTaNiykZNUVsVWXybVyOsKb47IzFRfVult8fERsAghggAUCwR+wXYnuraBupN7JKGd0nEi9r9qlW29FV4hbtlcT2QIQ1Rc1ghPXGVm9UNbWb65+Iqi2OjSWqn9flq/fHstYpZpAAIDBAAoBQN2J7L0ie/3et1cxUyT2urSNf73F2K16vEGFbRbMWsdVdz2bWJ+cx0O+n/nuwvo/Wq+klsd/SDytmkAAgMEACgGCfc35abF9hdbF5tLU+5qlWnFX1lByzZJ9aEXb/8MnL+57dI+uLS/udoQ1nxDPHxcbxtFXEtq6Oe/qRj7P1T3dEjt+c6XuQ/AMZj7lLQ5YxgwQAgQESAAR7xP5s8rpVXFSv1X3gkSicW3E37J/G2f3DaYm/RYSdxwzv/rr85dgl9y14DNWasfrsmMm267e39oXiss5gBD2Om+RvyfZjuj05rTMOy9vT8oeknr2oR7Q1DMMwjsvlVHxW72H47mLxM8wgAUBggAQAwR6xX09eeyNsMM4edioqnLbvd8sxV8ZfEUHP2/BFxEl7hogzibAimqV15t6HLRqXR+lshHW+b1WXLQq3v+h5v/3E0+tpjKwTO8/3SbaPvuhpiZ3efpfsY4rYzveZ61NK9W96nJIdvrtYzRlmkAAgMEACgGCO2B9+5bWn15YVUkscfbW/iMaWyLy5HKVVPSpevqrXtwJsit6mflyOqdk+GSJ6i3h+Vi75/+7hkK6SJuUDMe1waLP66Y+USZ/UimyjiO3tt4qa2YidtjeKuhr3+zxii7qcfRp+NpgwgwQAgQESAAQGSAAQzOcg//T9nz69VpemeM9HluxTq21VPlqv95If7+UkwzAMh/02KTck5Qzn7D4qv2xEnU882z9wninta6vLQyLn6SbnGgPn5XL9XfPymvT9FJ23TfeZlBkubz+I7Wn5pIy5jcPy9mH2nbVgBgkAAgMkAAjmiP27H/3y6XUaL/0R1PYQAXe9hktLLNH0VTkRd5L4t39ZHkn98dJ2qYPsd+NLRXL9NR2PG7o8JHQ3y/yBCq0jaSSODsMw7p1te/vqLG/eJ+l37v1ZMIMEAIEBEgAEc8T+7Vd//vQ6jStpbB1FnDom09zD43RMVnHEEje9dzKovs5vgLfF1vQ9XI6hk3gfuENh3r8eVkJzffLG0FStFdLsPrUiqbEfpj41jqqm8rN9Whyzooj9kaG9dLsqb8QMEgAEBkgAEMwR+89/eOPptXuVMrOSZ3nwgDeG6rh9OXae7yO2r7j6OVczel7c1xKPzjp4eX9Z3hr/HPtG989FvtC+qq5IhBX1N1sxdrZnej+5WOz8bqZ1sYoNABUxQAKAYI7Yb//x9JsL3pVhFTVf/fdy3JyWOb2udh9oqmTFcxBlLHEsEjuHwR2TZZla9cxF4ligntx292enykTqicZ+UVf09MNS+bN9WpwOcB77mu1ZMYMEAIEBEgAEc8R+54+fOv2HJXZZYsm8rmvFQuvFvKlKbbij4xptGKKI+TMN9CPbRmlbw1D1s7i47xW/W9HPwXI6QdZr/fcfkUzv0pNz6WMCNukUkIgNAPUwQAKAYI7YL98Wf/DGlbmSiHShbXc9BVPvcAyt1A/3/aXO8kXxyNuG9z0U9KnF98BU507/Sd4KoKYtqr0HQxlLPTMbZ8ROhU5XWD/fSD+MmEECgMAACQCCOWLv/+msOTi1VaJT5ruw4jHYZCKi5NxHRU0Ztczf2sttrKrz7+69/NvaVJz2MYMEAIEBEgAEc1gZH501rzz01pxWTwTqLYqnlftwlXoTzT6XJbWOt1GP3zkl3FfvaRNLe6JMdl/Rj8k+W7G94PvBDBIABAZIABDMEXvzQmy3TFtzw7BhiDZN1w39MMeMHvuUWrF/7jrnnPu42yiITZH4FylvPuVS6/NVEVb1I1OnbG9nKKMib/J6ux2TMmOyfdaPTfq35XK73Wk5fiPaGIZPis5OMYMEAIEBEgAEc8R++PLydsu02lR+GNzTdVO90Zjr7UfglEP0FED4dIenH8a46P0c3f0QZaZxKtfGcrl5tFvcd6P29dcp+yHa0NFxuc6tuFJ+GmfH2d+W+6Ei8LS9tPypTBp/1b67ne7T7mG5jXS7qmv3kF4J/+ZgwQwSAAQGSAAQzBH7xbdOr02rVdGVMssFoaptZ6zLxTFvBFOxy9KerDPTPxXBpvv7+mFpW0W23D6WtmVfTe8zc5wMx9YSZ6d1ivKb5QiabVvWtdzedNU3KVMpCp+3cTkCp9sn5Q2xWEXvdPv5PsvxebJ9e7qh/2Gf3u1CxAaAEAZIABAYIAFAMJ+DfP17+6fX3vM2Ref4NoYylc5jRc9d2dpQ9RvOrWXOv7n7ZzjXpdvK9aPtZxH9HNznEcUlMpbjajknGN1fnadzX3aTnLubn1dO/6Yu1Xk4fnTa/5ic7zu8PL3eJ68Pj0n5fVLmcbn8/06vz8olbbx4/GCxjWnbSV1vvTVYMIMEAIEBEgAEc8T+xnf+u7jde6nCnLp0YdqG2LcgMp/a0n/zxufpvqq9QF8LLqlpH3lzp01qtZGUMdy1Yr6kxnlayBK3SyL2NvkNhjSeTl6P+8XtaXSclhdlDsvxV9WZL7cccydxNimTvgcZpQ1xObePpd7JZT5v/WqwYAYJAAIDJAAI5oj95rdPEbtF5B0Gfyys1bblwQTD0OYumUn9gWj/an9fn7x3yVhOp2T7YVjRVbFzUsYQQdW+r/4m9kni6XZ/ud5JPeNyGRWFc32fxFPRhqrXEp8tK8n2uspjrqVM2r9539P9d4ekXHrzTfpafyUkZpAAIDBAAoBgjthf++b7i9tzK6wfs0bsSb3OfSIxPF+vIVYbfhHeEhflvuNe/81Z7ySyjOl2UX401GPsh4qLpvLOVd58vZcjaaRtywpzrpwl2rrLi3iuoverusTqs6E9U+RN/2kdRJn5P61R/O1gKOP/588MEgAUBkgAEMwR++uHvzbpgIpa7nqOOoZe3tffB1M0NvTJEjWt9brjtojultXgsxXZFeOzqZ4x0z9DXWql1xbPL8fcs785o7E3Yk8irzWCqthr2V/FZEvczvXJsj8RGwDaY4AEAMEcsb/03tst+zEMQ361tkr9BVHau79p9VjtG4zbkYhti9W6remF0r7obrng2hKRVdTM7m+IsJbo7b24e96GOwJbVm298TcXZyN9suxreW1tw1qXATNIABAYIAFAMEfsN/79t5b9KBKNzN56LKcAQhdui/K5lf5InJ2UD0TbV/sHYrVzZdgbl7P7GyK2jL+R2Jnbx7saXKtP8/5FonEkSlv75F2tJmIDQD0MkAAgmCP2595/t2U/ingjtvWi9GoXYjvjrNrXfI+x4b5kb1Q1R+xKdXnjb/p6MyYZKncPb4uVV+/qcUl73ogd6WtJvZZoG43F3jYs+2YwgwQAgQESAARzxP7Uy/806YA7Jje44DpXZ62Lr9W+1nuJp/vXuUfZuwJedCG2isCReFkS2SJR1RtnS1aJa73XWlHY2ralfKSM9W9q++UnEWYxgwQAgQESAARzxH7x0QcXy5Q9NqzBxdfOmJvrkyX2eiOv6odlxTdXb+Rib1P8LbmouFYUjNyDa+1H64uhravYkahq2df6CLBIHLasGEdXmy3xuWDlOsUMEgAEBkgAEOwR+7E8Yrf60SnLD02VPA6sxUXPun/+i7LlBdG93CNb65FUteJ2tD3LMfO2VbK/ZfXYUo9l35I2IvWkSlaeI5E+gxkkAAgMkAAgmCN27iLhy9ttT6K27NNiZdj6A1SrrgznHpPVIrZGLtCOttH6/cz/prbXWklWZXLRsdYqcc2o2SI+p4IXcbeK1SlmkAAgMEACgFC0ih1ZPS5ZMTaVD/y+cXaVuFbMq3nvbK1oXHMFt0VUrbV6XLrPpe3e31zOlVlzlThSvvb+nWMGCQACAyQACAyQACAUPayi5jMIW/+Y/OboPJ84/+/Wz+bz1t+qjZrn+Lx3f7S4pKZk/8jDD27p8pqaNslrbz/U9Mx6+Y+l7Uj/BmaQACAxQAKAUHSZj/fSGRWXs+W8v1zX6pKVHh4GUbN/LS7Bse4TKRPZfulvNXjjHqbSY1YybYvelSMwgwQAgQESAISiXzWstdo8DMOwOySr2q3v+LCUr9lGrZi79qP6VfmShy1EqNhq2V5iJ+r1WuMhDLX2rXknjff41/zs0qlexe8iM0gAEBggAUCwPw9ynzwPMonSk+3e5yLO//s5rhIr0ZjWIoZaI2jad28/al5wHd2ndN+d2F4zzrYof81TA2vW6cAMEgAEBkgAEBw/ueC7ONz0TMXc33qIuTneqOq9kNh6D2nkl+4i23PlakXMNWPxXK0Lj9eO1ZF9r9nXTttjBgkAAgMkAAjmiH1V6TBuiT4qnqrX87gcWW21PIZ/jZjr3X+NOBaJrTXjVK26eunTvb2fjjCDBACBARIABHPE3u9ePL3ebk/LlN7Hlc2ZV7ufKjOUsdw/XPJorBbRuJcLo2s+LuqWLhimr+3dcNxmBgkAAgMkAAjmiP344jOL2y1PEZ+UzzxRfLpd7C/LL2+fRPhUdGW4pK7S8rX3v6TR05lN7vBi4+7afo7vuRAzSAAQGCABQDBH7P9+9ktPr70xV8Xl7D6jXvm21Hup/tr7nPa93KdeWI7xLYl8brfglr5bSi+fkXpcwBwzSAAQGCABQDBH7H+99o2n1zoW9xF57yGKYKqXaIb78D1jOWaQACAwQAKAYI7Y773x7cXt0ZVQ4vBlx+1tPJXunhw31nXO+3Xc3u8xIGIDQBADJAAI5uz27hetk9L7da2oewtRp/c+dt+/zf1+t3o/9jnMIAFAYIAEAIEBEgCEq5+DbHFer9U5jxb1rn1ec+3LV1qff1r9+K18Pq3Jd67i+c5I/2q+t2OjuR4zSAAQGCABQLDfSfOF5TtpUtGp+7Wm69Hp+XjchPZfchzr1zk3NvpphWOD41HzGB8jj/0XN461+A4MQ6tjGds/8t30HqdjQV9Nx+xrtrqYQQKAwAAJAIL9JxeG1xa3eyNASRTxRiL/ND4WY2pF1VaxuskpgM7j+Rg8lrXeX91TAx3G7cD7KxsLvOON+AMRGwBiGCABQDBH7MeXp1Viy7TaOnX3T5k7WQUziMa8j7WKs6lmq7CVjkHNFfdqn2+wnh6PTaqH41RyxUHN7zIzSAAQGCABQDBH7MP+8rTVEhmKVq4CESK8Qk1MfsLF6+2PcU+221O+jRy/zfZyTlbHdWtsNo3ilvasmEECgMAACQCCOWLvHi5PW3eDYUq+0/WoeLUz3Gat4l/apxJrRqo1Yrhu+3rRsdZpjJzWx3aN78kapzhSrU53RKz9PWUGCQACAyQACOaI/enP7Be315z2X2tKf814qdzyamnocWIr6PHY9vgdvFU1xxFmkAAgMEACgGCO2K9/+E7LfjSz9o863bNb/gH4iTt5G1i2PYrHvk/8wFZXrCsAcL8YIAFAMOfPr7z7lyoNbsfl1XDgubFFwf5cs9/V2v7hT2zt1WkNAO4PAyQACOaI/eY7v3dVvD32HaVvNd7UtB3v9xj0+P1b+zu3fnv1j3mz7+gPje23aR0Abh8DJAAI5oj9zb//pkqDrab9vUSqHqN7j1H6mp/Xmp/RGm21OpYt+l7zu7jGd4gZJAAIDJAAINhXsf9xWsWuObWtNeW+5Siz3NZ9r3huxpWfida6uTXezq2+h1b1rvB4RGaQACAwQAKAYI7Yn3/3n3VajE63W0zX7yEerdHGPbyHuTWfYr/2e7uHz+vKT6dnBgkAAgMkAAgMkAAg2H+P4D+BVmqeR+Ac5FTrc2j3eN7smu310vbHeujDMPTTjxlmkAAgMEACgGCP2O+L7b3H5zXrH4Z1Lhu5h+N0zfZ6adui9/6leu9rQf+YQQKAwAAJAII9Yn8gthPH2ljzDo9heD7H1arHPkXc2/uZa/T+mEECgMAACQCCPWJ/2LAXtfUYJ4iwU2ufQqilx2PZuxs+ZswgAUBggAQAwR6xHxv2otStTt1vqd+31Fere3xPzx2r2ACwLgZIABDsEft6v/Nez3ONVs/1fXtxnDDDDBIABAZIABDsEXvd35W/Xbd6AfRzQpSGETNIABAYIAFAIGLfIiIisApmkAAgMEACgGCP2MS6KY4H0I9Nm2qZQQKAwAAJAAIRG2X4Pkw1ingw4nFnALAuBkgAEIjYQA38+7hLzCABQGCABACBARIABAZIABAYIAFAYIAEAIEBEgAEBkgAEBggAUBggAQAgQESAAQGSAAQGCABQGCABACBARIABAZIABAYIAFAYIAEAIEBEgAEBkgAEBggAUBggAQAgQESAAQGSAAQNuM48pPnALCAGSQACAyQACAwQAKAwAAJAAIDJAAIDJAAIDBAAoDAAAkAAgMkAAj/B5zEQL0bMiawAAAAAElFTkSuQmCC\n"
          },
          "metadata": {}
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [],
      "metadata": {
        "id": "2Jc_p-K6Rzqq"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# Pot algorithm for comparison\n",
        "\n",
        "T = n\n",
        "y = np.arange(n, dtype=np.float64)\n",
        "OT = np.zeros(shape=(n,T))\n",
        "\n",
        "# Gaussian distributions\n",
        "a1 = u0[I,:].cpu()\n",
        "a2 = u1[J,:].cpu()\n",
        "\n",
        "A = np.vstack((a1, a2)).T\n",
        "n_distributions = A.shape[1]\n",
        "\n",
        "K = ot.utils.dist0(n)\n",
        "K /= K.max()\n",
        "\n",
        "for t in range(T):\n",
        "  alpha = t/T  # 0<=alpha<=1\n",
        "  weights = np.array([1 - alpha, alpha])\n",
        "\n",
        "  # Wasserstein geodesic computation\n",
        "  reg = 0.0006 # Regularization parameter. Adjust if desired for smoothing\n",
        "  bary_wass = ot.bregman.barycenter(A, K, reg, weights)\n",
        "\n",
        "  OT[:,t] = bary_wass"
      ],
      "metadata": {
        "id": "yN_REWLIRztR"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "heatmap2d(OT)\n",
        "\n",
        "heatmap2d(u_test.detach().numpy())"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 673
        },
        "id": "9lTRz5b5Rzw4",
        "outputId": "d8b57165-01cc-4c43-c50a-a4616eb6f98e"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 400x400 with 1 Axes>"
            ],
            "image/png": "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\n"
          },
          "metadata": {}
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 400x400 with 1 Axes>"
            ],
            "image/png": "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\n"
          },
          "metadata": {}
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [],
      "metadata": {
        "id": "dI6pSvZgRz1U"
      },
      "execution_count": null,
      "outputs": []
    }
  ]
}