{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/ueharam1/miniconda3/envs/testgrelu/lib/python3.12/site-packages/tqdm/auto.py:21: TqdmWarning: IProgress not found. Please update jupyter and ipywidgets. See https://ipywidgets.readthedocs.io/en/stable/user_install.html\n",
      "  from .autonotebook import tqdm as notebook_tqdm\n",
      "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[33mmasatoshi136\u001b[0m (\u001b[33mmasa136\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 sys \n",
    "sys.path.append(\"../../\")\n",
    "sys.path.append(\"../../src/\")\n",
    "sys.path.append(\"../../src/model\")\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",
    "\n",
    "import scipy as sp\n",
    "from matplotlib import pyplot as plt\n",
    "\n",
    "import wandb\n",
    "wandb.login(host=\"https://api.wandb.ai\") \n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Load pre-trained model "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "wandb version 0.17.5 is available!  To upgrade, please run:\n",
       " $ pip install wandb --upgrade"
      ],
      "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/ueharam1/prj/RLfinetuning_Diffusion_Bioseq/tutorials/Human-enhancer/wandb/run-20240720_115426-3xukz9af</code>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "Syncing run <strong><a href='https://wandb.ai/masa136/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer/runs/3xukz9af' target=\"_blank\">honest-sun-15</a></strong> to <a href='https://wandb.ai/masa136/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/masa136/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer' target=\"_blank\">https://wandb.ai/masa136/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/masa136/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer/runs/3xukz9af' target=\"_blank\">https://wandb.ai/masa136/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer/runs/3xukz9af</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.1\n"
     ]
    },
    {
     "data": {
      "text/html": [
       " View run <strong style=\"color:#cdcd00\">honest-sun-15</strong> at: <a href='https://wandb.ai/masa136/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer/runs/3xukz9af' target=\"_blank\">https://wandb.ai/masa136/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer/runs/3xukz9af</a><br/> View project at: <a href='https://wandb.ai/masa136/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer' target=\"_blank\">https://wandb.ai/masa136/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer</a><br/>Synced 6 W&B file(s), 0 media file(s), 0 artifact file(s) and 0 other file(s)"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "Find logs at: <code>./wandb/run-20240720_115426-3xukz9af/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": [
    "run = wandb.init() \n",
    "artifact = run.use_artifact('fderc_diffusion/Diffusion-DNA-RNA/DNA-model:v0')\n",
    "dir = artifact.download()\n",
    "wandb.finish()\n",
    "\n",
    "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_unconditional_epoch=054.ckpt'\n",
    "\n",
    "config = ModelParameters() \n",
    "DEVICE = \"cuda:4\" # Any number is fine\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "ScoreNet(\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",
       "  (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",
       ")"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Introduce Two Models\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 =1)\n",
    "score_model = score_model.model\n",
    "score_model.cuda(device = DEVICE) \n",
    "\n",
    "\n",
    "original_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 =1)\n",
    "original_model = original_model.model\n",
    "original_model.cuda(device = DEVICE) "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "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:1.3\n"
     ]
    }
   ],
   "source": [
    "# Load Reward model\n",
    "from grelu.lightning import LightningModel\n",
    "model = LightningModel.load_from_checkpoint(\"artifacts/DNA-model:v0/reward_model.ckpt\")\n",
    "model.eval()\n",
    "model.to(DEVICE)\n",
    "\n",
    "def new_reward_model(x):\n",
    "    x = torch.nn.functional.softmax(x /0.1, -1)\n",
    "    seq = torch.transpose(x, 1, 2) \n",
    "    return model(seq)[:,0]\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "wandb version 0.17.5 is available!  To upgrade, please run:\n",
       " $ pip install wandb --upgrade"
      ],
      "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/ueharam1/prj/RLfinetuning_Diffusion_Bioseq/tutorials/Human-enhancer/wandb/run-20240720_115500-xz47ax9b</code>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "Syncing run <strong><a href='https://wandb.ai/masa136/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer/runs/xz47ax9b' target=\"_blank\">visionary-pine-16</a></strong> to <a href='https://wandb.ai/masa136/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/masa136/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer' target=\"_blank\">https://wandb.ai/masa136/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/masa136/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer/runs/xz47ax9b' target=\"_blank\">https://wandb.ai/masa136/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer/runs/xz47ax9b</a>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 50/50 [00:02<00:00, 21.28it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.54it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.89it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.11it/s]\n",
      "100%|██████████| 50/50 [00:03<00:00, 15.36it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 58.19it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 55.09it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.42it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.62it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.16it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.05it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.06it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 54.83it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 54.31it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.33it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 58.36it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 53.19it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.43it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.86it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.21it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.57it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 55.07it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 55.93it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 55.79it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.27it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.29it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.42it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.12it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.10it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.13it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.62it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.44it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 54.92it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 58.56it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.28it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 56.60it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.28it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.99it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.32it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.38it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.93it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 55.67it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 54.36it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.71it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.26it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.31it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.22it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.86it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.06it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.99it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.50it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 55.26it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.40it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.32it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.61it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.94it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.80it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.68it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.06it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.22it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.87it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.80it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 53.69it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 54.12it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.13it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 54.24it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.80it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.61it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.62it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.41it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.52it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.16it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 53.26it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 58.20it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.41it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 55.23it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 55.28it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.15it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.14it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.43it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.74it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.67it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 57.72it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 53.66it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.45it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.09it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.52it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.82it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 55.94it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.94it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.61it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.53it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.11it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 56.03it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.35it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 60.33it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 57.99it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 53.37it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.78it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.07it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.37it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 56.13it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.49it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 54.26it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 19.08it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.52it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.43it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.55it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.07it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.84it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 42.43it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 42.79it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.52it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.84it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.83it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.36it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.10it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 41.40it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.32it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.96it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 40.81it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.04it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 41.34it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 37.35it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.93it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.91it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.96it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.18it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 42.25it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.20it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 44.65it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.40it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.68it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 41.05it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.79it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.32it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.91it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.38it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.79it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.96it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 40.98it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.08it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.15it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.21it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.55it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 44.96it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.36it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 39.24it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.25it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.55it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.38it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.22it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 44.88it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.78it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.84it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 38.46it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.67it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.62it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.23it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.92it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 41.67it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.17it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.81it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.00it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.98it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.97it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 56.71it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.59it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.07it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.20it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.39it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.43it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.29it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 38.77it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.18it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.61it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.33it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 54.12it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.24it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.03it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.33it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.38it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.89it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.36it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.89it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.68it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.70it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.16it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.43it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.54it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 44.20it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.96it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 40.59it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.99it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.78it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 44.01it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.65it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.91it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.92it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.84it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.56it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.79it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.87it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.97it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.77it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.98it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 42.22it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 56.72it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.61it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.86it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.09it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.43it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.73it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.05it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.01it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.26it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.58it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.43it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 44.32it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.04it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 42.17it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 44.37it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 42.90it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.05it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.05it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.61it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.93it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 41.32it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 42.90it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.31it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.04it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 44.19it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.27it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.35it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.10it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.21it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.14it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.77it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.86it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.12it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 56.03it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.87it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 42.79it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.00it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.82it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.69it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.53it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.94it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.55it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.24it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.78it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.35it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.78it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.08it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.16it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.76it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 39.65it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.87it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.94it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.27it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.11it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.19it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 44.42it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.46it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.49it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.03it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.66it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.37it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.55it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.85it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.52it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.54it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.19it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.20it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.88it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 54.72it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.23it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.63it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 44.81it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.95it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.23it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.49it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.72it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.66it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.86it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.65it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.01it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.78it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.29it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.12it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.45it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.52it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.11it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.34it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.16it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 44.29it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.92it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.04it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.59it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.24it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.47it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 41.09it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.84it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 57.31it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.98it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.65it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.69it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.50it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 41.89it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.89it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.34it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 53.97it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.95it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.18it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.70it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.12it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.54it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.47it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.57it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.01it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.56it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.95it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.14it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 41.55it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.80it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.60it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.18it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 36.56it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.19it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.92it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 42.61it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.87it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.26it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.62it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.41it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 42.79it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.65it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.52it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.38it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.57it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 54.11it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.32it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.42it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.80it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.92it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 53.24it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.31it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.90it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.85it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.92it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.52it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.03it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.31it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 54.24it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.70it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 53.13it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.98it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.21it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 44.05it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.50it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 44.14it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.26it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.42it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.16it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.83it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.24it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.20it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.34it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.75it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.82it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.66it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.24it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.25it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.85it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.87it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.03it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.91it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.14it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 56.67it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.32it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.30it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.02it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.98it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.42it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.99it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.26it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.31it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 53.06it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.66it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.02it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.21it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.38it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.78it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 54.10it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.20it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 53.54it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.72it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.88it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 39.22it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.94it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 44.42it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 56.64it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.88it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.80it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.61it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.96it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.34it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.56it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.74it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.06it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.85it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.38it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 41.56it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.22it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.07it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.76it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.29it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 41.25it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.77it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.90it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.74it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.55it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.42it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 41.55it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.99it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.98it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.70it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.25it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.31it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.08it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.99it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 45.87it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.07it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.12it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.83it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.71it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.10it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.33it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.40it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.23it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 54.76it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.07it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.55it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.89it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.30it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.55it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 53.17it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.77it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 54.50it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.14it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.46it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.25it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.30it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.95it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.94it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.06it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 53.49it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.01it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.91it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.11it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 56.96it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.57it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.55it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.57it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.22it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 47.54it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.75it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.40it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.51it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.04it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.83it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 44.11it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.34it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 41.62it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.84it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 54.47it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.80it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.92it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.66it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.09it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 43.31it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 44.53it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.39it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.40it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.30it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 48.94it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.28it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 56.51it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.08it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.41it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.62it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 52.37it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.05it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 49.26it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 17.82it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 50.17it/s]\n",
      "100%|██████████| 50/50 [00:01<00:00, 46.23it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.38it/s]\n",
      "100%|██████████| 50/50 [00:00<00:00, 51.52it/s]\n",
      "100%|██████████| 50/50 [00:02<00:00, 18.48it/s]\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<style>\n",
       "    table.wandb td:nth-child(1) { padding: 0 10px; text-align: left ; width: auto;} td:nth-child(2) {text-align: left ; width: 100%}\n",
       "    .wandb-row { display: flex; flex-direction: row; flex-wrap: wrap; justify-content: flex-start; width: 100% }\n",
       "    .wandb-col { display: flex; flex-direction: column; flex-basis: 100%; flex: 1; padding: 10px; }\n",
       "    </style>\n",
       "<div class=\"wandb-row\"><div class=\"wandb-col\"><h3>Run history:</h3><br/><table class=\"wandb\"><tr><td>reward</td><td>▁▁▅▂▆▄▆▆▆▆▆▆▇▅▅▆▇▇▇▇▇▇▇▇▇▆▇▇▇▅▇██▇▇███▇▇</td></tr><tr><td>step</td><td>▁▁▁▁▂▂▂▂▂▃▃▃▃▃▃▄▄▄▄▄▅▅▅▅▅▅▆▆▆▆▆▇▇▇▇▇▇███</td></tr></table><br/></div><div class=\"wandb-col\"><h3>Run summary:</h3><br/><table class=\"wandb\"><tr><td>reward</td><td>5.73917</td></tr><tr><td>step</td><td>399</td></tr></table><br/></div></div>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       " View run <strong style=\"color:#cdcd00\">visionary-pine-16</strong> at: <a href='https://wandb.ai/masa136/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer/runs/xz47ax9b' target=\"_blank\">https://wandb.ai/masa136/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer/runs/xz47ax9b</a><br/> View project at: <a href='https://wandb.ai/masa136/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer' target=\"_blank\">https://wandb.ai/masa136/RLfinetuning_Diffusion_Bioseq-tutorials_Human-enhancer</a><br/>Synced 6 W&B file(s), 0 media file(s), 0 artifact file(s) and 0 other file(s)"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "Find logs at: <code>./wandb/run-20240720_115500-xz47ax9b/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": [
    "from src.model import ddsm_fine_tune  as fine_tune\n",
    "\n",
    "import os\n",
    "save_name = \"./log_finetune/\"\n",
    "isExist = os.path.exists(save_name)\n",
    "if not isExist:\n",
    "   os.makedirs(save_name)\n",
    "\n",
    "loss_curves, eval_curves = fine_tune.fine_tuning(score_model, new_reward_model, [new_reward_model], original_model,\n",
    "            learning_rate =1e-3, num_epoch = 400, length = 200, num_steps = 50, accmu = 4, gradient_start = 45, \\\n",
    "            batch_size = 128, save_name = save_name, entropy_coff = 0.0,  device= DEVICE)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Evaluation "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "ScoreNet(\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",
       "  (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",
       ")"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "#### Load Fine-Tuned Model \n",
    "time_dependent_weights = torch.tensor(np.load(config.time_schedule)['x'])\n",
    "score_finetuned_model = modeld.ScoreNet(time_dependent_weights=torch.sqrt(time_dependent_weights)) \n",
    "score_finetuned_model.load_state_dict(torch.load(\"./log_finetune/_399.pth\")) # Change the path to the fine-tuned model\n",
    "score_finetuned_model.cuda(device = DEVICE) "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [00:01<00:00, 53.42it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 54.42it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 56.09it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 56.68it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 56.58it/s]\n"
     ]
    }
   ],
   "source": [
    "### Get Samples from Fine-tuned Models \n",
    "\n",
    "sampler = ddsm.Euler_Maruyama_sampler\n",
    "\n",
    "\n",
    "allsamples_original = []\n",
    "for t in range(5):\n",
    "    samples=[]\n",
    "    score_finetuned_model.eval()\n",
    "    samples.append(sampler(score_finetuned_model,\n",
    "                        (200,4),\n",
    "                        batch_size=128,\n",
    "                        new_class = None,\n",
    "                        class_number = 1,\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.append(samples)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "allsamples = np.concatenate(allsamples_original, axis=1)\n",
    "allsamples = allsamples[0,:,:,:]\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)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [00:01<00:00, 52.67it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 56.57it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 55.69it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 56.73it/s]\n",
      "100%|██████████| 100/100 [00:01<00:00, 56.04it/s]\n"
     ]
    }
   ],
   "source": [
    "### Get Samples from Pre-Trained Models \n",
    "\n",
    "sampler = ddsm.Euler_Maruyama_sampler\n",
    "\n",
    "allsamples_original = []\n",
    "for t in range(5):\n",
    "    samples=[]\n",
    "    score_model.eval()\n",
    "    samples.append(sampler(original_model,\n",
    "                        (200,4),\n",
    "                        batch_size=128,\n",
    "                        new_class = None,\n",
    "                        class_number = 1,\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.append(samples)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "allsamples = np.concatenate(allsamples_original, axis=1)\n",
    "allsamples = allsamples[0,:,:,:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "generated_samples_before = []\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_before.append(model(batch).detach().cpu() ) \n",
    "\n",
    "generated_samples_before = np.concatenate(generated_samples_before)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<Figure size 8000x1000 with 0 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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9fIsa9++vVJSUrR+/XpdffXVFqaDt1j+OWtJGjJkiL7++murYwCAX8nNzVVGRoa6du1aaXlgYKC6du2qrVu3WpQM3mZEWQMAaq6srEwBAQEKCQmp8lx4eLjKy8stSAVfoKwBwE/VrVtXTqdT27dvr/JcWlqaWrRoYUEq+AJlDQB+KiAgQFdffbVSU1O1detWVVRUqKioSKtXr9Z3332nHj16WB0RXsL9rAHAj3Xu3FmStHLlSi1YsEAul0tNmzbV8OHDFRsba3E6eAtljTPiwIED2rp1q0pLS9W8eXO1bdtWAQEM7ADe0LlzZ1100UXKyclRcHCwoqKirI4EL6Os4VNut1urVq3SunXr1KFDB0VERGjp0qVyOBwaPny47Ha71RGBWiEgIED16tWzOgZ8hLKGT23fvl0bN27UP//5T9WvX1+S1KdPH82fP1+pqam64447LE4IAOZjHBI+9fXXX6t79+6eopakoKAg9enTR99//70KCgosTAcA/oGyhk8dOXJE55xzTpXlderUUUhIiPLz8y1IBQD+hbKGT9WrV0+7d++usvzgwYMqKytTTEzMmQ8F1EIVFRU6cOCADh8+bHUU+ADvWcOnunbtqnnz5qldu3aKj4+XJBUXF2vp0qW64IILFBERYXFCwP99+eWXWrNmjY4dOyaXy6W4uDglJSWpadOmVkeDl1DW8KnWrVurV69emj59upo2bSq73a6dO3cqISFBSUlJVscD/N5XX32l1atXq1+/fmrbtq1KS0v15Zdf6rXXXtN9993HDZJqCcoaPtezZ09deOGF2rZtm0pKStS9e3c1adJENpvN6miAX6uoqNCaNWt02223qX379pKksLAwXXHFFcrJydFnn32m/v37W5wS3kBZ44yIjIyscmcgAKcnJydHhYWFateuXZXnzj//fKWmplqQCr7ABDMA8FMhISFyuVwqLi6u8lxRUZGCg4MtSAVfqHFZl5SUaOvWrTp27FiV5xYuXOiVUACAvxcVFaUmTZpo/fr1lZZXVFTo888/1wUXXGBNMHhdjcr6hx9+ULNmzdSxY0c1aNBAc+bMqfT8iBEjvBoOAHBySUlJWr9+vRYtWqT09HRt2bJFM2bMUFlZmbp162Z1PHhJjcp6/PjxSk5O1pEjRzRr1iz961//0iuvvOJ53u12ez0gAODEGjdurAceeEDBwcFavny5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ZIjy8vI827LZbJoxY4batGmj6OhoSdKmTZvUvXt3xcTEqG3btlq6dGmNsw8ZMkTjx4+XJH366adq2LChXnrpJcXFxSk2NlbPPvusZ123262pU6eqVatWqlOnjq677jr99ttv1d7X7t27ZbPZVFxc7Fk2YMAATZo0qVr7nzRpkgYMGFBpW/PmzVOzZs0UExOjBx980LOuy+XS+PHjVb9+fSUkJOj111+XzWbz3LIV/mf+/Plq0aKFxo0bpzlz5niW//V4atmypdavX68xY8bI4XBo8ODBkqQDBw6of//+atCggeLj4zVp0iS5XC5J0ty5c3XJJZfo3//+t2JjY3XvvfdWO9cfv7d/aNq0qf773/+qU6dOioyM1HXXXacjR454nj/d47Znz56aOXOm5/GHH36opk2bVmv/fz0Ge/bsqUcffVSXX365nE6nunbtqj179ni2tXbtWrVr105Op1N33HGHbr75Zs/xiqooa8NVVFTojTfeUEJCgho1aiRJWrFihf7zn/9o4cKFOnz4sAYNGqQbb7xRpaWlmjdvnpKTkz1nurfccouefvppffzxx9q4caN27typnJwc/c///E+l/bzzzjtat26dsrOzlZmZqd69e2vs2LE6dOiQ5s6dq6FDhyo9Pf20fpZDhw4pIyNDe/bs0cqVK/XII4/op59+kiRNnz5db731lj766CNlZWXpwgsv9JSnt5xs/8ezZs0abd++XZs3b9acOXO0du1aSdLs2bOVmpqqTZs26YcffuCMvhaYM2eOBg0apOTkZG3btk3ffvutJFU5nnbt2qXLLrvMM/I1b948uVwu9enTRy1atNCePXu0YcMGLVu2TLNnz/Zs/5tvvlG9evW0b98+TZs27bSyzp8/X++++67279+v3NxcPf/885Lks+O2uvs/njfffFMvvfSSDh8+rLi4OD366KOSpJycHCUlJek///mPjhw5ot69e2vFihVezVnbUNaGGjt2rKKjo2W32/Xggw9q8uTJCgr6/Y6mr7zyiv7973/rvPPOU2BgoEaMGCGbzaavv/76uNuaP3++Jk6cqEaNGikqKkrPPPOM3n77bZWWlnrWGT9+vGJjYxUeHq558+apV69eSkpKUmBgoLp06aK+fftq8eLFJ8w7bdo0RUdHe75ycnKqrBMQEKCnnnpKISEh6ty5s9q0aaO0tDTPz/TUU0+p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",
      "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( (generated_samples_before[:,0], generated_samples[:,0]  ), axis= 0)\n",
    "\n",
    "type1 = ['Before Fine-Tuning' for i in range(len(generated_samples_before[:,0] ))]\n",
    "#type2 = ['Generated (Strong Sig)' for i in range(len(generated_samples[:,0] ))]\n",
    "type3 = ['After Fine-Tuning ' 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/DNA_output_high_finetune.png\")\n",
    "\n",
    "\n",
    "\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "GRELU1",
   "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
}
