{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/local/home/antonis/.conda/envs/neuroformer_clean/lib/python3.11/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"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Running in Jupyter\n",
      "CONTRASTIUVEEEEEEE False\n",
      "VISUAL: True\n",
      "PAST_STATE: True\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "11/22/2023 17:38:37 - INFO - neuroformer.model_neuroformer -   number of parameters: 2.510515e+07\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "256 9\n",
      " ///// <=----- Loading model from ./models/NF.15/Visnav_VR_Expt/lateral/Neuroformer/demo/(state_history=6,_state=6,_stimulus=6,_behavior=6,_self_att=6,_modalities=(n_behavior=25))/25 -----=> \\\\\\\n"
     ]
    }
   ],
   "source": [
    "import glob\n",
    "import os\n",
    "\n",
    "import sys\n",
    "import glob\n",
    "from pathlib import Path, PurePath\n",
    "path = Path.cwd()\n",
    "parent_path = path.parents[1]\n",
    "sys.path.append(str(PurePath(parent_path, 'neuroformer')))\n",
    "sys.path.append('neuroformer')\n",
    "\n",
    "import pandas as pd\n",
    "import numpy as np\n",
    "\n",
    "import numpy as np\n",
    "import torch\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "from scipy.stats import pearsonr\n",
    "from torch.utils.data.dataloader import DataLoader\n",
    "\n",
    "from neuroformer.model_neuroformer import load_model_and_tokenizer\n",
    "from neuroformer.utils import get_attr\n",
    "from neuroformer.utils import (set_seed, running_jupyter, \n",
    "                                 all_device, recursive_print,\n",
    "                                 create_modalities_dict)\n",
    "from neuroformer.datasets import load_visnav, load_V1AL\n",
    "\n",
    "parent_path = os.path.dirname(os.path.dirname(os.getcwd())) + \"/\"\n",
    "\n",
    "# set up logging\n",
    "import logging\n",
    "logging.basicConfig(\n",
    "        format=\"%(asctime)s - %(levelname)s - %(name)s -   %(message)s\",\n",
    "        datefmt=\"%m/%d/%Y %H:%M:%S\",\n",
    "        level=logging.INFO,\n",
    ")\n",
    "\n",
    "from neuroformer.default_args import DefaultArgs, parse_args\n",
    "\n",
    "if running_jupyter(): # or __name__ == \"__main__\":\n",
    "    print(\"Running in Jupyter\")\n",
    "    args = DefaultArgs()\n",
    "    # args.dataset = \"medial\"\n",
    "    # args.ckpt_path = \"./models/NF.15/Visnav_VR_Expt/medial/Neuroformer/predict_all_behavior/(state_history=6,_state=6,_stimulus=6,_behavior=6,_self_att=6,_modalities=(n_behavior=25))/25\"\n",
    "    \n",
    "    args.dataset = \"lateral\"\n",
    "    args.ckpt_path = \"./models/NF.15/Visnav_VR_Expt/lateral/Neuroformer/demo/(state_history=6,_state=6,_stimulus=6,_behavior=6,_self_att=6,_modalities=(n_behavior=25))/25\"\n",
    "    args.predict_modes = ['speed']\n",
    "else:\n",
    "    print(\"Running in terminal\")\n",
    "    args = parse_args()\n",
    "\n",
    "\n",
    "# SET SEED - VERY IMPORTANT\n",
    "set_seed(args.seed)\n",
    "\n",
    "print(f\"CONTRASTIUVEEEEEEE {args.contrastive}\")\n",
    "print(f\"VISUAL: {args.visual}\")\n",
    "print(f\"PAST_STATE: {args.past_state}\")\n",
    "\n",
    "\n",
    "config, tokenizer, model = load_model_and_tokenizer(args.ckpt_path)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "DATASET: lateral\n"
     ]
    }
   ],
   "source": [
    "\"\"\" \n",
    "\n",
    "-- DATA --\n",
    "neuroformer/data/OneCombo3_V1AL/\n",
    "df = response\n",
    "video_stack = stimulus\n",
    "DOWNLOAD DATA URL = https://drive.google.com/drive/folders/1jNvA4f-epdpRmeG9s2E-2Sfo-pwYbjeY?usp=sharing\n",
    "\n",
    "\n",
    "\"\"\"\n",
    "print(f\"DATASET: {args.dataset}\")\n",
    "if args.dataset in [\"lateral\", \"medial\"]:\n",
    "    data, intervals, train_intervals, \\\n",
    "    test_intervals, finetune_intervals, \\\n",
    "    callback = load_visnav(args.dataset, config, \n",
    "                           selection=config.selection if hasattr(config, \"selection\") else None)\n",
    "elif args.dataset == \"V1AL\":\n",
    "    data, intervals, train_intervals, \\\n",
    "    test_intervals, finetune_intervals, \\\n",
    "    callback = load_V1AL(config)\n",
    "\n",
    "spikes = data['spikes']\n",
    "stimulus = data['stimulus']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "\" structure:\\n{\\n    type_of_modality:\\n        {name of modality: {'data':data, 'dt': dt, 'predict': True/False},\\n        ...\\n        }\\n    ...\\n}\\n\""
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "window = config.window.curr\n",
    "window_prev = config.window.prev\n",
    "dt = config.resolution.dt\n",
    "\n",
    "\n",
    "# -------- #\n",
    "\n",
    "spikes_dict = {\n",
    "    \"ID\": data['spikes'],\n",
    "    \"Frames\": data['stimulus'],\n",
    "    \"Interval\": intervals,\n",
    "    \"dt\": config.resolution.dt,\n",
    "    \"id_block_size\": config.block_size.id,\n",
    "    \"prev_id_block_size\": config.block_size.prev_id,\n",
    "    \"frame_block_size\": config.block_size.frame,\n",
    "    \"window\": config.window.curr,\n",
    "    \"window_prev\": config.window.prev,\n",
    "    \"frame_window\": config.window.frame,\n",
    "}\n",
    "\n",
    "\"\"\" structure:\n",
    "{\n",
    "    type_of_modality:\n",
    "        {name of modality: {'data':data, 'dt': dt, 'predict': True/False},\n",
    "        ...\n",
    "        }\n",
    "    ...\n",
    "}\n",
    "\"\"\"\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Min Interval: 0.1\n",
      "Intervals:  24092\n",
      "Window:  0.05\n",
      "Window Prev:  0.05\n",
      "Population Size:  2026\n",
      "ID Population Size:  2026\n",
      "DT Population Size:  9\n",
      "Using explicitly passed intervals\n",
      "Min Interval: 0.1\n",
      "Intervals:  5000\n",
      "Window:  0.05\n",
      "Window Prev:  0.05\n",
      "Population Size:  2026\n",
      "ID Population Size:  2026\n",
      "DT Population Size:  9\n",
      "Using explicitly passed intervals\n",
      "Min Interval: 0.1\n",
      "Intervals:  240\n",
      "Window:  0.05\n",
      "Window Prev:  0.05\n",
      "Population Size:  2026\n",
      "ID Population Size:  2026\n",
      "DT Population Size:  9\n",
      "Using explicitly passed intervals\n",
      "id_prev torch.Size([700]) torch.int64\n",
      "dt_prev torch.Size([700]) torch.float32\n",
      "pad_prev torch.Size([]) torch.int64\n",
      "id torch.Size([100]) torch.int64\n",
      "dt torch.Size([100]) torch.float32\n",
      "pad torch.Size([]) torch.int64\n",
      "interval torch.Size([]) torch.float32\n",
      "trial torch.Size([]) torch.int64\n",
      "frames torch.Size([1, 3, 30, 100]) torch.float32\n",
      "cid torch.Size([2]) torch.float32\n",
      "pid torch.Size([2]) torch.float32\n",
      "id torch.Size([2, 100]) torch.int64\n",
      "dt torch.Size([2, 100]) torch.int64\n"
     ]
    }
   ],
   "source": [
    "from neuroformer.data_utils import NFDataloader\n",
    "\n",
    "n_test_iters = 5000\n",
    "n_test_idx = np.random.choice(len(test_intervals), n_test_iters, replace=False)\n",
    "test_intervals = test_intervals[n_test_idx]\n",
    "\n",
    "modalities = create_modalities_dict(data, config.modalities) if hasattr(config, \"modalities\") else None\n",
    "frames = {'feats': stimulus, 'callback': callback, 'window': config.window.frame, 'dt': config.resolution.dt}\n",
    "\n",
    "train_dataset = NFDataloader(spikes_dict, tokenizer, config, dataset=args.dataset, \n",
    "                             frames=frames, intervals=train_intervals, modalities=modalities)\n",
    "test_dataset = NFDataloader(spikes_dict, tokenizer, config, dataset=args.dataset, \n",
    "                            frames=frames, intervals=test_intervals, modalities=modalities)\n",
    "finetune_dataset = NFDataloader(spikes_dict, tokenizer, config, dataset=args.dataset, \n",
    "                                frames=frames, intervals=finetune_intervals, modalities=modalities)\n",
    "\n",
    "    \n",
    "# print(f'train: {len(train_dataset)}, test: {len(test_dataset)}')\n",
    "iterable = iter(train_dataset)\n",
    "x, y = next(iterable)\n",
    "recursive_print(x)\n",
    "\n",
    "# Update the config\n",
    "config.id_vocab_size = tokenizer.ID_vocab_size\n",
    "\n",
    "# Create a DataLoader\n",
    "loader = DataLoader(test_dataset, batch_size=2, shuffle=True, num_workers=0)\n",
    "iterable = iter(loader)\n",
    "x, y = next(iterable)\n",
    "recursive_print(y)\n",
    "preds, features, loss = model(x, y)\n",
    "\n",
    "# Set training parameters\n",
    "MAX_EPOCHS = 300\n",
    "BATCH_SIZE = 32 * 5\n",
    "SHUFFLE = True\n",
    "\n",
    "if config.gru_only:\n",
    "    model_name = \"GRU\"\n",
    "elif config.mlp_only:\n",
    "    model_name = \"MLP\"\n",
    "elif config.gru2_only:\n",
    "    model_name = \"GRU_2.0\"\n",
    "else:\n",
    "    model_name = \"Neuroformer\"\n",
    "\n",
    "CKPT_PATH = args.ckpt_path\n",
    "\n",
    "# Define the parameters\n",
    "sample = True\n",
    "top_p = 0.95\n",
    "top_p_t = 0.95\n",
    "temp = 1.\n",
    "temp_t = 1.\n",
    "frame_end = 0\n",
    "true_past = args.true_past\n",
    "get_dt = True\n",
    "gpu = True\n",
    "pred_dt = True\n",
    "\n",
    "# # Run the prediction function\n",
    "# results_trial = generate_spikes(model, test_dataset, window, \n",
    "#                                 window_prev, tokenizer, \n",
    "#                                 sample=sample, top_p=top_p, top_p_t=top_p_t, \n",
    "#                                 temp=temp, temp_t=temp_t, frame_end=frame_end, \n",
    "#                                 true_past=true_past,\n",
    "#                                 get_dt=get_dt, gpu=gpu, pred_dt=pred_dt,\n",
    "#                                 plot_probs=False)\n",
    "\n",
    "# # Create a filename string with the parameters\n",
    "# filename = f\"results_trial_sample-{sample}_top_p-{top_p}_top_p_t-{top_p_t}_temp-{temp}_temp_t-{temp_t}_frame_end-{frame_end}_true_past-{true_past}_get_dt-{get_dt}_gpu-{gpu}_pred_dt-{pred_dt}.pkl\"\n",
    "\n",
    "# # Save the results in a pickle file\n",
    "# save_inference_path = os.path.join(CKPT_PATH, \"inference\")\n",
    "# if not os.path.exists(save_inference_path):\n",
    "#     os.makedirs(save_inference_path)\n",
    "\n",
    "# print(f\"Saving inference results in {os.path.join(save_inference_path, filename)}\")\n",
    "\n",
    "# with open(os.path.join(save_inference_path, filename), \"wb\") as f:\n",
    "#     pickle.dump(results_trial, f)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "preds, features, loss = model(x, y)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "dict_keys(['id', 'frames', 'id_prev', 'raw_frames', 'clip', 'last_layer'])\n"
     ]
    }
   ],
   "source": [
    "print(features.keys())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(30117,)"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "data['speed'].shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 79/79 [01:52<00:00,  1.42s/it]\n"
     ]
    }
   ],
   "source": [
    "from tqdm import tqdm\n",
    "\n",
    "features['last_layer']\n",
    "device = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
    "model.to(device)\n",
    "\n",
    "features_model = {}\n",
    "features_model['last_layer'] = []\n",
    "features_model['interval'] = []\n",
    "\n",
    "model.eval()\n",
    "\n",
    "loader = DataLoader(test_dataset, batch_size=64, shuffle=False, num_workers=0)\n",
    "\n",
    "with torch.no_grad():\n",
    "    for i, (x, y) in tqdm(enumerate(loader), total=len(loader)):\n",
    "        x = all_device(x, device)\n",
    "        y = all_device(y, device)\n",
    "        preds, features, loss = model(x, y)\n",
    "        last_layer_features = torch.unbind(features['last_layer'], dim=0)\n",
    "        interval_features = torch.unbind(x['interval'], dim=0)\n",
    "        for last_layer, interval in zip(last_layer_features, interval_features):\n",
    "            features_model['last_layer'].append(last_layer.cpu().detach().numpy())\n",
    "            features_model['interval'].append(interval.cpu().detach().numpy())    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "feats: (5000, 100, 256), speed_idx: (5000,)\n"
     ]
    }
   ],
   "source": [
    "from neuroformer.data_utils import get_frame_idx\n",
    "\n",
    "speed_idx = [get_frame_idx(i, 0.05) for i in features_model['interval']]\n",
    "speed_idx = np.array(speed_idx)\n",
    "speed = data['speed'][speed_idx]\n",
    "feats = np.array(features_model['last_layer'])\n",
    "\n",
    "print(f\"feats: {feats.shape}, speed_idx: {speed_idx.shape}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "R2: 0.9999407484548073\n"
     ]
    }
   ],
   "source": [
    "from sklearn.linear_model import LinearRegression\n",
    "\n",
    "# Reshape feats to be 2-dimensional\n",
    "feats_reshaped = feats.reshape(feats.shape[0], -1)\n",
    "\n",
    "# Initialize the model\n",
    "reg = LinearRegression()\n",
    "\n",
    "# Fit the reg to the data\n",
    "reg.fit(feats_reshaped, speed)\n",
    "\n",
    "# Now you can use reg.predict() to make predictions\n",
    "predicted_speed = reg.predict(feats_reshaped)\n",
    "\n",
    "from sklearn.metrics import mean_squared_error, mean_absolute_error, r2_score\n",
    "\n",
    "# Calculate the actual speed\n",
    "actual_speed = speed\n",
    "\n",
    "# Calculate the metrics\n",
    "r2 = r2_score(actual_speed, predicted_speed)\n",
    "\n",
    "print(f\"R2: {r2}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 79/79 [01:51<00:00,  1.41s/it]\n"
     ]
    }
   ],
   "source": [
    "from tqdm import tqdm\n",
    "\n",
    "# Initialize the dictionary to store the features\n",
    "features_model = {\n",
    "    'state': [],\n",
    "    'history': [],\n",
    "    'stimulus': [],\n",
    "    'interval': [],\n",
    "    'x_id': [],\n",
    "}\n",
    "\n",
    "# Move the model to the device\n",
    "device = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
    "model.to(device)\n",
    "\n",
    "# Set the model to evaluation mode\n",
    "model.eval()\n",
    "\n",
    "# Create the DataLoader\n",
    "loader = DataLoader(test_dataset, batch_size=64, shuffle=False, num_workers=0)\n",
    "\n",
    "# Define the hook function\n",
    "def hook(name):\n",
    "    def hook_fn(module, input, output):\n",
    "        features_model[name].append(output.cpu().detach().numpy())\n",
    "    return hook_fn\n",
    "\n",
    "# Attach the hook to the last layer of each block\n",
    "hook_handle_state = model.neural_visual_transformer.neural_state_blocks[0].register_forward_hook(hook('state'))\n",
    "hook_handle_history = model.neural_visual_transformer.neural_state_history_blocks[0].register_forward_hook(hook('history'))\n",
    "hook_handle_stimulus = model.neural_visual_transformer.neural_state_stimulus_blocks[0].register_forward_hook(hook('stimulus'))\n",
    "\n",
    "# Iterate over the DataLoader\n",
    "with torch.no_grad():\n",
    "    for i, (x, y) in tqdm(enumerate(loader), total=len(loader)):\n",
    "        x = all_device(x, device)\n",
    "        y = all_device(y, device)\n",
    "        \n",
    "        # Forward pass\n",
    "        preds, features, loss = model(x, y)\n",
    "        \n",
    "        # Collect the intervals\n",
    "        interval_features = torch.unbind(x['interval'], dim=0)\n",
    "        features_model['x_id'].append(x['id'].cpu().detach().numpy())\n",
    "        \n",
    "\n",
    "# Remove the hooks\n",
    "hook_handle_state.remove()\n",
    "hook_handle_history.remove()\n",
    "hook_handle_stimulus.remove()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.collections.PathCollection at 0x7fd6bcc54810>"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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p0ePHta4ceNKnzudTisyMtJKSUkp9/ibb77ZGjBgQJnXevbsad155522ttNu1b0OVf034+0kWR9++GGlxzzwwANWhw4dyrw2ePBgKzEx0caWuVdVrsOyZcssSdavv/7qljbVpL1791qSrC+//LLCYzztbwU9L7VYTk6O8vLylJCQUPpaSEiIevbsqfT09Bps2dlJT09Xo0aN1L1799LXEhIS5OfnpzVr1lR67jvvvKOmTZuqY8eOmjJlio4cOWJ3c40pLi7WunXryvw8/fz8lJCQUOHPMz09vczxkpSYmOjVP/8zuQ6SdOjQIbVo0ULNmzfXddddp40bN7qjuR7HF38nzkZMTIwiIiLUt29fffXVVzXdHFsUFBRIkho3blzhMZ72e+FzGzOi6k7euw0LCyvzelhYmFff183Lyzula7dOnTpq3Lhxpd/XsGHD1KJFC0VGRiorK0t//etftXXrVn3wwQd2N9mIX375RU6ns9yf55YtW8o9Jy8vz+d+/mdyHdq2bavXXntNnTt3VkFBgZ566inFx8dr48aNtW6j14p+JwoLC3X06FHVq1evhlrmXhEREZo7d666d++uoqIivfLKK+rdu7fWrFmj2NjYmm6eMS6XSxMnTtSll16qjh07Vnicp/2tILx4uMmTJ+uJJ56o9JjNmzerXbt2bmpRzanqtThTvx8T06lTJ0VERKhPnz7asWOHWrVqdcZ14fni4uIUFxdX+jw+Pl7t27fXSy+9pEcffbQGW4aa0rZtW7Vt27b0eXx8vHbs2KFnn31Wb731Vg22zKxx48YpOztbq1atqummVAvhxcNNmjRJI0eOrPSYCy+88Ixqh4eHS5Ly8/MVERFR+np+fr5iYmLOqKadqnotwsPDTxmYWVJSov3795d+z1XRs2dPSdL27du9Irw0bdpU/v7+ys/PL/N6fn5+hd93eHh4tY73BmdyHf6obt266tq1q7Zv325HEz1aRb8TwcHBtabXpSI9evTwug/5yowfP750QsPpehg97W8FY148XLNmzdSuXbtKHwEBAWdUOyoqSuHh4VqyZEnpa4WFhVqzZk2Z/wv1FFW9FnFxcTpw4IDWrVtXeu7SpUvlcrlKA0lVZGZmSlKZYOfJAgIC1K1btzI/T5fLpSVLllT484yLiytzvCQtWrTII3/+VXUm1+GPnE6nNmzY4DU/e5N88XfClMzMTJ/4nbAsS+PHj9eHH36opUuXKioq6rTneNzvRY0ME4YtfvjhBysjI8NKTk62zjnnHCsjI8PKyMiwDh48WHpM27ZtrQ8++KD0+eOPP241atTI+vjjj62srCzruuuus6KioqyjR4/WxLdgTL9+/ayuXbtaa9assVatWmW1adPGGjp0aOnXf/rpJ6tt27bWmjVrLMuyrO3bt1vTp0+3vv32WysnJ8f6+OOPrQsvvNDq1atXTX0LZ2TBggVWYGCgNW/ePGvTpk3WHXfcYTVq1MjKy8uzLMuy/vKXv1iTJ08uPf6rr76y6tSpYz311FPW5s2brWnTpll169a1NmzYUFPfghHVvQ7JycnWF198Ye3YscNat26dNWTIECsoKMjauHFjTX0Lxhw8eLD0b4Ek65lnnrEyMjKsH374wbIsy5o8ebL1l7/8pfT4//3vf1b9+vWt+++/39q8ebM1Z84cy9/f30pLS6upb8GI6l6HZ5991vroo4+sbdu2WRs2bLAmTJhg+fn5WYsXL66pb8GYu+66ywoJCbGWL19u5ebmlj6OHDlSeoyn/60gvPiQESNGWJJOeSxbtqz0GEnW66+/Xvrc5XJZjzzyiBUWFmYFBgZaffr0sbZu3er+xhu2b98+a+jQodY555xjBQcHW6NGjSoT4nJycspcm127dlm9evWyGjdubAUGBlqtW7e27r//fqugoKCGvoMzN3v2bOuCCy6wAgICrB49elirV68u/drll19ujRgxoszx7733nnXRRRdZAQEBVocOHazPPvvMzS22R3Wuw8SJE0uPDQsLs/r372+tX7++Blpt3skpv398nPz+R4wYYV1++eWnnBMTE2MFBARYF154YZm/Gd6qutfhiSeesFq1amUFBQVZjRs3tnr37m0tXbq0ZhpvWHnX4Y+fDZ7+t8JhWZbltm4eAACAs8SYFwAA4FUILwAAwKsQXgAAgFchvAAAAK9CeAEAAF6F8AIAALwK4QUAAHgVwgsAAPAqhBcAAOBVCC8AAMCrEF4AAIBXIbwAAACv8v8AwqmRGioYUi0AAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.scatter(actual_speed, predicted_speed)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(64, 100)"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "features_model['x_id'][0].shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(5000, 25600)"
      ]
     },
     "execution_count": 14,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "feats_reshaped.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Neuroformer(\n",
       "  (tok_emb): Embedding(2026, 256)\n",
       "  (temp_emb): LearntTemporalEmbedding(\n",
       "    (temp_emb): Sequential(\n",
       "      (0): Linear(in_features=1, out_features=128, bias=True)\n",
       "      (1): GELU(approximate='none')\n",
       "      (2): Linear(in_features=128, out_features=256, bias=True)\n",
       "      (3): Dropout(p=0.2, inplace=False)\n",
       "    )\n",
       "  )\n",
       "  (temp_emb_prev): LearntTemporalEmbedding(\n",
       "    (temp_emb): Sequential(\n",
       "      (0): Linear(in_features=1, out_features=128, bias=True)\n",
       "      (1): GELU(approximate='none')\n",
       "      (2): Linear(in_features=128, out_features=256, bias=True)\n",
       "      (3): Dropout(p=0.2, inplace=False)\n",
       "    )\n",
       "  )\n",
       "  (modality_embeddings): ModuleDict()\n",
       "  (modality_projection_heads): ModuleDict()\n",
       "  (frame_3d_emb): PositionalEncoding3D()\n",
       "  (id_drop): Dropout(p=0.2, inplace=False)\n",
       "  (im_drop): Dropout(p=0.2, inplace=False)\n",
       "  (drop): Dropout(p=0.2, inplace=False)\n",
       "  (video_encoder): VideoEncoder(\n",
       "    (conv_block): Sequential(\n",
       "      (0): Conv3d(1, 256, kernel_size=(3, 5, 5), stride=(3, 5, 5))\n",
       "      (1): Rearrange('b e t h w -> b t h w e')\n",
       "      (2): LayerNorm((256,), eps=1e-05, elementwise_affine=True)\n",
       "      (3): ReLU()\n",
       "    )\n",
       "  )\n",
       "  (feature_encoder): FeatureEncoder(\n",
       "    (neural_state_blocks): ModuleList(\n",
       "      (0-5): 6 x Block(\n",
       "        (ln1): LayerNorm((256,), eps=1e-05, elementwise_affine=True)\n",
       "        (ln2): LayerNorm((256,), eps=1e-05, elementwise_affine=True)\n",
       "        (attn): MultiheadAttention(\n",
       "          (query): Linear(in_features=256, out_features=256, bias=True)\n",
       "          (key): Linear(in_features=256, out_features=256, bias=True)\n",
       "          (value): Linear(in_features=256, out_features=256, bias=True)\n",
       "          (attn_drop): Dropout(p=0.2, inplace=False)\n",
       "          (resid_drop): Dropout(p=0.2, inplace=False)\n",
       "          (proj): Linear(in_features=256, out_features=256, bias=True)\n",
       "        )\n",
       "        (mlp): Sequential(\n",
       "          (0): Linear(in_features=256, out_features=1024, bias=True)\n",
       "          (1): GELU(approximate='none')\n",
       "          (2): Linear(in_features=1024, out_features=256, bias=True)\n",
       "          (3): Dropout(p=0.2, inplace=False)\n",
       "        )\n",
       "        (ln_f): LayerNorm((256,), eps=1e-05, elementwise_affine=True)\n",
       "      )\n",
       "    )\n",
       "    (frame_state_blocks): ModuleList(\n",
       "      (0-5): 6 x Block(\n",
       "        (ln1): LayerNorm((256,), eps=1e-05, elementwise_affine=True)\n",
       "        (ln2): LayerNorm((256,), eps=1e-05, elementwise_affine=True)\n",
       "        (attn): MultiheadAttention(\n",
       "          (query): Linear(in_features=256, out_features=256, bias=True)\n",
       "          (key): Linear(in_features=256, out_features=256, bias=True)\n",
       "          (value): Linear(in_features=256, out_features=256, bias=True)\n",
       "          (attn_drop): Dropout(p=0.2, inplace=False)\n",
       "          (resid_drop): Dropout(p=0.2, inplace=False)\n",
       "          (proj): Linear(in_features=256, out_features=256, bias=True)\n",
       "        )\n",
       "        (mlp): Sequential(\n",
       "          (0): Linear(in_features=256, out_features=1024, bias=True)\n",
       "          (1): GELU(approximate='none')\n",
       "          (2): Linear(in_features=1024, out_features=256, bias=True)\n",
       "          (3): Dropout(p=0.2, inplace=False)\n",
       "        )\n",
       "        (ln_f): LayerNorm((256,), eps=1e-05, elementwise_affine=True)\n",
       "      )\n",
       "    )\n",
       "  )\n",
       "  (clip): CLIP(\n",
       "    (proj): Linear(in_features=256, out_features=256, bias=False)\n",
       "  )\n",
       "  (neural_visual_transformer): MultimodalTransformer(\n",
       "    (modalities_blocks): ModuleDict()\n",
       "    (neural_state_blocks): ModuleList(\n",
       "      (0-5): 6 x Block(\n",
       "        (ln1): LayerNorm((256,), eps=1e-05, elementwise_affine=True)\n",
       "        (ln2): LayerNorm((256,), eps=1e-05, elementwise_affine=True)\n",
       "        (attn): MultiheadAttention(\n",
       "          (query): Linear(in_features=256, out_features=256, bias=True)\n",
       "          (key): Linear(in_features=256, out_features=256, bias=True)\n",
       "          (value): Linear(in_features=256, out_features=256, bias=True)\n",
       "          (attn_drop): Dropout(p=0.2, inplace=False)\n",
       "          (resid_drop): Dropout(p=0.2, inplace=False)\n",
       "          (proj): Linear(in_features=256, out_features=256, bias=True)\n",
       "        )\n",
       "        (mlp): Sequential(\n",
       "          (0): Linear(in_features=256, out_features=1024, bias=True)\n",
       "          (1): GELU(approximate='none')\n",
       "          (2): Linear(in_features=1024, out_features=256, bias=True)\n",
       "          (3): Dropout(p=0.2, inplace=False)\n",
       "        )\n",
       "        (ln_f): LayerNorm((256,), eps=1e-05, elementwise_affine=True)\n",
       "      )\n",
       "    )\n",
       "    (neural_state_history_blocks): ModuleList(\n",
       "      (0-5): 6 x Block(\n",
       "        (ln1): LayerNorm((256,), eps=1e-05, elementwise_affine=True)\n",
       "        (ln2): LayerNorm((256,), eps=1e-05, elementwise_affine=True)\n",
       "        (attn): MultiheadAttention(\n",
       "          (query): Linear(in_features=256, out_features=256, bias=True)\n",
       "          (key): Linear(in_features=256, out_features=256, bias=True)\n",
       "          (value): Linear(in_features=256, out_features=256, bias=True)\n",
       "          (attn_drop): Dropout(p=0.2, inplace=False)\n",
       "          (resid_drop): Dropout(p=0.2, inplace=False)\n",
       "          (proj): Linear(in_features=256, out_features=256, bias=True)\n",
       "        )\n",
       "        (mlp): Sequential(\n",
       "          (0): Linear(in_features=256, out_features=1024, bias=True)\n",
       "          (1): GELU(approximate='none')\n",
       "          (2): Linear(in_features=1024, out_features=256, bias=True)\n",
       "          (3): Dropout(p=0.2, inplace=False)\n",
       "        )\n",
       "        (ln_f): LayerNorm((256,), eps=1e-05, elementwise_affine=True)\n",
       "      )\n",
       "    )\n",
       "    (neural_state_stimulus_blocks): ModuleList(\n",
       "      (0-5): 6 x Block(\n",
       "        (ln1): LayerNorm((256,), eps=1e-05, elementwise_affine=True)\n",
       "        (ln2): LayerNorm((256,), eps=1e-05, elementwise_affine=True)\n",
       "        (attn): MultiheadAttention(\n",
       "          (query): Linear(in_features=256, out_features=256, bias=True)\n",
       "          (key): Linear(in_features=256, out_features=256, bias=True)\n",
       "          (value): Linear(in_features=256, out_features=256, bias=True)\n",
       "          (attn_drop): Dropout(p=0.2, inplace=False)\n",
       "          (resid_drop): Dropout(p=0.2, inplace=False)\n",
       "          (proj): Linear(in_features=256, out_features=256, bias=True)\n",
       "        )\n",
       "        (mlp): Sequential(\n",
       "          (0): Linear(in_features=256, out_features=1024, bias=True)\n",
       "          (1): GELU(approximate='none')\n",
       "          (2): Linear(in_features=1024, out_features=256, bias=True)\n",
       "          (3): Dropout(p=0.2, inplace=False)\n",
       "        )\n",
       "        (ln_f): LayerNorm((256,), eps=1e-05, elementwise_affine=True)\n",
       "      )\n",
       "    )\n",
       "    (ln_f): LayerNorm((256,), eps=1e-05, elementwise_affine=True)\n",
       "  )\n",
       "  (head_id): Linear(in_features=256, out_features=2026, bias=False)\n",
       "  (head_dt): Linear(in_features=256, out_features=9, bias=False)\n",
       ")"
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "title_mapping = {\n",
    "    'history': 'history',\n",
    "    'stimulus': 'history + stimulus',\n",
    "    'state': 'history + stimulus + state'\n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Feature: history + stimulus + state\n",
      "R2: 0.635742823843209\n",
      "-----------------\n",
      "Feature: history\n",
      "R2: 0.22999429697899065\n",
      "-----------------\n",
      "Feature: history + stimulus\n",
      "R2: 0.5341393800678469\n",
      "-----------------\n"
     ]
    }
   ],
   "source": [
    "# fit models for each feature\n",
    "\n",
    "for feature in ['state', 'history', 'stimulus']:\n",
    "    print(f\"Feature: {title_mapping[feature]}\")\n",
    "    # Reshape feats to be 2-dimensional\n",
    "    \n",
    "    feats_reshaped = np.vstack([feat[:, -1].reshape(feat.shape[0], -1) for feat in features_model[feature]])\n",
    "\n",
    "    # Initialize the model\n",
    "    reg = LinearRegression()\n",
    "\n",
    "    # Fit the reg to the data\n",
    "    reg.fit(feats_reshaped, speed)\n",
    "\n",
    "    # Now you can use reg.predict() to make predictions\n",
    "    predicted_speed = reg.predict(feats_reshaped)\n",
    "\n",
    "    # Calculate the actual speed\n",
    "    actual_speed = speed\n",
    "\n",
    "    # Calculate the metrics\n",
    "    r2 = r2_score(actual_speed, predicted_speed)\n",
    "\n",
    "    print(f\"R2: {r2}\")\n",
    "    print(\"-----------------\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.collections.PathCollection at 0x7fd6bcc74ed0>"
      ]
     },
     "execution_count": 19,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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H/4gfPPNN0J1dbXw6aefCn369BGuvvrqcL2FgHzwwQeCwWAQVqxYIVRWVgr333+/kJKSItTW1gqCIAh33HGHMH/+fMf227ZtE2JjY4XnnntO2Ldvn7Bw4UIhLi5O2LNnT7jegiyk7odFixYJ//rXv4Sqqiph165dwowZM4SEhARh79694XoLsmloaHCcCwAI//M//yPs3r1bOHLkiCAIgjB//nzhjjvucGz/ww8/CImJicJvfvMbYd++fcLLL78sxMTECCUlJeF6C7KQuh+WLl0qfPLJJ8LBgweFPXv2CA8//LCg1+uFdevWhestyGb27NmC0WgUNm3aJNTU1Dh+zp0759hG7ecKBi8R5K677hIAdPjZuHGjYxsAwptvvun43WazCU888YSQkZEhGAwG4dprrxUOHDgQ+sbL7NSpU8LMmTOFzp07C8nJycI999zjEsRVV1e77JujR48KV199tZCamioYDAahX79+wm9+8xvBYrGE6R0E7sUXXxQuu+wyIT4+Xhg5cqSwfft2x79dc801wl133eWy/d/+9jfh8ssvF+Lj44VBgwYJq1evDnGLlSFlPzzyyCOObTMyMoQpU6YIZWVlYWi1/OxTft1/7O//rrvuEq655poOz8nLyxPi4+OFPn36uJwztErqfliyZInQt29fISEhQUhNTRXGjx8vbNiwITyNl5mn/eB+bVD7uUInCIIQsm4eIiIioiAx54WIiIg0hcELERERaQqDFyIiItIUBi9ERESkKQxeiIiISFMYvBAREZGmMHghIiIiTWHwQkRERJrC4IWIiIg0hcELERERaQqDFyIiItIUBi9ERESkKf8frrmqQtaDGPMAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.scatter(actual_speed, predicted_speed)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "R2: 0.052081271436897336\n"
     ]
    }
   ],
   "source": [
    "rand_feats = np.random.rand(feats_reshaped.shape[0], feats_reshaped.shape[1])\n",
    "\n",
    "model_rand = LinearRegression()\n",
    "model_rand.fit(rand_feats, speed)\n",
    "pred_rand = model_rand.predict(rand_feats)\n",
    "\n",
    "r2_rand = r2_score(speed, pred_rand)\n",
    "print(f\"R2: {r2_rand}\")\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "neuroformer",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.11.5"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
