{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import sys\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib\n",
    "import pickle\n",
    "\n",
    "sys.path.append(\"..\")\n",
    "\n",
    "os.environ[\"CUDA_VISIBLE_DEVICES\"] = \"0\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "CKP = \"/restricteddata/ukaea/checkpoints/separate_zf_first_residual_512/20250213_225802\"\n",
    "device = \"cuda\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val: 165\n"
     ]
    }
   ],
   "source": [
    "import yaml\n",
    "from omegaconf import OmegaConf\n",
    "\n",
    "from utils import load_model_and_config\n",
    "from models import get_model\n",
    "\n",
    "from dataset.cyclone import CycloneDataset\n",
    "\n",
    "cfg = OmegaConf.create(yaml.safe_load(open(f\"{CKP}/config.yaml\", \"r\")))\n",
    "\n",
    "# traindata = CycloneDataset(\n",
    "#     active_keys=cfg.dataset.active_keys,\n",
    "#     path=cfg.dataset.path,\n",
    "#     split=\"train\",\n",
    "#     random_seed=cfg.seed,\n",
    "#     test_ratio=0.0,\n",
    "#     normalization=cfg.dataset.normalization,\n",
    "#     spatial_ifft=cfg.dataset.spatial_ifft,\n",
    "#     input_seq_length=cfg.model.input_seq_length,\n",
    "#     target_seq_length=cfg.model.bundle_seq_length,\n",
    "#     trajectories=cfg.dataset.training_trajectories,\n",
    "# )\n",
    "\n",
    "data = CycloneDataset(\n",
    "    active_keys=cfg.dataset.active_keys,\n",
    "    path=cfg.dataset.path,\n",
    "    split=\"val\",\n",
    "    random_seed=cfg.seed,\n",
    "    normalization=None,\n",
    "    spatial_ifft=cfg.dataset.spatial_ifft,\n",
    "    bundle_seq_length=cfg.model.bundle_seq_length,\n",
    "    trajectories=cfg.dataset.validation_trajectories,\n",
    "    subsample=cfg.dataset.subsample,\n",
    "    separate_zf=cfg.dataset.separate_zf,\n",
    "    no_zf=cfg.dataset.no_zf,\n",
    ")\n",
    "\n",
    "print(f\"Val: {len(data)}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def force_aspect(ax, aspect=1):\n",
    "    im = ax.get_images()\n",
    "    extent = im[0].get_extent()\n",
    "    ax.set_aspect(abs((extent[1] - extent[0]) / (extent[3] - extent[2])) / aspect)\n",
    "\n",
    "\n",
    "def distribution_5D(x, **kwargs):\n",
    "    _ = kwargs\n",
    "    labels = [\"vpar\", \"vmu\", \"s\", \"x\", \"y\"]\n",
    "\n",
    "    if isinstance(x, torch.Tensor):\n",
    "        x = x.cpu().detach().numpy()\n",
    "\n",
    "    comb = torch.combinations(torch.arange(5), 2).tolist()\n",
    "\n",
    "    fig, ax = plt.subplots(5, 5, figsize=(20, 20))\n",
    "    for i in range(5):\n",
    "        for j in range(5):\n",
    "            if [i, j] not in comb:\n",
    "                ax[i, j].remove()\n",
    "\n",
    "    c_map = matplotlib.colormaps[\"coolwarm\"]\n",
    "    c_map.set_bad(\"k\")\n",
    "\n",
    "    imin = -1\n",
    "    for i, j in comb:\n",
    "        other = tuple([o for o in range(5) if o != i and o != j])\n",
    "        xx = x[0].std(other)\n",
    "        xx[xx == 0] = np.nan\n",
    "        ax[i, j].matshow(xx, cmap=c_map)\n",
    "\n",
    "        if i > imin:\n",
    "            ax[i, j].set_ylabel(labels[i], fontsize=20)\n",
    "            ax[i, j].set_xlabel(labels[j], fontsize=20)\n",
    "            imin = i\n",
    "\n",
    "        force_aspect(ax[i, j])\n",
    "\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "cfg.model.swin.norm_output = False\n",
    "cfg.model.swin.abs_pe = False\n",
    "cfg.model.swin.act_fn = \"GELU\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Parameters: 86.4M\n",
      "Loading model /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/best.pth (stopped at epoch 200) with loss 0.171674\n"
     ]
    }
   ],
   "source": [
    "model = get_model(cfg, dataset=data)\n",
    "last = False\n",
    "path = f\"{CKP}/best.pth\" if not last else f\"{CKP}/ckp.pth\"\n",
    "\n",
    "model, _, _ = load_model_and_config(path, model, device)\n",
    "\n",
    "model = model.to(device)\n",
    "model = model.eval()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_pearson_correlation(x, y):\n",
    "    # shape of [c, ...]\n",
    "    n_channels = x.shape[0]\n",
    "    x = x.view(n_channels, -1)\n",
    "    y = y.view(n_channels, -1)\n",
    "    x = x - torch.mean(x, dim=1, keepdim=True)\n",
    "    y = y - torch.mean(y, dim=1, keepdim=True)\n",
    "    cov = torch.sum(x * y, dim=1)\n",
    "    std_x = torch.linalg.norm(x, dim=1)\n",
    "    std_y = torch.linalg.norm(y, dim=1)\n",
    "    return torch.mean(cov / (std_x * std_y))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import numpy as np\n",
    "\n",
    "from utils import expand_as\n",
    "\n",
    "\n",
    "def invert_ifft(x):\n",
    "    # invert fft on spatial\n",
    "    knth = np.moveaxis(x, 0, -1).copy()\n",
    "    knth = knth.view(dtype=np.complex64)\n",
    "    # shift freqs to correct range\n",
    "    knth = np.fft.fftn(knth, axes=(3, 4))\n",
    "    knth = np.fft.ifftshift(knth, axes=(3, 4))\n",
    "    knth = np.stack([knth.real, knth.imag]).squeeze().astype(\"float32\")\n",
    "    return knth\n",
    "\n",
    "\n",
    "def normalize(x, file_index):\n",
    "    shift, scale = 0.0, 1.0\n",
    "    if cfg.dataset.normalization_scope == \"sample\":\n",
    "        if cfg.dataset.normalization == \"zscore\":\n",
    "            shift = x.mean((2, 3, 4, 5, 6), keepdims=True)\n",
    "            scale = x.std((2, 3, 4, 5, 6), keepdims=True)\n",
    "        if cfg.dataset.normalization == \"minmax\":\n",
    "            x_min = x.min((2, 3, 4, 5, 6), keepdims=True)\n",
    "            x_max = x.max((2, 3, 4, 5, 6), keepdims=True)\n",
    "            shift = x_min\n",
    "            scale = x_max - x_min\n",
    "    if cfg.dataset.normalization_scope == \"dataset\":\n",
    "        if cfg.dataset.normalization == \"zscore\":\n",
    "            shift = expand_as(data.dataset_stats[file_index][\"mean\"][None], x)\n",
    "            scale = expand_as(data.dataset_stats[file_index][\"std\"][None], x)\n",
    "        if cfg.dataset.normalization == \"minmax\":\n",
    "            x_min = expand_as(data.dataset_stats[file_index][\"min\"][None], x)\n",
    "            x_max = expand_as(data.dataset_stats[file_index][\"max\"][None], x)\n",
    "            shift = x_min\n",
    "            scale = x_max - x_min\n",
    "        scale = torch.tensor(scale, device=x.device)\n",
    "        shift = torch.tensor(shift, device=x.device)\n",
    "    return (x - shift) / scale, shift, scale"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "ONESTEP = False\n",
    "cyclone_name = \"_\".join(data.files[0].split(\"/\")[-1].split(\".\")[0].split(\"_\")[:-1])\n",
    "OUT_DIR = f\"{CKP}/{'onestep' if ONESTEP else 'autoreg'}/{cyclone_name}/{'best' if not last else 'ckp'}\"\n",
    "os.makedirs(OUT_DIR, exist_ok=True)\n",
    "IDX_0 = 0\n",
    "IDX_END = len(data) - 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "def modify_fds_dat(path):\n",
    "    with open(path, \"r\") as infile:\n",
    "        content = infile.read()\n",
    "        content = content.replace(\"DTIM    =  2.000000000000000E-002\", \"DTIM    =  0.0\")\n",
    "        content = content.replace(\n",
    "            \"NT_REMAIN       =           0\", \"NT_REMAIN       =           1\"\n",
    "        )\n",
    "        content = content.replace(\"TIME    =   192.753733197446     \", \"TIME    =   0\")\n",
    "\n",
    "    with open(path, \"w\") as outfile:\n",
    "        outfile.write(content)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "def modify_input_dat(path):\n",
    "    with open(path, \"r\") as infile:\n",
    "        content = infile.read()\n",
    "        content = content.replace(\"READ_FILE  = .false.\", \"READ_FILE  = .true.\")\n",
    "        content = content.replace(\"DTIM   = 0.02\", \"DTIM   = 0.0\")\n",
    "        content = content.replace(\"out3d_interval = 3\", \"out3d_interval = 1\")\n",
    "        content = content.replace(\"keep_dumps = .true.\", \"! keep_dumps = .true.\")\n",
    "        content = content.replace(\"ndump_ts = 3\", \"! ndump_ts = 3\")\n",
    "\n",
    "    with open(path, \"w\") as outfile:\n",
    "        outfile.write(content)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K01\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K02\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K03\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K04\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K05\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K06\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K07\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K08\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K09\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K10\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K11\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K12\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K13\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K14\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K15\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K16\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K17\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K18\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K19\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K20\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K21\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K22\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K23\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K24\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K25\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K26\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K27\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K28\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K29\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K30\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K31\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K32\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K33\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K34\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K35\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K36\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K37\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K38\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K39\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K40\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K41\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K42\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K43\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K44\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K45\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K46\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K47\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K48\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K49\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K50\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K51\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K52\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K53\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K54\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K55\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K56\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K57\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K58\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K59\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K60\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K61\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K62\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K63\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K64\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K65\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K66\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K67\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K68\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K69\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K70\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K71\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K72\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K73\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K74\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K75\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K76\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K77\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K78\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K79\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K80\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K81\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K82\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K83\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K84\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K85\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K86\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K87\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K88\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K89\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K90\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K91\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K92\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K93\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K94\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K95\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K96\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K97\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K98\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K99\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K100\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K101\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K102\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K103\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K104\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K105\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K106\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K107\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K108\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K109\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K110\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K111\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K112\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K113\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K114\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K115\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K116\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K117\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K118\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K119\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K120\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K121\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K122\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K123\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K124\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K125\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K126\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K127\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K128\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K129\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K130\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K131\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K132\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K133\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K134\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K135\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K136\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K137\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K138\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K139\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K140\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K141\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K142\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K143\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K144\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K145\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K146\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K147\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K148\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K149\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K150\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K151\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K152\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K153\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K154\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K155\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K156\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K157\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K158\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K159\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K160\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K161\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K162\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K163\n",
      "Writing file /restricteddata/ukaea/checkpoints/separate_zf_first_residual_deep/20250221_175419/autoreg/cyclone4_2_2_ifft_separate/best/K164\n"
     ]
    }
   ],
   "source": [
    "losses = []\n",
    "sample_0 = data[IDX_0]\n",
    "xt = sample_0.x.to(device).unsqueeze(0)\n",
    "itg = sample_0.itg.to(device).unsqueeze(0)\n",
    "f_idx = sample_0.file_index.item()\n",
    "timesteps = data.get_timesteps(torch.tensor([0], dtype=torch.long))\n",
    "files = []\n",
    "gt_corr = {}\n",
    "model_corr = {}\n",
    "\n",
    "with torch.no_grad():\n",
    "    for idx in range(IDX_0, IDX_END + 1):\n",
    "        if ONESTEP:\n",
    "            xt = data[idx].x.to(device).unsqueeze(0)\n",
    "\n",
    "        yt = data[idx].y.to(device).unsqueeze(0)\n",
    "        ts = timesteps[:, idx].to(device)\n",
    "        xt, shift, scale = normalize(xt, f_idx)\n",
    "        yt, gt_shift, gt_scale = normalize(yt, f_idx)\n",
    "        gt_corr[ts] = compute_pearson_correlation(\n",
    "            xt.squeeze()[:2] + xt.squeeze()[2:], yt.squeeze()[:2] + yt.squeeze()[2:]\n",
    "        )\n",
    "        xt = model(xt, timestep=ts, itg=itg)\n",
    "        model_corr[ts] = compute_pearson_correlation(\n",
    "            xt.squeeze()[:2] + xt.squeeze()[2:], yt.squeeze()[:2] + yt.squeeze()[2:]\n",
    "        )\n",
    "        xt = xt / xt.std((2, 3, 4, 5, 6), keepdims=True)\n",
    "        # denormalize\n",
    "        # Try to use GT scale and shift for autoregressive\n",
    "        xt = xt * gt_scale + gt_shift\n",
    "\n",
    "        b_xt = xt.squeeze(0).cpu().numpy()\n",
    "        if cfg.dataset.spatial_ifft:\n",
    "            if cfg.dataset.separate_zf:\n",
    "                # zonal flow are separate channel, add them to original channel before fft\n",
    "                # gt_zf = data[idx+1].y[2:, ...].numpy()\n",
    "                zf = invert_ifft(b_xt[2:, ...])\n",
    "                no_zf = invert_ifft(b_xt[:2, ...])\n",
    "                b_xt = np.zeros_like(zf)\n",
    "                b_xt[..., 0] = zf[..., 0]\n",
    "                b_xt[..., 1:] = no_zf[..., :-1]\n",
    "                # zf = b_xt[2:, ...]\n",
    "                # no_zf = b_xt[:2, ...]\n",
    "                # b_xt = zf + no_zf\n",
    "                # b_xt = invert_ifft(b_xt)\n",
    "            else:\n",
    "                b_xt = invert_ifft(b_xt)\n",
    "        b_xt = b_xt.astype(\"float64\").reshape(-1, order=\"F\")\n",
    "        # dump to file\n",
    "        if OUT_DIR:\n",
    "            dirtarget = os.path.join(\n",
    "                OUT_DIR, f\"K{str((int(idx)+1)*cfg.dataset.subsample).zfill(2)}\"\n",
    "            )\n",
    "            os.makedirs(dirtarget, exist_ok=True)\n",
    "            ftarget = os.path.join(dirtarget, \"FDS\")\n",
    "            os.system(\n",
    "                f\"cp {data.files[0].replace(\"preprocessed\", \"raw\").replace(\"_ifft\", \"\").replace(\"_separate_zf\", \"\").replace(\".h5\", \"\")}/input.dat {dirtarget}\"\n",
    "            )\n",
    "            os.system(\n",
    "                f\"cp {data.files[0].replace(\"preprocessed\", \"raw\").replace(\"_ifft\", \"\").replace(\"_separate_zf\", \"\").replace(\".h5\", \"\")}/FDS.dat {dirtarget}\"\n",
    "            )\n",
    "            modify_fds_dat(f\"{dirtarget}/FDS.dat\")\n",
    "            modify_input_dat(f\"{dirtarget}/input.dat\")\n",
    "            with open(ftarget, \"wb\") as f:\n",
    "                files.append(ftarget)\n",
    "                print(f\"Writing file {dirtarget}\")\n",
    "                f.write(b_xt)\n",
    "\n",
    "            os.system(f\"chmod -R 777 {dirtarget}/*\")\n",
    "\n",
    "pickle.dump(model_corr, open(f\"{OUT_DIR}/model_corr.pkl\", \"wb\"))\n",
    "pickle.dump(gt_corr, open(f\"{OUT_DIR}/gt_corr.pkl\", \"wb\"))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 126,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x7f9b70232570>"
      ]
     },
     "execution_count": 126,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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+ViLwohSqcTQoUORmZnZ4HhmZiZGjRrV5Os+/PBDzJw5Ex988AGmTJly3vdRqVTQ6XQNHr7CNUaIiKg9SU5Oxrp162A2mz3HLBYLPvzww0ZnrQaDFt+mWbhwId566y28/fbbOHToEO69917k5uZ6UtzixYsxY8YMz/kffvghZsyYgX/9618YMWIEioqKUFRUhOrqau+1og3qekZk3CSPiIjagSFDhiAlJQUbN270HNu4cSOSk5MbLIlhtVpxzz33IC4uDmq1GmPGjMGvv/7a4FqbNm1Cz549odFoMG7cOJw8efKs99u5cycuuugiaDQaJCcn45577oGx3hAGf2hxGJk2bRqWLVuGp59+GoMHD8b27duxadMmpKamAgAKCwuRm5vrOf/111+Hw+HA3LlzkZiY6HnMnz/fe61oA89MGg5eJSIKWkIImGwOvz9auHqGxy233ILVq1d7/vz2229j1qxZDc558MEHsWHDBrzzzjv47bff0L17d0ycOBEVFRUAgLy8PFx99dWYPHkysrKyMGfOHDz00EMNrrFv3z5MnDgRV199Nfbu3Yv169djx44dmDdvXqvqbq1WrcB611134a677mr0uTVr1jT489atW1vzFn7j4rReIqKgZ7Y70ffxr/3+vgefngitsuUftdOnT8fixYtx8uRJSCQS/PDDD1i3bp3nM9VoNGLFihVYs2YNJk2aBAB48803kZmZiVWrVuGBBx7AihUrkJGRgZdeegkSiQS9evXCvn378Pzzz3ve54UXXsANN9yABQsWAAB69OiBl19+GRdffDFWrFhx1rohvtLp96Zx1bBnhIiI2peYmBhMmTIF77zzDoQQmDJlCmJiYjzPHz9+HHa7HaNHj/YcUygUGD58OA4dOgQAOHToEEaMGNFgtuvIkSMbvM/u3btx7NgxrF271nNMCAGXy4WcnBz06dPHV01sgGGEt2mIiIKeRiHDwaf9vyeaRtH6WZqzZs3y3C557bXXGjxXd/vnXEttNOcWkcvlwu2334577rnnrOf8OViWYaRuX5pQhhEiomAlkUhadbskkC6//HLYbDYAOGtz2e7du0OpVGLHjh244YYbAAB2ux27du3y3HLp27cvPvnkkwav++mnnxr8eciQIThw4AC6B3grlFYtehZMPDv2smeEiIjaEZlMhkOHDuHQoUNnbUAXEhKCO++8Ew888AC++uorHDx4ELfeeitMJhNmz54NALjjjjtw/PhxLFy4EEeOHMEHH3xw1rjORYsW4ccff8TcuXORlZWF7OxsfPbZZ7j77rv91UwADCNwGut6RjiAlYiI2pdzrbX13HPP4ZprrsH06dMxZMgQHDt2DF9//TUiIyMBuG+zbNiwAZ9//jkGDRqElStXYsmSJQ2uMXDgQGzbtg3Z2dkYO3YsLrjgAjz22GNITEz0edvqa/GuvYHgy117T997LwxffoX4hx9G1IzpXr02ERH5X92uvY3tIkved66ft0927Q1GHMBKREQUWAwjNVxnhIiIKJAYRtgzQkREFFAMI3V703BqLxERUUAwjLBnhIiIKKAYRrg3DRERUUB16jDistkg7HYA7BkhIiIKlM4dRmp7RQBAqtUGsBIiIqLOq3OHkdrBqxKNBhJ5x9qzgIiIKFh07jDCwatEREQB17nDSN20XoYRIiJqJ4qKijB//nx0794darUa8fHxGDNmDFauXInhw4dDIpE0+UhLSwt0+a3Sqe9NsGeEiIjakxMnTmD06NGIiIjAkiVLMGDAADgcDhw9ehRvv/025s2bhwkTJgAA8vLyMHz4cGzevBn9+vUDgLN29+0oGEbAMEJERO3DXXfdBblcjl27diGk3mfTgAEDcM0110AIAYlEAsC9QR0AREdHIyEhISD1ekunDiPO2ts0XGOEiCi4CSEgzGa/v69Eo/GEh/MpLy/HN998gyVLljQIIg2u18xrdTSdOoy4jCYA7BkhIgp2wmzGkSFD/f6+vX7bDUkzl444duwYhBDo1atXg+MxMTGeXpC5c+fi+eef93qdgcYBrACk3JeGiIjaiTN7P3755RdkZWWhX79+sFqtAarKtzp5zwjHjBARdQYSjQa9ftsdkPdtru7du0MikeDw4cMNjmdkZAAANC24VkfDMAKGESKiYCeRSJp9uyRQoqOjcdlll+HVV1/F3Xff3eS4kWDUuW/TGGvXGeEAViIiageWL18Oh8OBYcOGYf369Th06BCOHDmC999/H4cPH+6wU3fPp1P3jIRddhkUXbpCPWBAoEshIiJCt27dsGfPHixZsgSLFy/G6dOnoVKp0LdvX9x///246667Al2iT0iEECLQRZyPXq9HeHg4qqurodPpAl0OERG1YxaLBTk5OUhPT4darQ50OUHvXD/v5n5+d+rbNERERBR4DCNEREQUUAwjREREFFAMI0RERBRQDCNEREQUUAwjREQUlFwuV6BL6BS88XPu1OuMEBFR8FEqlZBKpSgoKEBsbCyUSmXQ7nYbSEII2Gw2lJaWQiqVQqlUtvpaDCNERBRUpFIp0tPTUVhYiIKCgkCXE/S0Wi1SUlIglbb+ZgvDCBERBR2lUomUlBQ4HA44nc5AlxO0ZDIZ5HJ5m3ueGEaIiCgoSSQSKBQKKBSKQJdC58EBrERERBRQDCNEREQUUAwjREREFFAdYsxI3cbCer0+wJUQERFRc9V9btd9jjelQ4QRg8EAAEhOTg5wJURERNRSBoMB4eHhTT4vEeeLK+2Ay+VCQUEBwsLC2jR9SK/XIzk5GXl5edDpdF6ssH3pLO0EOk9b2c7gwnYGn87S1pa2UwgBg8GApKSkc65D0iF6RqRSKbp27eq16+l0uqD+x1Kns7QT6DxtZTuDC9sZfDpLW1vSznP1iNThAFYiIiIKKIYRIiIiCqhOFUZUKhWeeOIJqFSqQJfiU52lnUDnaSvbGVzYzuDTWdrqq3Z2iAGsREREFLw6Vc8IERERtT8MI0RERBRQDCNEREQUUAwjREREFFCdKowsX74c6enpUKvVGDp0KL7//vtAl9Qm27dvx9SpU5GUlASJRIJPPvmkwfNCCDz55JNISkqCRqPBJZdcggMHDgSm2DZYunQp/vSnPyEsLAxxcXG48sorceTIkQbnBENbV6xYgYEDB3oWExo5ciS+/PJLz/PB0MbGLF26FBKJBAsWLPAcC4a2Pvnkk5BIJA0eCQkJnueDoY118vPzcdNNNyE6OhparRaDBw/G7t27Pc8HS1vT0tLO+juVSCSYO3cugOBpp8PhwKOPPor09HRoNBpkZGTg6aefhsvl8pzj9baKTmLdunVCoVCIN998Uxw8eFDMnz9fhISEiFOnTgW6tFbbtGmTeOSRR8SGDRsEAPHxxx83eP65554TYWFhYsOGDWLfvn1i2rRpIjExUej1+sAU3EoTJ04Uq1evFvv37xdZWVliypQpIiUlRdTU1HjOCYa2fvbZZ+KLL74QR44cEUeOHBEPP/ywUCgUYv/+/UKI4GjjmX755ReRlpYmBg4cKObPn+85HgxtfeKJJ0S/fv1EYWGh51FSUuJ5PhjaKIQQFRUVIjU1VcycOVP8/PPPIicnR2zevFkcO3bMc06wtLWkpKTB32dmZqYAILZs2SKECJ52PvPMMyI6Olr873//Ezk5OeKjjz4SoaGhYtmyZZ5zvN3WThNGhg8fLu64444Gx3r37i0eeuihAFXkXWeGEZfLJRISEsRzzz3nOWaxWER4eLhYuXJlACr0npKSEgFAbNu2TQgR3G2NjIwUb731VlC20WAwiB49eojMzExx8cUXe8JIsLT1iSeeEIMGDWr0uWBpoxBCLFq0SIwZM6bJ54OprWeaP3++6Natm3C5XEHVzilTpohZs2Y1OHb11VeLm266SQjhm7/TTnGbxmazYffu3ZgwYUKD4xMmTMDOnTsDVJVv5eTkoKioqEGbVSoVLr744g7f5urqagBAVFQUgOBsq9PpxLp162A0GjFy5MigbOPcuXMxZcoUjB8/vsHxYGprdnY2kpKSkJ6ejuuvvx4nTpwAEFxt/OyzzzBs2DBce+21iIuLwwUXXIA333zT83wwtbU+m82G999/H7NmzYJEIgmqdo4ZMwbffvstjh49CgD4/fffsWPHDkyePBmAb/5OO8RGeW1VVlYGp9OJ+Pj4Bsfj4+NRVFQUoKp8q65djbX51KlTgSjJK4QQWLhwIcaMGYP+/fsDCK627tu3DyNHjoTFYkFoaCg+/vhj9O3b1/MfeDC0EQDWrVuH3377Db/++utZzwXL3+eFF16Id999Fz179kRxcTGeeeYZjBo1CgcOHAiaNgLAiRMnsGLFCixcuBAPP/wwfvnlF9xzzz1QqVSYMWNGULW1vk8++QRVVVWYOXMmgOD5dwsAixYtQnV1NXr37g2ZTAan04lnn30Wf/vb3wD4pq2dIozUkUgkDf4shDjrWLAJtjbPmzcPe/fuxY4dO856Lhja2qtXL2RlZaGqqgobNmzAzTffjG3btnmeD4Y25uXlYf78+fjmm2+gVqubPK+jt3XSpEme7wcMGICRI0eiW7dueOeddzBixAgAHb+NAOByuTBs2DAsWbIEAHDBBRfgwIEDWLFiBWbMmOE5LxjaWt+qVaswadIkJCUlNTgeDO1cv3493n//fXzwwQfo168fsrKysGDBAiQlJeHmm2/2nOfNtnaK2zQxMTGQyWRn9YKUlJScleyCRd2o/WBq8913343PPvsMW7ZsQdeuXT3Hg6mtSqUS3bt3x7Bhw7B06VIMGjQI//73v4Oqjbt370ZJSQmGDh0KuVwOuVyObdu24eWXX4ZcLve0JxjaWl9ISAgGDBiA7OzsoPr7TExMRN++fRsc69OnD3JzcwEE13+fdU6dOoXNmzdjzpw5nmPB1M4HHngADz30EK6//noMGDAA06dPx7333oulS5cC8E1bO0UYUSqVGDp0KDIzMxscz8zMxKhRowJUlW+lp6cjISGhQZttNhu2bdvW4doshMC8efOwceNGfPfdd0hPT2/wfDC19UxCCFit1qBq46WXXop9+/YhKyvL8xg2bBhuvPFGZGVlISMjI2jaWp/VasWhQ4eQmJgYVH+fo0ePPmuq/dGjR5GamgogOP/7XL16NeLi4jBlyhTPsWBqp8lkglTaMB7IZDLP1F6ftLVVw147oLqpvatWrRIHDx4UCxYsECEhIeLkyZOBLq3VDAaD2LNnj9izZ48AIF588UWxZ88ez3Tl5557ToSHh4uNGzeKffv2ib/97W8dcprZnXfeKcLDw8XWrVsbTKszmUyec4KhrYsXLxbbt28XOTk5Yu/eveLhhx8WUqlUfPPNN0KI4GhjU+rPphEiONp63333ia1bt4oTJ06In376SVxxxRUiLCzM8/+cYGijEO7p2XK5XDz77LMiOztbrF27Vmi1WvH+++97zgmWtgohhNPpFCkpKWLRokVnPRcs7bz55ptFly5dPFN7N27cKGJiYsSDDz7oOcfbbe00YUQIIV577TWRmpoqlEqlGDJkiGdqaEe1ZcsWAeCsx8033yyEcE+/euKJJ0RCQoJQqVTioosuEvv27Qts0a3QWBsBiNWrV3vOCYa2zpo1y/PvMzY2Vlx66aWeICJEcLSxKWeGkWBoa926CwqFQiQlJYmrr75aHDhwwPN8MLSxzueffy769+8vVCqV6N27t3jjjTcaPB9Mbf36668FAHHkyJGznguWdur1ejF//nyRkpIi1Gq1yMjIEI888oiwWq2ec7zdVokQQrSuT4WIiIio7TrFmBEiIiJqvxhGiIiIKKAYRoiIiCigGEaIiIgooBhGiIiIKKAYRoiIiCigGEaIiIgooBhGiIiIKKAYRoiIiCigGEaIiIgooBhGiIiIKKAYRoiIiCig/h+vwz/DPiAkkQAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.plot(\n",
    "    [val.cpu().item() for val in model_corr.keys()],\n",
    "    [val.cpu().item() for val in model_corr.values()],\n",
    "    c=\"tab:blue\",\n",
    "    label=\"Model\",\n",
    ")\n",
    "plt.plot(\n",
    "    [val.cpu().item() for val in gt_corr.keys()],\n",
    "    [val.cpu().item() for val in gt_corr.values()],\n",
    "    c=\"tab:red\",\n",
    "    label=\"GT\",\n",
    ")\n",
    "plt.legend()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 107,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from matplotlib import animation, colormaps\n",
    "\n",
    "\n",
    "def force_aspect(ax, aspect=1):\n",
    "    im = ax.get_images()\n",
    "    extent = im[0].get_extent()\n",
    "    ax.set_aspect(abs((extent[1] - extent[0]) / (extent[3] - extent[2])) / aspect)\n",
    "\n",
    "\n",
    "def plot4x4_animate(kfiles, dataset, title=\"\", frames=5, start_idx=0, average=True):\n",
    "    plt.rcParams[\"animation.html\"] = \"jshtml\"\n",
    "    plt.ioff()\n",
    "\n",
    "    labels = [r\"v_{par}\", r\"v_{\\mu}\", r\"s\", r\"k_x\", r\"k_y\"]\n",
    "    comb = torch.combinations(torch.arange(5), 2).tolist()\n",
    "\n",
    "    fig, ax = plt.subplots(5, 5, figsize=(30, 12))\n",
    "    for i in range(5):\n",
    "        for j in range(5):\n",
    "            ax[i, j].remove()\n",
    "\n",
    "    fig.suptitle(title)\n",
    "    c_map = colormaps[\"RdBu\"]\n",
    "    preds = []\n",
    "    for kfname in kfiles[:frames]:\n",
    "        with open(kfname, \"rb\") as fid:\n",
    "            kt = np.fromfile(fid, dtype=np.float64)\n",
    "        kt = np.reshape(kt, (2, 32, 8, 16, 255, 32), order=\"F\").astype(\"float32\")\n",
    "        preds.append(kt)\n",
    "\n",
    "    def animate(t):\n",
    "        sample = dataset[start_idx + t]\n",
    "\n",
    "        xp = preds[t]\n",
    "        xgt = sample.y.numpy()\n",
    "        if cfg.dataset.spatial_ifft:\n",
    "            xgt = invert_ifft(xgt)\n",
    "        ts = sample.timestep.numpy().item()\n",
    "        fig.suptitle(f\"ts={ts:.2f}\", fontsize=30)\n",
    "\n",
    "        for i, j in comb:\n",
    "            other = tuple([o for o in range(5) if o != i and o != j])\n",
    "\n",
    "            if average:\n",
    "                xp_plot = xp[0].mean(other)\n",
    "                xgt_plot = xgt[0].mean(other)\n",
    "            else:\n",
    "                xp_plot = (\n",
    "                    torch.tensor(xp[0]).permute(i, j, *other).numpy()[:, :, 0, 0, 0]\n",
    "                )\n",
    "                xgt_plot = (\n",
    "                    torch.tensor(xgt[0]).permute(i, j, *other).numpy()[:, :, 0, 0, 0]\n",
    "                )\n",
    "\n",
    "            ax_ij = ax[i, j]\n",
    "            pos = ax_ij.get_position()\n",
    "\n",
    "            # create two new axes within the same space as the original subplot\n",
    "            plot_width = 0.475 * pos.width\n",
    "            left_margin = 0.0 * pos.width\n",
    "            x_left_1 = pos.x0 + left_margin\n",
    "            x_left_2 = x_left_1 + plot_width\n",
    "            y = pos.y0\n",
    "            h = pos.height\n",
    "            ax1 = fig.add_axes([x_left_1, y, plot_width, h])\n",
    "            ax2 = fig.add_axes([x_left_2, y, plot_width, h])\n",
    "\n",
    "            # compute shared vmin and vmax\n",
    "            vmin = min(xgt_plot.min(), xp_plot.min())\n",
    "            vmax = max(xgt_plot.max(), xp_plot.max())\n",
    "\n",
    "            ax1.matshow(xp_plot, cmap=c_map)  # , vmin=vmin, vmax=vmax)\n",
    "            ax2.matshow(xgt_plot, cmap=c_map)  # , vmin=vmin, vmax=vmax)\n",
    "\n",
    "            if i == 0:\n",
    "                # Set axis labels\n",
    "                ax1.set_title(r\"PRED\", fontsize=24)\n",
    "                ax2.set_title(r\"GT\", fontsize=24)\n",
    "\n",
    "            if (\n",
    "                j == 1\n",
    "                or (i == 1 and j == 2)\n",
    "                or (i == 2 and j == 3)\n",
    "                or (i == 3 and j == 4)\n",
    "            ):\n",
    "                ax_ij.set_ylabel(rf\"${labels[i]}$\", fontsize=14)\n",
    "\n",
    "            if i == 3 or j == 1 or (i == 1 and j == 2) or (i == 2 and j == 3):\n",
    "                ax_ij.set_xlabel(rf\"${labels[j]}$\", fontsize=14)\n",
    "\n",
    "            # Remove axis ticks and labels\n",
    "            ax1.set_xticks([])\n",
    "            ax1.set_yticks([])\n",
    "            ax2.set_xticks([])\n",
    "            ax2.set_yticks([])\n",
    "            ax1.tick_params(labelleft=False, labelbottom=False)\n",
    "            ax2.tick_params(labelleft=False, labelbottom=False)\n",
    "            # Force aspect ratio\n",
    "            force_aspect(ax1)\n",
    "            force_aspect(ax2)\n",
    "\n",
    "    return animation.FuncAnimation(fig, animate, frames=frames)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 108,
   "metadata": {},
   "outputs": [],
   "source": [
    "ani = plot4x4_animate(files, dataset=data, frames=IDX_END - IDX_0, start_idx=IDX_0)\n",
    "writer = animation.PillowWriter(fps=1, bitrate=600)\n",
    "ani.save(\"onestep.gif\" if ONESTEP else \"autoreg.gif\", writer=writer)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "mhd",
   "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.12.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
