{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "e7dcb621-c281-42db-85e8-618ae5cc8541",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import random as random\n",
    "import torch\n",
    "import torch.nn as nn \n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "import random\n",
    "from tqdm import tqdm\n",
    "import scipy.io\n",
    "from multiprocessing import Pool\n",
    "\n",
    "################################################################\n",
    "###      Data generation                                     ###\n",
    "################################################################\n",
    "def RBF(x1, x2, params):\n",
    "    length_scale, output_scale = params \n",
    "    diffs = np.expand_dims(x1 / length_scale, 1) - \\\n",
    "            np.expand_dims(x2 / length_scale, 0)\n",
    "    r2 = np.sum(diffs**2, axis=2)\n",
    "    return output_scale * np.exp(-0.5 * r2)\n",
    "\n",
    "def generate_gaussain_sample(gp_params,N,T):\n",
    "    np.random.seed(None)\n",
    "    gp_samples = np.zeros((N,T))\n",
    "    length_scale, output_scale = gp_params \n",
    "    jitter = 1e-10\n",
    "    X = np.linspace(0.0, 1.0, T)[:,None]\n",
    "    K = RBF(X, X, gp_params)\n",
    "    L = np.linalg.cholesky(K + jitter*np.eye(T))\n",
    "    for i in range(N):\n",
    "        gp_sample = np.dot(L, np.random.normal(loc=0,scale=1,size=T))\n",
    "        gp_samples[i,:] = gp_sample\n",
    "    return gp_samples\n",
    "\n",
    "def gen_ks(h,T,tpoints,d,Nx,deltas,sigma,u,x,N):\n",
    "    xs = x[::int(N//Nx)]\n",
    "    tmax = T\n",
    "    nmax = round(tmax/h)\n",
    "    \n",
    "    k = np.transpose(np.conj(np.concatenate((np.arange(0, N/2), np.array([0]), np.arange(-N/2+1, 0)))))*(2*np.pi/d) \n",
    "    \n",
    "    Xs = np.zeros((tpoints,Nx)) #Used to record the status at all times\n",
    "    nplt = int((tmax/tpoints)/h)\n",
    "    ux = u[np.arange(0,len(u),int(N/Nx))]\n",
    "    v = np.fft.fft(u)\n",
    "    uu = np.array([ux])\n",
    "    tt = 0\n",
    "    g = -0.5j*k\n",
    "    \n",
    "    for n in range(1, nmax): \n",
    "        rh = deltas[n-1]\n",
    "        L = k**2 - rh * k**4\n",
    "        E = np.exp(h*L)\n",
    "        E_2 = np.exp(h*L/2)\n",
    "        M = 16\n",
    "        r = np.exp(1j*np.pi*(np.arange(1, M+1)-0.5) / M) \n",
    "        LR = h*np.transpose(np.repeat([L], M, axis=0)) + np.repeat([r], N, axis=0)\n",
    "        Q = h*np.real(np.mean((np.exp(LR/2)-1)/LR, axis=1))\n",
    "        f1 = h*np.real(np.mean((-4-LR+np.exp(LR)*(4-3*LR+LR**2))/LR**3, axis=1))\n",
    "        f2 = h*np.real(np.mean((2+LR+np.exp(LR)*(-2+LR))/LR**3, axis=1))\n",
    "        f3 = h*np.real(np.mean((-4-3*LR-LR**2+np.exp(LR)*(4-LR))/LR**3, axis=1))\n",
    "        \n",
    "        rifftv = np.real(np.fft.ifft(v))\n",
    "        Nv = g*np.fft.fft(rifftv**2)\n",
    "        a = E_2*v + Q*Nv\n",
    "        riffta = np.real(np.fft.ifft(a))\n",
    "        Na = g*np.fft.fft(riffta**2)\n",
    "        b = E_2*v + Q*Na\n",
    "        rifftb = np.real(np.fft.ifft(b))\n",
    "        Nb = g*np.fft.fft(rifftb**2)\n",
    "        c = E_2*a + Q*(2*Nb-Nv)\n",
    "        rifftc = np.real(np.fft.ifft(c))\n",
    "        Nc = g*np.fft.fft(rifftc**2)\n",
    "        v = E*v + Nv*f1 + 2*(Na+Nb)*f2 + Nc*f3\n",
    "        if n%nplt == 0:\n",
    "            #print(n//nplt,\"/\",tpoints)\n",
    "            u = np.real(np.fft.ifft(v))\n",
    "            # add noise\n",
    "            #noise = (np.random.rand(128)-0.5)*2*sigma\n",
    "            noise = np.random.randn(N)*sigma\n",
    "            u = u + noise\n",
    "            v = np.fft.fft(u)\n",
    "            ux = u[np.arange(0,len(u),int(N/Nx))]\n",
    "            uu = np.append(uu, np.array([ux]), axis=0)\n",
    "            tt = np.hstack((tt, n))\n",
    "    if True not in np.isnan(uu):\n",
    "        Xs[:,:] = uu\n",
    "    return(Xs,xs,u)\n",
    "\n",
    "def gen_OneData(ind, gp_params, h, T, tpoints, d, nx, sigma, y0, x, Nx, Nt):\n",
    "    #deltas = np.linspace(0.076,0.0816,int(T/h))\n",
    "    #deltas = np.ones(int(T/h))*0.081\n",
    "    deltas = generate_gaussain_sample(gp_params, 1, Nt)[0]\n",
    "    deltas = deltas - deltas.min() + 0.01\n",
    "    ksdat,xs,u = gen_ks(h,T,tpoints,d,nx,deltas,sigma,y0,x,Nx)\n",
    "    return(ind, ksdat.T, xs)\n",
    "\n",
    "h = 0.01\n",
    "dt = 0.01\n",
    "tpoints = 100\n",
    "T = dt * tpoints\n",
    "ts = np.arange(tpoints) * dt\n",
    "\n",
    "N = 20\n",
    "d = 2*np.pi\n",
    "Nx = 128\n",
    "nx = 64\n",
    "x = d*np.transpose(np.conj(np.arange(-Nx/2+1, Nx/2+1))) / Nx \n",
    "\n",
    "sigma = 0.00\n",
    "\n",
    "length_scale = 0.5\n",
    "output_scale = 0.1\n",
    "gp_params = (length_scale, output_scale)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "1d45999b-7a5c-46ba-b36b-a395948276d9",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 1/1 [00:00<00:00,  1.26it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Data successfully generated! (20, 64, 100)\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "y0 = generate_gaussain_sample(gp_params, 1, Nx)[0]\n",
    "\n",
    "Xs = np.zeros((N,nx,tpoints))\n",
    "Nprocesses = 20\n",
    "for ii in tqdm(range(N//Nprocesses)):\n",
    "    pool = Pool(processes=Nprocesses)\n",
    "    result = []\n",
    "    for i in range(Nprocesses):\n",
    "        ind = ii*Nprocesses + i\n",
    "        result.append(pool.apply_async(gen_OneData, \\\n",
    "                args = (ind, gp_params, h, T, tpoints, d, nx, sigma, y0, x, Nx, tpoints)))\n",
    "    pool.close()\n",
    "    pool.join()\n",
    "    for i in range(Nprocesses):\n",
    "        ind, ksdata, xs = result[i].get()\n",
    "        Xs[ind] = ksdata\n",
    "\n",
    "scipy.io.savemat('dataset/data_ksV2.mat', mdict={'Xs': Xs, 'ts': ts, 'xs': xs})\n",
    "print(\"Data successfully generated!\", Xs.shape)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b453cd0e-a6c7-4f6d-afe2-4da7ebf3778a",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "a60252d2-8916-4f0e-9604-b6bbc0809e6a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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PBMvf/nL8eIXvxgPPPAAzjvvvEE0FQCwZs0arFmzRoXXrl2L0047DbfddhtuuOGGgbWrKt/85jdx5plnYtWqVZn4ufrcj0b6qoSOPfZYBEGAvXv3ZuL37t2L8fFxa57x8fHc9PJapcxu6KbtkptuugmbN2/GT3/6U7z85S/PTXvSSSfh2GOPxa5du464zZIjabuk2WziVa96lWpXHZ77wYMHsXXrVrz3ve8trGc2njtTjr52Qq1WCytXrsS2bdtUXBRF2LZtW+atq7NmzZpMegC4//77VfoVK1ZgfHw8k2b//v34zW9+4yyzX20HgM9//vO44YYbcN999+Hss88urGfPnj149tlncfzxx/ek3UD3bdcJwxCPPPKIatdcf+5APLVjenoa73rXuwrrmY3nzpSk357wrVu3iqGhIXHHHXeIxx9/XLz//e8XCxcuVMO/7373u8W1116r0v/qV78SjUZD3HTTTeKPf/yj2LRpk3WIfuHCheLee+8Vf/jDH8RFF100a0PFVdq+efNm0Wq1xPe///3MUPCBAweEEEIcOHBAfPSjHxXbt28XTz75pPjZz34mzjrrLHHKKaeIqampgbb9+uuvFz/5yU/En//8Z7Fjxw5x6aWXiuHhYfHYY49lvt9cfO6S173udeKSSy4x4vv53Jli+t4JCSHEzTffLF74wheKVqslVq1aJX7961+re+ecc464/PLLM+m/973viZe85CWi1WqJl73sZeJHP/pR5n4UReKTn/ykWLJkiRgaGhLnnXee2Llz58Dbvnz5cnmiWOazadMmIYQQhw4dEueff7447rjjRLPZFMuXLxcbNmww5uMMou1XX321SrtkyRLx5je/WTz88MOZ8ubqcxdCiCeeeEIAED/96U+Nsvr93Jl8eCsPhmEGCq8dYxhmoHAnxDDMQOFOiGGYgcKdEMMwA4U7IYZhBgp3QgzDDBTuhBiGGSjcCTEMM1C4E2IYZqBwJ8QwzEDhTohhmIHCnRDDMAPl/wMrCgRLucizBAAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 300x200 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "\n",
    "i = 6\n",
    "s = Xs[i].transpose()\n",
    "xx_mesh, tt_mesh = np.meshgrid(xs, ts)\n",
    "fig, ax = plt.subplots(1, 1, figsize=(3, 2))\n",
    "fig.set_tight_layout(True)\n",
    "pcm = ax.pcolormesh(tt_mesh, xx_mesh, s, vmin=s.min(), vmax=s.max())\n",
    "plt.colorbar(pcm)\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6e877ca7-40a4-43b4-891c-4b3f755ea8cc",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "34c5a55a-fe58-4d38-809e-056b955c1bed",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0401778c-cddf-47f4-846f-b7a318ebbb56",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "41659610-f2d1-45aa-95e2-2b95b20bae76",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.11.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
