{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Failed to detect the name of this notebook, you can set it manually with the WANDB_NOTEBOOK_NAME environment variable to enable code saving.\n",
      "\u001b[34m\u001b[1mwandb\u001b[0m: Currently logged in as: \u001b[33msarosavo\u001b[0m. Use \u001b[1m`wandb login --relogin`\u001b[0m to force relogin\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "True"
      ]
     },
     "execution_count": 1,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2\n",
    "%matplotlib inline\n",
    "\n",
    "import torch\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import scipy as sp\n",
    "import sys \n",
    "sys.path.append(\"../../\")\n",
    "sys.path.append(\"../../src/\")\n",
    "sys.path.append(\"../../src/model\")\n",
    "\n",
    "from src.model import ddsm as ddsm\n",
    "from src.model import ddsm_model as modeld\n",
    "from src.model.lightning_model_diffusion import LightningDiffusion as lightning_dif\n",
    "import src.utils.sequence as utils\n",
    "\n",
    "import wandb\n",
    "wandb.login(host=\"https://api.wandb.ai\") "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Preparation "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Any numebr is fine \n",
    "DEVICE = \"cuda:1\""
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "If you wanna download a good model already, download it from W&B"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "Tracking run with wandb version 0.17.4"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "Run data is saved locally in <code>/raid/home/zhaoy183/projects/RLfinetuning_Diffusion_Bioseq/tutorials/Human-enhancer/wandb/run-20240718_154709-zjmxxl1m</code>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "Syncing run <strong><a href='https://wandb.ai/sarosavo/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer/runs/zjmxxl1m' target=\"_blank\">fresh-sunset-15</a></strong> to <a href='https://wandb.ai/sarosavo/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer' target=\"_blank\">Weights & Biases</a> (<a href='https://wandb.me/run' target=\"_blank\">docs</a>)<br/>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       " View project at <a href='https://wandb.ai/sarosavo/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer' target=\"_blank\">https://wandb.ai/sarosavo/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer</a>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       " View run at <a href='https://wandb.ai/sarosavo/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer/runs/zjmxxl1m' target=\"_blank\">https://wandb.ai/sarosavo/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer/runs/zjmxxl1m</a>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\u001b[34m\u001b[1mwandb\u001b[0m: Downloading large artifact DNA-dataset:v0, 3766.17MB. 5 files... \n",
      "\u001b[34m\u001b[1mwandb\u001b[0m:   5 of 5 files downloaded.  \n",
      "Done. 0:0:7.2\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6cece09a12f04407a20982173dcd890f",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "VBox(children=(Label(value='0.052 MB of 0.052 MB uploaded\\r'), FloatProgress(value=1.0, max=1.0)))"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       " View run <strong style=\"color:#cdcd00\">fresh-sunset-15</strong> at: <a href='https://wandb.ai/sarosavo/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer/runs/zjmxxl1m' target=\"_blank\">https://wandb.ai/sarosavo/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer/runs/zjmxxl1m</a><br/> View project at: <a href='https://wandb.ai/sarosavo/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer' target=\"_blank\">https://wandb.ai/sarosavo/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer</a><br/>Synced 7 W&B file(s), 0 media file(s), 0 artifact file(s) and 1 other file(s)"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "Find logs at: <code>./wandb/run-20240718_154709-zjmxxl1m/logs</code>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "The new W&B backend becomes opt-out in version 0.18.0; try it out with `wandb.require(\"core\")`! See https://wandb.me/wandb-core for more information."
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "Tracking run with wandb version 0.17.4"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "Run data is saved locally in <code>/raid/home/zhaoy183/projects/RLfinetuning_Diffusion_Bioseq/tutorials/Human-enhancer/wandb/run-20240718_154727-5b9tc9lw</code>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "Syncing run <strong><a href='https://wandb.ai/sarosavo/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer/runs/5b9tc9lw' target=\"_blank\">good-river-16</a></strong> to <a href='https://wandb.ai/sarosavo/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer' target=\"_blank\">Weights & Biases</a> (<a href='https://wandb.me/run' target=\"_blank\">docs</a>)<br/>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       " View project at <a href='https://wandb.ai/sarosavo/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer' target=\"_blank\">https://wandb.ai/sarosavo/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer</a>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       " View run at <a href='https://wandb.ai/sarosavo/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer/runs/5b9tc9lw' target=\"_blank\">https://wandb.ai/sarosavo/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer/runs/5b9tc9lw</a>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\u001b[34m\u001b[1mwandb\u001b[0m: Downloading large artifact DNA-model:v0, 1318.41MB. 3 files... \n",
      "\u001b[34m\u001b[1mwandb\u001b[0m:   3 of 3 files downloaded.  \n",
      "Done. 0:0:2.0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "5f60d74c9ef5433582744478afa7d303",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "VBox(children=(Label(value='0.052 MB of 0.052 MB uploaded\\r'), FloatProgress(value=1.0, max=1.0)))"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       " View run <strong style=\"color:#cdcd00\">good-river-16</strong> at: <a href='https://wandb.ai/sarosavo/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer/runs/5b9tc9lw' target=\"_blank\">https://wandb.ai/sarosavo/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer/runs/5b9tc9lw</a><br/> View project at: <a href='https://wandb.ai/sarosavo/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer' target=\"_blank\">https://wandb.ai/sarosavo/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer</a><br/>Synced 7 W&B file(s), 0 media file(s), 0 artifact file(s) and 1 other file(s)"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "Find logs at: <code>./wandb/run-20240718_154727-5b9tc9lw/logs</code>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "The new W&B backend becomes opt-out in version 0.18.0; try it out with `wandb.require(\"core\")`! See https://wandb.me/wandb-core for more information."
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "\n",
    "run = wandb.init()\n",
    "artifact = run.use_artifact('fderc_diffusion/Diffusion-DNA-RNA/DNA-dataset:v0')\n",
    "dir = artifact.download()\n",
    "wandb.finish()\n",
    "\n",
    "run = wandb.init()\n",
    "artifact = run.use_artifact('fderc_diffusion/Diffusion-DNA-RNA/DNA-model:v0')\n",
    "dir = artifact.download()\n",
    "wandb.finish()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "LightningDiffusion(\n",
       "  (model): ScoreNet_Conditional(\n",
       "    (embed): Sequential(\n",
       "      (0): GaussianFourierProjection()\n",
       "      (1): Linear(in_features=256, out_features=256, bias=True)\n",
       "    )\n",
       "    (linear): Conv1d(4, 256, kernel_size=(9,), stride=(1,), padding=(4,))\n",
       "    (blocks): ModuleList(\n",
       "      (0-1): 2 x Conv1d(256, 256, kernel_size=(9,), stride=(1,), padding=(4,))\n",
       "      (2): Conv1d(256, 256, kernel_size=(9,), stride=(1,), padding=(16,), dilation=(4,))\n",
       "      (3): Conv1d(256, 256, kernel_size=(9,), stride=(1,), padding=(64,), dilation=(16,))\n",
       "      (4): Conv1d(256, 256, kernel_size=(9,), stride=(1,), padding=(256,), dilation=(64,))\n",
       "      (5-6): 2 x Conv1d(256, 256, kernel_size=(9,), stride=(1,), padding=(4,))\n",
       "      (7): Conv1d(256, 256, kernel_size=(9,), stride=(1,), padding=(16,), dilation=(4,))\n",
       "      (8): Conv1d(256, 256, kernel_size=(9,), stride=(1,), padding=(64,), dilation=(16,))\n",
       "      (9): Conv1d(256, 256, kernel_size=(9,), stride=(1,), padding=(256,), dilation=(64,))\n",
       "      (10-11): 2 x Conv1d(256, 256, kernel_size=(9,), stride=(1,), padding=(4,))\n",
       "      (12): Conv1d(256, 256, kernel_size=(9,), stride=(1,), padding=(16,), dilation=(4,))\n",
       "      (13): Conv1d(256, 256, kernel_size=(9,), stride=(1,), padding=(64,), dilation=(16,))\n",
       "      (14): Conv1d(256, 256, kernel_size=(9,), stride=(1,), padding=(256,), dilation=(64,))\n",
       "      (15-16): 2 x Conv1d(256, 256, kernel_size=(9,), stride=(1,), padding=(4,))\n",
       "      (17): Conv1d(256, 256, kernel_size=(9,), stride=(1,), padding=(16,), dilation=(4,))\n",
       "      (18): Conv1d(256, 256, kernel_size=(9,), stride=(1,), padding=(64,), dilation=(16,))\n",
       "      (19): Conv1d(256, 256, kernel_size=(9,), stride=(1,), padding=(256,), dilation=(64,))\n",
       "    )\n",
       "    (denses): ModuleList(\n",
       "      (0-19): 20 x Dense(\n",
       "        (dense): Linear(in_features=256, out_features=256, bias=True)\n",
       "      )\n",
       "    )\n",
       "    (norms): ModuleList(\n",
       "      (0-19): 20 x GroupNorm(1, 256, eps=1e-05, affine=True)\n",
       "    )\n",
       "    (cls_layers): ModuleList(\n",
       "      (0-19): 20 x Dense(\n",
       "        (dense): Linear(in_features=256, out_features=256, bias=True)\n",
       "      )\n",
       "    )\n",
       "    (embed_class): Embedding(3, 256)\n",
       "    (relu): ReLU()\n",
       "    (softplus): Softplus(beta=1.0, threshold=20.0)\n",
       "    (final): Sequential(\n",
       "      (0): Conv1d(256, 256, kernel_size=(1,), stride=(1,))\n",
       "      (1): GELU(approximate='none')\n",
       "      (2): Conv1d(256, 4, kernel_size=(1,), stride=(1,))\n",
       "    )\n",
       "  )\n",
       ")"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "class ModelParameters:\n",
    "    diffusion_weights_file = 'artifacts/DNA-dataset:v0/steps400.cat4.speed_balance.time4.0.samples100000.pth'\n",
    "    time_schedule = 'artifacts/DNA-dataset:v0/time_dependent.npz'\n",
    "    checkpoint_path = 'artifacts/DNA-model:v0/diffusion_epoch=035.ckpt'\n",
    "\n",
    "config = ModelParameters() \n",
    "DEVICE = \"cuda:1\" # Any number is fine\n",
    "\n",
    "score_model = lightning_dif.load_from_checkpoint(checkpoint_path = config.checkpoint_path, weight_file = config.diffusion_weights_file, time_schedule = config.time_schedule, all_class_number = 2)\n",
    "score_model.cuda(device = DEVICE) \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "sampler = ddsm.Euler_Maruyama_sampler\n",
    "yS = 1.0 * torch.ones(128)\n",
    "yS = yS.type(torch.LongTensor)\n",
    "yS = yS.to(DEVICE)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Generate Samples with good properties  "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [00:04<00:00, 24.53it/s]\n",
      "100%|██████████| 100/100 [00:02<00:00, 33.57it/s]\n",
      "100%|██████████| 100/100 [00:03<00:00, 33.25it/s]\n",
      "100%|██████████| 100/100 [00:02<00:00, 33.56it/s]\n",
      "100%|██████████| 100/100 [00:03<00:00, 31.85it/s]\n"
     ]
    }
   ],
   "source": [
    "# generate sequence samples conditioned on 1 (high expression level)\n",
    "\n",
    "allsamples_original_weak = []\n",
    "for t in range(5):\n",
    "    samples=[]\n",
    "    score_model.eval()\n",
    "    samples.append(sampler(score_model,\n",
    "                        (200,4),\n",
    "                        batch_size=128,\n",
    "                        new_class = yS,\n",
    "                        class_number = 2,\n",
    "                        strength = 10, \n",
    "                        max_time=  4.0,\n",
    "                        min_time= 1.0/400,\n",
    "                        time_dilation=1,\n",
    "                        num_steps=100, \n",
    "                        eps=1e-5,\n",
    "                        speed_balanced= True,\n",
    "                        device= DEVICE, \n",
    "                        ).cpu().detach().numpy())\n",
    "    allsamples_original_weak.append(samples)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(640, 200, 4)\n"
     ]
    }
   ],
   "source": [
    "allsamples_weak = np.concatenate(allsamples_original_weak, axis=1)\n",
    "allsamples = allsamples_weak[0,:,:,:]\n",
    "print(allsamples.shape)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Get offline data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(10193, 200, 4)\n"
     ]
    }
   ],
   "source": [
    "data = np.load(\"artifacts/DNA-dataset:v0/y_HepG2.npz\")['x']\n",
    "label = np.load(\"artifacts/DNA-dataset:v0/y_HepG2.npz\")['y'] \n",
    "valsamples = data[label == 1, :, :] # Select data with label=1 \n",
    "print(valsamples.shape)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Plot properties "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\u001b[34m\u001b[1mwandb\u001b[0m: Downloading large artifact human_state_dict:latest, 939.29MB. 1 files... \n",
      "\u001b[34m\u001b[1mwandb\u001b[0m:   1 of 1 files downloaded.  \n",
      "Done. 0:0:0.9\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "LightningModel(\n",
       "  (model): EnformerPretrainedModel(\n",
       "    (embedding): EnformerTrunk(\n",
       "      (conv_tower): EnformerConvTower(\n",
       "        (blocks): ModuleList(\n",
       "          (0): Sequential(\n",
       "            (0): Conv1d(4, 768, kernel_size=(15,), stride=(1,), padding=same)\n",
       "            (1): ConvBlock(\n",
       "              (norm): Norm(\n",
       "                (layer): BatchNorm1d(768, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
       "              )\n",
       "              (conv): Conv1d(768, 768, kernel_size=(1,), stride=(1,), padding=same)\n",
       "              (act): Activation(\n",
       "                (layer): GELU()\n",
       "              )\n",
       "              (pool): Pool(\n",
       "                (layer): AttentionPool(\n",
       "                  (pool_fn): Rearrange('b d (n p) -> b d n p', p=2)\n",
       "                  (to_attn_logits): Conv2d(768, 768, kernel_size=(1, 1), stride=(1, 1), bias=False)\n",
       "                )\n",
       "              )\n",
       "              (dropout): Dropout(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "              (channel_transform): ChannelTransform(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "            )\n",
       "          )\n",
       "          (1): Sequential(\n",
       "            (0): ConvBlock(\n",
       "              (norm): Norm(\n",
       "                (layer): BatchNorm1d(768, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
       "              )\n",
       "              (conv): Conv1d(768, 768, kernel_size=(5,), stride=(1,), padding=same)\n",
       "              (act): Activation(\n",
       "                (layer): GELU()\n",
       "              )\n",
       "              (pool): Pool(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "              (dropout): Dropout(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "            )\n",
       "            (1): ConvBlock(\n",
       "              (norm): Norm(\n",
       "                (layer): BatchNorm1d(768, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
       "              )\n",
       "              (conv): Conv1d(768, 768, kernel_size=(1,), stride=(1,), padding=same)\n",
       "              (act): Activation(\n",
       "                (layer): GELU()\n",
       "              )\n",
       "              (pool): Pool(\n",
       "                (layer): AttentionPool(\n",
       "                  (pool_fn): Rearrange('b d (n p) -> b d n p', p=2)\n",
       "                  (to_attn_logits): Conv2d(768, 768, kernel_size=(1, 1), stride=(1, 1), bias=False)\n",
       "                )\n",
       "              )\n",
       "              (dropout): Dropout(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "              (channel_transform): ChannelTransform(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "            )\n",
       "          )\n",
       "          (2): Sequential(\n",
       "            (0): ConvBlock(\n",
       "              (norm): Norm(\n",
       "                (layer): BatchNorm1d(768, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
       "              )\n",
       "              (conv): Conv1d(768, 896, kernel_size=(5,), stride=(1,), padding=same)\n",
       "              (act): Activation(\n",
       "                (layer): GELU()\n",
       "              )\n",
       "              (pool): Pool(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "              (dropout): Dropout(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "            )\n",
       "            (1): ConvBlock(\n",
       "              (norm): Norm(\n",
       "                (layer): BatchNorm1d(896, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
       "              )\n",
       "              (conv): Conv1d(896, 896, kernel_size=(1,), stride=(1,), padding=same)\n",
       "              (act): Activation(\n",
       "                (layer): GELU()\n",
       "              )\n",
       "              (pool): Pool(\n",
       "                (layer): AttentionPool(\n",
       "                  (pool_fn): Rearrange('b d (n p) -> b d n p', p=2)\n",
       "                  (to_attn_logits): Conv2d(896, 896, kernel_size=(1, 1), stride=(1, 1), bias=False)\n",
       "                )\n",
       "              )\n",
       "              (dropout): Dropout(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "              (channel_transform): ChannelTransform(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "            )\n",
       "          )\n",
       "          (3): Sequential(\n",
       "            (0): ConvBlock(\n",
       "              (norm): Norm(\n",
       "                (layer): BatchNorm1d(896, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
       "              )\n",
       "              (conv): Conv1d(896, 1024, kernel_size=(5,), stride=(1,), padding=same)\n",
       "              (act): Activation(\n",
       "                (layer): GELU()\n",
       "              )\n",
       "              (pool): Pool(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "              (dropout): Dropout(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "            )\n",
       "            (1): ConvBlock(\n",
       "              (norm): Norm(\n",
       "                (layer): BatchNorm1d(1024, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
       "              )\n",
       "              (conv): Conv1d(1024, 1024, kernel_size=(1,), stride=(1,), padding=same)\n",
       "              (act): Activation(\n",
       "                (layer): GELU()\n",
       "              )\n",
       "              (pool): Pool(\n",
       "                (layer): AttentionPool(\n",
       "                  (pool_fn): Rearrange('b d (n p) -> b d n p', p=2)\n",
       "                  (to_attn_logits): Conv2d(1024, 1024, kernel_size=(1, 1), stride=(1, 1), bias=False)\n",
       "                )\n",
       "              )\n",
       "              (dropout): Dropout(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "              (channel_transform): ChannelTransform(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "            )\n",
       "          )\n",
       "          (4): Sequential(\n",
       "            (0): ConvBlock(\n",
       "              (norm): Norm(\n",
       "                (layer): BatchNorm1d(1024, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
       "              )\n",
       "              (conv): Conv1d(1024, 1152, kernel_size=(5,), stride=(1,), padding=same)\n",
       "              (act): Activation(\n",
       "                (layer): GELU()\n",
       "              )\n",
       "              (pool): Pool(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "              (dropout): Dropout(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "            )\n",
       "            (1): ConvBlock(\n",
       "              (norm): Norm(\n",
       "                (layer): BatchNorm1d(1152, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
       "              )\n",
       "              (conv): Conv1d(1152, 1152, kernel_size=(1,), stride=(1,), padding=same)\n",
       "              (act): Activation(\n",
       "                (layer): GELU()\n",
       "              )\n",
       "              (pool): Pool(\n",
       "                (layer): AttentionPool(\n",
       "                  (pool_fn): Rearrange('b d (n p) -> b d n p', p=2)\n",
       "                  (to_attn_logits): Conv2d(1152, 1152, kernel_size=(1, 1), stride=(1, 1), bias=False)\n",
       "                )\n",
       "              )\n",
       "              (dropout): Dropout(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "              (channel_transform): ChannelTransform(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "            )\n",
       "          )\n",
       "          (5): Sequential(\n",
       "            (0): ConvBlock(\n",
       "              (norm): Norm(\n",
       "                (layer): BatchNorm1d(1152, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
       "              )\n",
       "              (conv): Conv1d(1152, 1280, kernel_size=(5,), stride=(1,), padding=same)\n",
       "              (act): Activation(\n",
       "                (layer): GELU()\n",
       "              )\n",
       "              (pool): Pool(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "              (dropout): Dropout(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "            )\n",
       "            (1): ConvBlock(\n",
       "              (norm): Norm(\n",
       "                (layer): BatchNorm1d(1280, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
       "              )\n",
       "              (conv): Conv1d(1280, 1280, kernel_size=(1,), stride=(1,), padding=same)\n",
       "              (act): Activation(\n",
       "                (layer): GELU()\n",
       "              )\n",
       "              (pool): Pool(\n",
       "                (layer): AttentionPool(\n",
       "                  (pool_fn): Rearrange('b d (n p) -> b d n p', p=2)\n",
       "                  (to_attn_logits): Conv2d(1280, 1280, kernel_size=(1, 1), stride=(1, 1), bias=False)\n",
       "                )\n",
       "              )\n",
       "              (dropout): Dropout(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "              (channel_transform): ChannelTransform(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "            )\n",
       "          )\n",
       "          (6): Sequential(\n",
       "            (0): ConvBlock(\n",
       "              (norm): Norm(\n",
       "                (layer): BatchNorm1d(1280, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
       "              )\n",
       "              (conv): Conv1d(1280, 1536, kernel_size=(5,), stride=(1,), padding=same)\n",
       "              (act): Activation(\n",
       "                (layer): GELU()\n",
       "              )\n",
       "              (pool): Pool(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "              (dropout): Dropout(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "            )\n",
       "            (1): ConvBlock(\n",
       "              (norm): Norm(\n",
       "                (layer): BatchNorm1d(1536, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
       "              )\n",
       "              (conv): Conv1d(1536, 1536, kernel_size=(1,), stride=(1,), padding=same)\n",
       "              (act): Activation(\n",
       "                (layer): GELU()\n",
       "              )\n",
       "              (pool): Pool(\n",
       "                (layer): AttentionPool(\n",
       "                  (pool_fn): Rearrange('b d (n p) -> b d n p', p=2)\n",
       "                  (to_attn_logits): Conv2d(1536, 1536, kernel_size=(1, 1), stride=(1, 1), bias=False)\n",
       "                )\n",
       "              )\n",
       "              (dropout): Dropout(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "              (channel_transform): ChannelTransform(\n",
       "                (layer): Identity()\n",
       "              )\n",
       "            )\n",
       "          )\n",
       "        )\n",
       "      )\n",
       "      (transformer_tower): EnformerTransformerTower(\n",
       "        (blocks): ModuleList(\n",
       "          (0-1): 2 x EnformerTransformerBlock(\n",
       "            (norm): Norm(\n",
       "              (layer): LayerNorm((1536,), eps=1e-05, elementwise_affine=True)\n",
       "            )\n",
       "            (mha): Attention(\n",
       "              (to_q): Linear(in_features=1536, out_features=512, bias=False)\n",
       "              (to_k): Linear(in_features=1536, out_features=512, bias=False)\n",
       "              (to_v): Linear(in_features=1536, out_features=1536, bias=False)\n",
       "              (to_out): Linear(in_features=1536, out_features=1536, bias=True)\n",
       "              (to_rel_k): Linear(in_features=192, out_features=512, bias=False)\n",
       "              (pos_dropout): Dropout(p=0.01, inplace=False)\n",
       "              (attn_dropout): Dropout(p=0.05, inplace=False)\n",
       "            )\n",
       "            (dropout): Dropout(\n",
       "              (layer): Dropout(p=0.4, inplace=False)\n",
       "            )\n",
       "            (ffn): FeedForwardBlock(\n",
       "              (dense1): LinearBlock(\n",
       "                (norm): Norm(\n",
       "                  (layer): LayerNorm((1536,), eps=1e-05, elementwise_affine=True)\n",
       "                )\n",
       "                (linear): Linear(in_features=1536, out_features=3072, bias=True)\n",
       "                (dropout): Dropout(\n",
       "                  (layer): Dropout(p=0.4, inplace=False)\n",
       "                )\n",
       "                (act): Activation(\n",
       "                  (layer): ReLU()\n",
       "                )\n",
       "              )\n",
       "              (dense2): LinearBlock(\n",
       "                (norm): Norm(\n",
       "                  (layer): Identity()\n",
       "                )\n",
       "                (linear): Linear(in_features=3072, out_features=1536, bias=True)\n",
       "                (dropout): Dropout(\n",
       "                  (layer): Dropout(p=0.4, inplace=False)\n",
       "                )\n",
       "                (act): Activation(\n",
       "                  (layer): Identity()\n",
       "                )\n",
       "              )\n",
       "            )\n",
       "          )\n",
       "        )\n",
       "      )\n",
       "      (pointwise_conv): ConvBlock(\n",
       "        (norm): Norm(\n",
       "          (layer): BatchNorm1d(1536, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
       "        )\n",
       "        (conv): Conv1d(1536, 3072, kernel_size=(1,), stride=(1,), padding=same)\n",
       "        (act): Activation(\n",
       "          (layer): GELU()\n",
       "        )\n",
       "        (pool): Pool(\n",
       "          (layer): Identity()\n",
       "        )\n",
       "        (dropout): Dropout(\n",
       "          (layer): Identity()\n",
       "        )\n",
       "      )\n",
       "      (act): Activation(\n",
       "        (layer): GELU()\n",
       "      )\n",
       "      (crop): Crop(\n",
       "        (layer): Identity()\n",
       "      )\n",
       "    )\n",
       "    (head): ConvHead(\n",
       "      (channel_transform): ChannelTransformBlock(\n",
       "        (norm): Norm(\n",
       "          (layer): Identity()\n",
       "        )\n",
       "        (conv): ChannelTransform(\n",
       "          (layer): Conv1d(3072, 3, kernel_size=(1,), stride=(1,), padding=same)\n",
       "        )\n",
       "        (act): Activation(\n",
       "          (layer): Identity()\n",
       "        )\n",
       "        (dropout): Dropout(\n",
       "          (layer): Identity()\n",
       "        )\n",
       "      )\n",
       "      (pool): AdaptivePool(\n",
       "        (layer): AdaptiveAvgPool1d(output_size=1)\n",
       "      )\n",
       "    )\n",
       "  )\n",
       "  (loss): MSELoss()\n",
       "  (activation): Identity()\n",
       "  (val_metrics): MetricCollection(\n",
       "    (mse): MSE()\n",
       "    (pearson): PearsonCorrCoef(\n",
       "      (pearson): PearsonCorrCoef()\n",
       "    ),\n",
       "    prefix=val_\n",
       "  )\n",
       "  (test_metrics): MetricCollection(\n",
       "    (mse): MSE()\n",
       "    (pearson): PearsonCorrCoef(\n",
       "      (pearson): PearsonCorrCoef()\n",
       "    ),\n",
       "    prefix=test_\n",
       "  )\n",
       "  (transform): Identity()\n",
       ")"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from grelu.lightning import LightningModel\n",
    "\n",
    "model = LightningModel.load_from_checkpoint(\"artifacts/DNA-model:v0/reward_model.ckpt\")\n",
    "model.eval()\n",
    "model.to(DEVICE)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(640, 3, 1)\n"
     ]
    }
   ],
   "source": [
    "data_loader = torch.utils.data.DataLoader(valsamples.astype(\"float32\"), batch_size = 128 , num_workers=0)\n",
    "val_samples = []\n",
    "\n",
    "for batch in data_loader:\n",
    "    batch = torch.permute(batch, (0, 2, 1)).to(DEVICE)\n",
    "    val_samples.append(model(batch).detach().cpu() ) \n",
    "\n",
    "val_samples = np.concatenate(val_samples)\n",
    "\n",
    "generated_samples = []\n",
    "data_loader = torch.utils.data.DataLoader(allsamples.astype(\"float32\"), batch_size = 128 , num_workers=0)\n",
    "for batch in data_loader:\n",
    "    batch = (batch > 0.5) * torch.ones_like(batch)\n",
    "    batch = torch.permute(batch, (0, 2, 1)).to(DEVICE)\n",
    "    generated_samples.append( model(batch).detach().cpu() ) \n",
    "\n",
    "generated_samples = np.concatenate(generated_samples)\n",
    "print(generated_samples.shape)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "3.4895487\n",
      "4.535376\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 8000x1000 with 0 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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Tp20YMGCcttWrVp1XHXm5eUpMzNTTz/9tM4++2xJ0ocfflit9+jcubOCwaDWrFkTbulnZmbq+++/Dx+zZs0a2bathx9+WB7PwdGFr732Wrn3iYuLUygUKrdt8+bNysvL09///nelpqZKklavXl2t+lB/HDrvHnvsMY0ZM+aIX3x79eqlPXv2KCYmRm3btj2mv69Lly7y+XzKzs7WgAEDqvXajz76SNdcc034S3xhYaF27txZ7pjKzolevXpp48aN6tChQ6XvW5XzEZHBOOko99hjj6msrEyDBw/WsmXLlJOTo4yMDA0aNEgtW7bUxIkTq/V+N954ozZv3qzx48dry5Yteu211/T8889LUrXC/qcaNWqklJQUzZw5U1lZWXr//fd1++23V+s9OnXqpLS0NN1444369NNPtWbNGl133XXlvph06NBBBw4c0KOPPqrt27frX//6l5588sly79O2bVsVFhZq8eLFys3NVXFxsVq3bq24uLjw695++2397W9/O6afFfXDE088oWAwqD59+mjOnDnatGmTMjMz9eKLL2rz5s3yer2SpIEDB6pv37666KKLtGjRIu3cuVMff/yx7rrrrip/EUxMTNSdd96p2267TbNnz9a2bdu0du1aPfroo5o9e/YRX9uxY0e9+eabWrdundavX6/LL7+83BUu6eA5sWzZMn355ZfKzc2VJI0fP14ff/yxbrnlFq1bt05bt27VW2+9pVtuuUVS1c5HRIjTN8Vx/Hbu3Gmuvvpq06xZMxMbG2tSU1PNmDFjTG5ubrnjKusgUpUhWDNmzDCSTElJiTHm8EOwfmrq1KmmTZs24efvvvuuOeWUU4zP5zM9evQwS5YsKdfprSq9u3fv3m2GDRtmfD6fad26dXjIyk9/pkceecS0aNHC+P1+M3jwYPPCCy+U6/RmjDF/+MMfTEpKSrkhWC+//LJp27at8fl8pm/fvubtt98+aj2o37766itzyy23mHbt2pnY2FjToEEDc8YZZ5h//OMfpqioKHxcQUGBGTNmjDnxxBPD5+fIkSNNdna2MaZq549t22batGmmU6dOJjY21jRp0sQMHjzYLF261BjzY8exn/6eG3PwvDrvvPOM3+83qamp5rHHHjMDBgww48aNCx+zYsUK06NHD+Pz+coNwVq5cqUZNGiQadCggUlISDA9evQwEydODO+vyvmI48dSlTiqiRMn6sknn1ROTo7TpQBAvcI9aVTwxBNP6PTTT1dKSoo++ugj/eMf/whf5gIA1B5CGhVs3bpVDz74oL799lu1bt1ad9xxhyZMmOB0WQBQ73C5GwAAl6J3NwAALkVIAwDgUoQ0AAAuRUgDAOBShDQAAC5FSAMA4FKENFAPnHvuubr11ludLgNANRHSAAC4FCEN1HHXXHONli5dqn/+85+yLEuWZSkmJkZTpkwpd9y6detkWZaysrIkHVz1bMaMGRoyZIj8fr9OOukk/fvf/y73mpycHF166aVq2LChGjdurAsvvLDCUogAjh0hDdRx//znP9W3b19df/312r17t3bv3q37779fs2bNKnfcrFmzdM4555RbQ/juu+/Wb3/7W61fv14jR47U7373O23atEmSdODAAQ0ePFiJiYlavny5PvroIzVo0EBpaWnav39/rf6MQF1FSAN1XHJysuLi4hQIBNS8eXM1b95co0aNUmZmplauXCnpYOC+/PLLuvbaa8u99pJLLtF1112nk08+WX/729/Up08fPfroo5KkOXPmyLZtPfPMM+revbtOOeUUzZo1S9nZ2VqyZElt/5hAnURIA/XQiSeeqGHDhum5556TJKWnp6usrEyXXHJJueP69u1b4fmhlvT69euVlZWlxMRENWjQQA0aNFDjxo1VWlqqbdu21c4PAtRxrIIF1FPXXXedrrzySk2dOlWzZs3SZZddpkAgUOXXFxYWqnfv3nrppZcq7GvSpEkkSwXqLUIaqAfi4uIUCoXKbRs6dKgSEhI0Y8YMZWRkaNmyZRVe98knn+iqq64q97xnz56SpF69emnOnDlq2rSpkpKSavYHAOopLncD9UDbtm316aefaufOncrNzZVt2/J6vbrmmms0YcIEdezYscKlbUl6/fXX9dxzz2nLli269957tXLlSt1yyy2SpJEjR+qEE07QhRdeqOXLl2vHjh1asmSJxo4dqy+++KK2f0SgTiKkgXrgzjvvlNfrVZcuXdSkSRNlZ2dLkkaPHq39+/dr1KhRlb7u/vvv16uvvqoePXrohRde0CuvvKIuXbpIkgKBgJYtW6bWrVvrN7/5jU455RSNHj1apaWltKyBCLGMMcbpIgA4Y/ny5Tr//POVk5OjZs2aldtnWZbmzp2riy66yJniAHBPGqiPysrKtHfvXt1333265JJLKgQ0AHfgcjdQD73yyitq06aNvv/+e/3v//6v0+UAOAwudwMA4FK0pAEAcClCGgAAlyKkAQBwKUIaAACXIqQBAHApQhoAAJcipAEAcClCGgAAl/p/nx5D/DJ5y+cAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 500x500 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "%matplotlib inline\n",
    "import seaborn as sns\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "plt.figure(figsize=(80, 10))\n",
    "\n",
    "compare = np.concatenate( (val_samples[:,0], generated_samples[:,0]  ), axis= 0)\n",
    "\n",
    "type1 = ['Original data' for i in range(len(val_samples[:,0] ))]\n",
    "#type2 = ['Generated (Strong Sig)' for i in range(len(generated_samples[:,0] ))]\n",
    "type3 = ['Generated ' for i in range(len(generated_samples[:,0]))]\n",
    "\n",
    "type =   type1 +  type3\n",
    "data_dict = {'type': type, 'HepG2': compare[:,0] }\n",
    "plot_data = pd.DataFrame(data_dict)\n",
    "fig = sns.catplot(data=plot_data, x = 'type', y =  'HepG2', hue=\"type\", kind=\"boxen\" )\n",
    "sns.set_context(\"paper\", rc={\"figure.figsize\": (80, 10)})\n",
    "#fig.savefig(\"../media/RNA_output_high.png\")\n",
    "\n",
    "print(np.mean(val_samples[:,0]))\n",
    "print(np.mean(generated_samples[:,0]))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Generate normal samples"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "sampler = ddsm.Euler_Maruyama_sampler\n",
    "yS = 2.0 * torch.ones(128)\n",
    "yS = yS.type(torch.LongTensor)\n",
    "yS = yS.to(DEVICE)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|          | 0/100 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [00:03<00:00, 33.22it/s]\n",
      "100%|██████████| 100/100 [00:02<00:00, 33.45it/s]\n",
      "100%|██████████| 100/100 [00:02<00:00, 33.47it/s]\n",
      "100%|██████████| 100/100 [00:03<00:00, 33.02it/s]\n",
      "100%|██████████| 100/100 [00:02<00:00, 33.59it/s]\n",
      "100%|██████████| 100/100 [00:02<00:00, 33.67it/s]\n",
      "100%|██████████| 100/100 [00:03<00:00, 33.27it/s]\n",
      "100%|██████████| 100/100 [00:03<00:00, 33.13it/s]\n",
      "100%|██████████| 100/100 [00:03<00:00, 31.98it/s]\n",
      "100%|██████████| 100/100 [00:02<00:00, 33.54it/s]\n"
     ]
    }
   ],
   "source": [
    "# Generate sequences with y=2 (unconditional)\n",
    "\n",
    "allsamples_original = []\n",
    "for t in range(10):\n",
    "    samples=[]\n",
    "    score_model.eval()\n",
    "    samples.append(sampler(score_model,\n",
    "                        (200,4),\n",
    "                        batch_size=128,\n",
    "                        new_class = yS,\n",
    "                        class_number = 2,\n",
    "                        strength = 2, \n",
    "                        max_time=  4.0,\n",
    "                        min_time= 1.0/400,\n",
    "                        time_dilation=1,\n",
    "                        num_steps=100, \n",
    "                        eps=1e-5,\n",
    "                        speed_balanced= True,\n",
    "                        device= DEVICE, \n",
    "                        ).cpu().detach().numpy())\n",
    "    allsamples_original.append(samples)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(1280, 200, 4)\n"
     ]
    }
   ],
   "source": [
    "allsamples = np.concatenate(allsamples_original, axis=1)\n",
    "allsamples = allsamples[0,:,:,:]\n",
    "print(allsamples.shape)\n",
    "\n",
    "valsamples = np.load(\"artifacts/DNA-dataset:v0/y_HepG2.npz\")['x']\n",
    "# label = np.load(\"artifacts/DNA-dataset:v0/y_HepG2.npz\")['y'] "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "seq_gen_int = np.argmax(allsamples, axis=-1)\n",
    "seq_gen_str = [''.join([['A','C','G','T'][num] for num in seq]) for seq in seq_gen_int]\n",
    "gen_kmers = utils.count_kmers(seq_gen_str)\n",
    "seq_val_int = np.argmax(valsamples, axis=-1)\n",
    "seq_val_str = [''.join([['A','C','G','T'][num] for num in seq]) for seq in seq_val_int]\n",
    "val_kmers = utils.count_kmers(seq_val_str)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA0wAAANTCAYAAACdHZ35AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjkuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/TGe4hAAAACXBIWXMAAA9hAAAPYQGoP6dpAABoHUlEQVR4nO3de3yT5f3/8XfStGmbpmdbWk5VKYoFxhSViSIoqHgA5IeKE0VRRB3fOXGeD8AGurmDziM4RQY6t+HwMCdflSmI06lDEcq3SpUz1GJb2qalTZrm/v1RG1voXRraNGnyej4efYzcuZN+0nvmzjv3dX0ui2EYhgAAAAAAh7CGugAAAAAACFcEJgAAAAAwQWACAAAAABMEJgAAAAAwQWACAAAAABMEJgAAAAAwQWACAAAAABMEJgAAAAAwQWACAAAAABMEJgBAl7nhhht03XXXhboMUw0NDbriiiuUnp6upKQkVVVVhbokAECYIzABQAht2rRJU6dOVU5OjpKSkpSXl6fLL79cn376aahLOyKLFi3SM888E+oyTL300kt69913tX37dtXU1CglJeWQfZ5++mkdf/zxSk1NVVpamk4//XS9++67Iai255g3b55OP/30UJcBAEFBYAKAEFmzZo1OOeUUZWdn68MPP5TL5dKGDRs0btw4rVixItTlBaSxsVE+ny/UZRzW119/rWOOOUbJycmm+4wbN07vvfeeKisrVVZWpp/+9Ke64IIL9O2333ZjpW1raGgIdQkAEHUITAAQIrNmzdKUKVP0hz/8QXl5ebJYLEpNTdWMGTP04IMP+vdbunSpBg8erOTkZA0ePFh/+tOf/Pdt375dFotFS5Ys0dChQ+VwOHT66adr9+7devzxx9W/f3+lpqZq1qxZamxs9D/OYrHo97//vU4++WQlJSXplFNO0X//+1///WvWrNFpp52mjIwMpaWl6ayzztKGDRta3W+xWPSXv/xFAwcOVGJiovbt26err75a06ZNkyQZhqH7779fffr0kdPpVJ8+fXT33Xf7n6OoqEjjx49XZmam+vTpo1mzZrUaIjd69GjdfPPN+vGPf6yUlBT17dtXTz31VLt/0/ae8+qrr9YvfvELffjhh0pKStL48ePbfI6jjz5aWVlZ/tcQExOjuro67dixw/T3Ll26VH369NGTTz6p/v37y+Fw6Oqrr5bL5dKN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AAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYIDABAAAAgAkCEwAAAACYCFlgevzxxzV8+HDZ7XZNnTrVdL///Oc/Ovfcc5WRkaGMjAxdcMEFKi4u7sZKAQAAAESrkAWm3Nxc3XvvvZo5c2a7++3fv18zZszQ1q1bVVJSosGDB2vChAndVCUAAACAaGYL1S+ePHmyJGnDhg0qKysz3W/8+PGtbt9666166KGHVF5eroyMjDYfU1JSopKSkkO2FxUVdaJiAAAAANEmZIHpSK1du1a9evUyDUuStHjxYs2fP78bqwIAAAAQiXpUYNq6datmz56tRx99tN39Zs2a1eawvaKiIk2bNi1Y5QEAAACIMD0mMO3atUtjx47VHXfcocsuu6zdfXNycpSTk9NNlQEAAACIVD2irfju3bt11lln6frrr9ecOXNCXQ4AAACAKBGyK0xer9f/4/P5VF9fr5iYGMXGxrbab+/evRozZoymTZumO++8M0TVAgAAAIhGIbvCtGDBAiUkJGjhwoVasWKFEhIS/C3Gk5KStG7dOknSH//4R3311Vf6zW9+o6SkJP/Pzp07Q1U6AAAAgChhMQzDCHUR3eXTTz/VSSedpPXr1+vEE08MdTkAAAAAQqSj2aBHzGECAAAAgFAgMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJggMAEAAACACQITAAAAAJgIWWB6/PHHNXz4cNntdk2dOrXdfdeuXavBgwcrMTFRJ598sj7//PNuqhIAAABANAtZYMrNzdW9996rmTNntrtfeXm5Jk6cqNtvv1379+/X5ZdfrgkTJsjtdndTpQAAAACiVcgC0+TJkzVp0iRlZma2u9/KlSs1YMAAXXXVVbLb7brlllvk8/m0evXqbqoUAAAAQLSyhbqAwyksLNSwYcP8ty0Wi4YOHarCwkJdcMEFbT6mpKREJSUlh2wvKioKVpkAAAAAIlDYB6aamhqlpaW12paamiqXy2X6mMWLF2v+/PnBLg0AAABAhAv7wJSUlKSqqqpW26qqquR0Ok0fM2vWLE2YMOGQ7UVFRZo2bVqX1wgAAAAgMoV9YBo8eLCefvpp/23DMLRx40bdeOONpo/JyclRTk5Od5QHAAAAIIKFrOmD1+tVfX29vF6vfD6f6uvr1dDQcMh+kydPVnFxsZ5//nl5PB794Q9/kCSNHTu2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ZZv7r127VkVFRXr22Wc7FJgWL16s+fPnd1m9QDRy2G3KctoPmcO0Z/8BpSXGhbo8AACAbhGSwFRTU6OUlNYTxlNTU+VyuQ7Z1+Px6Cc/+YmWLVsmq7Vj32jPmjVLEyZMOGR7UVGRpk2bdmRFA1EoP9spSdrncsvb2Kg9lfWSIdlirPp4WwUd8wAAQMQLSWBKSkpSdXV1q21VVVVyOp2H7PvQQw9p9OjROvHEEzv8/Dk5OcrJyel0nUC0i7NZVdA7RXlurzbuqlR2spSbmkDHPAAAEDVCEpgGDhwoi8WizZs3+5s4bNiwodUQvWarV6/Wpk2b9NJLL0mSKioq9Nlnn+mjjz7Sc8891611A9Gs3uvzhyWpdce8PLeXBW0BAEBECsknHIfDoSlTpuiee+7R8uXLtWPHDj3zzDNtBqCVK1fK4/H4b0+ePFmTJk3S9ddf350lA1GNjnkAACBahWziwRNPPKHY2Fjl5ORo3LhxuvPOO/0d8pKSkrRu3TpJUnp6unr16uX/iYuLU3JyslJTU0NVOhB1WnbMa4mOeQAAINKFbAxNamqqVqxY0eZ9NTU1po9bs2ZNkCoCYMasY155jVvZyfEMxwMAABGLTzkAOqRlxzyp6cpSdnK8fzsAAEAkIjAB6JCWHfM8Xp/ibFauLAEAgIjHpx0AAXHYbTR4AAAAUYOZ2gAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABggsAEAAAAACYITAAAAABgImSBqbKyUpdeeqmcTqdyc3P1yCOPtLnff/7zH5177rnKyMhQRkaGLrjgAhUXF3dvsQAAAACiUsgC0+zZs+V2u7Vnzx69+eabeuCBB7Rq1apD9tu/f79mzJihrVu3qqSkRIMHD9aECRNCUDEAAACAaGMLxS+tra3VihUrtH79eiUnJ2vIkCGaOXOmlixZovHjx7fa9+Dbt956qx566CGVl5crIyOjzecvKSlRSUnJIduLioq67kUAAAAAiHghCUxbtmyRz+fT4MGD/duGDRumlStXHvaxa9euVa9evUzDkiQtXrxY8+fP75JaAQAAAESvkASmmpoapaSktNqWmpoql8vV7uO2bt2q2bNn69FHH213v1mzZrU5bK+oqEjTpk0LvGAAAAAAUSkkgSkpKUnV1dWttlVVVcnpdJo+ZteuXRo7dqzuuOMOXXbZZe0+f05OjnJycrqkVgAAAADRKyRNHwYOHCiLxaLNmzf7t23YsKHVEL2Wdu/erbPOOkvXX3+95syZ011lAgAAAIhyIQlMDodDU6ZM0T333COXy6XCwkI988wzmjFjxiH77t27V2PGjNG0adN05513hqBaAAAAANEqZG3Fn3jiCcXGxionJ0fjxo3TnXfe6e+Il5SUpHXr1kmS/vjHP+qrr77Sb37zGyUlJfl/du7cGarSAQAAAESJkMxhkpqaPKxYsaLN+2pqavz/njt3rubOndtdZQEAAACAX8iuMAEAAABAuCMwAQAAAIAJAhMAAAAAmCAwAQAAAIAJAhMAAAAAmCAwAQAAAIAJAhMAAAAAmCAwAQAAAIAJAhMAAAAAmLCFugBEtlq3Vx6vT3E2qxx2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99xjXHLJJUZdXZ3h8XgO2Y9zevfp6DHhvN49Ono8OK93TsQFpv379xtTpkwxHA6H0atXL+Phhx/23+dwOIz33nvPf/vdd981CgoKjPj4eGP48OHGZ5991v0FR7iOHo958+YZkgyHw9HqZ8eOHSGqPDIF8t9HS3TTCZ5AjklRUZFx+umnG4mJicaAAQOMJUuWhKDiyBbI8XjqqaeM/Px8Iykpyejbt69x1113GY2NjSGoOnLNnTvXkNTqZ/r06YZhcE4PlY4eE87r3SOQ/0Za4rweGIthGEYormwBAAAAQLiLqDlMAAAAANCVCEwAAAAAYILABAAAAAAmCEwAAAAAYILABAAAAAAmCEwAAAAAYILABAAAAAAmCEwAgLCUl5enZ555JtRlAADCwOOPP67hw4fLbrdr6tSpAT1248aNOuuss5SUlKSjjjpKd999d0CPJzABAFpJSkry/8TFxSkmJqbVtnXr1oW6RLRh6dKl6tOnT6jLAICgyM3N1b333quZM2cG9LiKigqNGzdO06dPV1lZmXbs2BFw4CIwAQBaqamp8f/cfvvtOuOMM1ptO+OMM/z7ejyeEFYaGg0NDaEuAQCizuTJkzVp0iRlZmYect8nn3yiUaNGKS0tTYMGDdLKlSv99/3+97/X2LFjNX36dMXHxysxMVFDhw4N6HcTmAAAHTZv3jydfvrpuu+++5Sbm6thw4ZJkiwWi1avXu3fb/v27bJYLPrqq6/829544w2deuqpSktLU35+vh599NEO/16v16sbbrhBP/zhD7V79+4291mzZo0sFov++te/auDAgUpMTNRFF12kyspK3XfffcrJyVFmZqbmzp3b6nFffPGFLrzwQmVnZ6t379666aabVFtb678/Ly9Pc+fO1XnnnSen06nf/e53bf7+3bt364orrlCfPn2UnJysYcOG6dNPP5Uk1dfX64477tDRRx+ttLQ0nXHGGfroo48O+bu2dPXVV2vatGmt6vjFL36h888/X06nU8cee6xefvllSdK6det0ww03aO/evf4rgS+88EKH/74A0FOVlJTovPPO05w5c1RWVqalS5fquuuuU1FRkSTpww8/VGZmpk4//XRlZmZq3Lhx/vs6isAEAAjIf/7zH8XGxmrr1q3673//26HHvPvuu/rxj3+sBx54QOXl5Xr55Zf1m9/8pkMf6vfv36/zzjtPe/fu1bp16w477Oyf//yn1q9fr+3bt+vLL7/UqaeequzsbO3atUuvv/66Fi5cqA8//FCSVFZWpjPOOENnn322du7cqc8//1xbtmzRz372s1bPuXjxYs2dO1fV1dX66U9/esjvrKur01lnnaW4uDht2LBBlZWV+vOf/6yMjAxJ0m233aY33nhDb7/9tkpLSzVp0iSNHTvWNPyZeeaZZ7RgwQJVVVXpJz/5iaZPn67q6mqdccYZWrRokXJzc/1XAq+44oqAnhsAeqLly5dr7NixmjRpkmJiYnTqqafq4osv1ooVKyRJu3bt0nPPPaff/va32rt3r0aOHKmJEyfK6/V2+HcQmAAAAcnOztZ9993nH9rQEQ8//LBuvPFGnX322bJarRo8eLBuuOEGPffcc+0+rri4WCNGjNDQoUP1yiuvKCkp6bC/a+HChXI6ncrKytIFF1wgSZo9e7ZsNptGjBihQYMG6eOPP5YkLVu2TAMGDNAtt9wiu92uzMxMzZ8/X8uWLVNjY6P/Oa+55hr96Ec/ksViafM1//Of/1R5ebkWLVqkzMxMWa1WnXDCCerfv798Pp+effZZLViwQAMGDFBcXJxuvfVWHXPMMXr++ec79Pdrdt111+nEE0+U1WrVrFmz5HK5Av6mFAAiyfbt2/Xqq68qNTXV//PXv/5VJSUlkqTExERNnDhRI0aMUFxcnO677z7t3LlTX375ZYd/hy1YxQMAIlP//v1lsVgCekxxcbFWr16tp556yr+tsbFR/fr1a/dxy5YtU2xsrO6++25Zrd9/x3fDDTf4w0b//v21efNm/305OTn+fzscjla3m7e5XC5/XevXr1dqaqr/fsMwZLFY9M0336h3796SpKOPPrrdOrdt26a8vDzZ7fZD7isrK1NdXZ2OPfbYVtsHDBignTt3tvu8B8vNzW31OiT5XwsARKN+/fpp6tSpWrp0aZv3BzpfqS1cYQIABKRlcGmWlJTUat7P3r17W93fq1cv3XnnnaqsrPT/uFyuVkGnLfPnz9f48eN1xhlnaNeuXf7tixYt8g89O9xztKdXr146/fTTW9VVVVWl+vp6f1iS2n7NLeXl5Wn79u1tNsHIzMxUfHy8vv7661bbv/76a39gdDqdrf5+0qF/w8M5XI0A0JN5vV7V19fL6/XK5/Opvr5eDQ0NmjZtmlatWqV//OMf8nq98ng8+uijj/xX36+99lq9+uqr+u9//yuv16sHHnhAeXl5Ou644zr8u3l3BQB02vDhw7V06VLV19ertLRU8+fPb3X/zTffrMcee0z/+te/5PV65fV6VVhYqPfee6/d57VarVq8eLEuvvhijRw5Ul988UWX1n3NNdfos88+05NPPqkDBw7IMAzt2rVLr7zySkDPc+GFFyotLU033XSTysrKZBiG/u///k87duyQ1WrVjBkzdP/992vr1q3yeDx6+OGH9dVXX/nnGQ0fPlybNm3S+++/r8bGRq1YseKwf5uD9erVS2VlZSovLw/ocQDQEyxYsEAJCQlauHChVqxYoYSEBM2cOVN9+vTRG2+8oUceeUTZ2dnKzc3VXXfdJbfbLUk688wz9bvf/U6TJ09WRkaG1qxZo1deeUU2W8cH2hGYAACd9sQTT+ibb77xdyC68sorW90/adIkLV++XPfff7+ysrKUlZWl6667TmVlZR16/gceeEC33HKLRo0apU8++aTL6u7Xr58+/PBDvf322zr22GOVmpqqc889V5s2bQroeRISEvTOO++opqZGQ4YMUUpKiq644gpVVFRIkn7729/qnHPO0ZgxY5SVlaW///3vevvtt9W3b19JTSf0u+++W5MnT9ZRRx2lNWvW6P/9v/8XUA1nnXWWJk6cqIEDByo1NVV//vOfA3o8AISzefPmyTCMVj/Nw/BOOukk/etf/1J5ebnKysr0zjvv+Lu4Sk3zP3fu3Kmqqir961//0vHHHx/Q77YYhmF04WsBAAAAgIjBFSYAAAAAMEFgAgAAAAATBCYAAAAAMEFgAgAAAAATBCYAAAAAMEFgAgAAAAATBCYAAAAAMEFgAgAAAAATBCYAAAAAMEFgAgAAAAATBCYAAAAAMEFgAgAAAAAT/x/LhEe1M7D1IwAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 1000x1000 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/plain": [
       "PearsonRResult(statistic=0.9746297954070254, pvalue=5.1155880885736966e-42)"
      ]
     },
     "execution_count": 16,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "kmer_set = set(gen_kmers.keys()) | set(val_kmers.keys())\n",
    "counts = np.zeros((len(kmer_set), 2))\n",
    "for i, kmer in enumerate(kmer_set):\n",
    "    if kmer in gen_kmers:\n",
    "        counts[i][1] = gen_kmers[kmer] * valsamples.shape[0]/allsamples.shape[0]\n",
    "    if kmer in val_kmers:\n",
    "        counts[i][0] = val_kmers[kmer]\n",
    "        \n",
    "fig, ax = plt.subplots(figsize=(10, 10))\n",
    "ax.scatter(counts[:, 0], counts[:, 1], alpha=0.2)\n",
    "ax.set_title(\"Comparison of 3-mer counts\")\n",
    "ax.set_xlabel(\"True k-mer count\")\n",
    "ax.set_ylabel(\"Generated k-mer count\")\n",
    "ax.set_ylim((-5, np.max(counts) + 5))\n",
    "ax.set_xlim((-5, np.max(counts) + 5))\n",
    "plt.show()\n",
    "\n",
    "sp.stats.pearsonr(counts[:, 0], counts[:, 1])"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "diffusion",
   "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.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
