{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "c993ffcf",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-01-04T18:15:40.957042Z",
     "start_time": "2023-01-04T18:15:40.362164Z"
    }
   },
   "outputs": [],
   "source": [
    "import os\n",
    "import copy\n",
    "\n",
    "import PIL.Image\n",
    "import pandas as pd\n",
    "import numpy as np\n",
    "import concurrent.futures\n",
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "5e997687",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-01-04T18:42:28.907470Z",
     "start_time": "2023-01-04T18:42:28.862144Z"
    }
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.image.AxesImage at 0x7ff8426c01f0>"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "def _get_positional_encoding(g, n_hiddens):\n",
    "        pe = np.zeros((g, n_hiddens))\n",
    "        for pos in range(g):\n",
    "            for i in range(0, n_hiddens, 2):\n",
    "                demon = 3 ** ((2 * i) / n_hiddens)\n",
    "                pe[pos, i] = np.sin(pos / demon)\n",
    "                pe[pos, i + 1] = np.cos(pos / demon)\n",
    "        return pe\n",
    "    \n",
    "pe = _get_positional_encoding(10, 128)\n",
    "\n",
    "plt.imshow(pe)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "70de8d45",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-01-04T18:15:40.959800Z",
     "start_time": "2023-01-04T18:15:40.958347Z"
    }
   },
   "outputs": [],
   "source": [
    "W, H = 128, 128  # global width and height for image resolution\n",
    "VOL = '/s/chopin/l/grad/stock/nvme/data/ai2es/mlhub/nasa_tc'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "b6fe9a4a",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-01-04T18:15:40.968782Z",
     "start_time": "2023-01-04T18:15:40.960558Z"
    }
   },
   "outputs": [],
   "source": [
    "def partition_data_by_storm(train_df, test_df, train_fraction=0.8, seed=1):\n",
    "    storms = train_df.storm_id.unique()\n",
    "    n_storms = len(storms)\n",
    "    train_frac = int(n_storms * train_fraction)\n",
    "    inds = np.arange(n_storms)\n",
    "\n",
    "    np.random.seed(seed)\n",
    "    np.random.shuffle(inds)\n",
    "\n",
    "    train_storm_ids = storms[inds[:train_frac]]\n",
    "    val_storm_ids = storms[inds[train_frac:]]\n",
    "\n",
    "    train = train_df.loc[train_df.storm_id.isin(\n",
    "        train_storm_ids)].reset_index(drop=True)\n",
    "    val = train_df.loc[train_df.storm_id.isin(\n",
    "        val_storm_ids)].reset_index(drop=True)\n",
    "\n",
    "    return train, val, test_df\n",
    "\n",
    "def read_images(files, threads=12):\n",
    "    \"\"\"Read images by filenames to a numpy array\n",
    "\n",
    "    N = number of images\n",
    "    W = width of images\n",
    "    H = height of images\n",
    "    C = number of channels \n",
    "\n",
    "    :param files: list of files to read\n",
    "    :param threads: number of threads to use for reading\n",
    "    :return: images: (N,W,H,C) numpy array of images\n",
    "    \"\"\"\n",
    "    def _reading_thread(files, n_images, section):\n",
    "        \"\"\"Helper function for reading in different threads\"\"\"\n",
    "        for i, f in enumerate(files[section:section+n_images]):\n",
    "            with PIL.Image.open(f).convert('L') as im:\n",
    "                # convert from 366 x 366 to ...\n",
    "                images[i + section] = np.expand_dims(im.resize((W, H)), axis=0) / 255.\n",
    "\n",
    "    images = np.zeros((len(files), 1, W, H))\n",
    "    n_images = len(files) // threads\n",
    "\n",
    "    with concurrent.futures.ThreadPoolExecutor(max_workers=threads) as executor:\n",
    "        for section in [n_images * i for i in range(threads + 1)]:\n",
    "            executor.submit(_reading_thread, files, n_images, section)\n",
    "\n",
    "    return images\n",
    "\n",
    "\n",
    "def load_data(train, val, test):\n",
    "    \"\"\"Convert DataFrame to numpy array for training, \n",
    "    validation, and test datasets\n",
    "\n",
    "    F = number of target output features\n",
    "\n",
    "    :param train: _training_ df with file_name and wind_speed variables \n",
    "    :param val: _validation_ df with file_name and wind_speed variables \n",
    "    :param test: _test_ df with file_name and wind_speed variables \n",
    "    :return: X_: (N,W,H,C) numpy array of _ images\n",
    "    :return: T_: (N,F) numpy array of _ targets\n",
    "    \"\"\"\n",
    "    Xtrain = read_images(train.file_name.values)\n",
    "    Ttrain = train.wind_speed.values.reshape(-1, 1)\n",
    "\n",
    "    Xval = read_images(val.file_name.values)\n",
    "    Tval = val.wind_speed.values.reshape(-1, 1)\n",
    "\n",
    "    Xtest = read_images(test.file_name.values)\n",
    "    Ttest = test.wind_speed.values.reshape(-1, 1)\n",
    "\n",
    "    return Xtrain, Ttrain, Xval, Tval, Xtest, Ttest"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "bd6f8fc3",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-01-04T18:15:41.139985Z",
     "start_time": "2023-01-04T18:15:40.969849Z"
    }
   },
   "outputs": [],
   "source": [
    "train_df = pd.read_csv(os.path.join(VOL, 'train_metadata.csv'))\n",
    "test_df = pd.read_csv(os.path.join(VOL, 'test_metadata.csv'))\n",
    "train, val, test = partition_data_by_storm(train_df, test_df, train_fraction=0.8, seed=1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "10d0a424",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-01-04T18:15:41.426843Z",
     "start_time": "2023-01-04T18:15:41.422289Z"
    }
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "((56664, 8), (13593, 8), (44377, 8))"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "train.shape, val.shape, test.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "0e60ab22",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-01-04T18:16:23.205969Z",
     "start_time": "2023-01-04T18:15:41.949845Z"
    }
   },
   "outputs": [],
   "source": [
    "Xtrain, Ttrain, Xval, Tval, Xtest, Ttest = load_data(train, val, test)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "fcb43fe2",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-01-04T18:16:23.209627Z",
     "start_time": "2023-01-04T18:16:23.207196Z"
    }
   },
   "outputs": [],
   "source": [
    "def categorize(targets):\n",
    "    \"\"\"\n",
    "    Saffir-Simpson Hurricane Wind Scale\n",
    "    1: 64-82 kt\n",
    "    2: 83-95 kt\n",
    "    3: 96-112 kt\n",
    "    4: 113-136 kt\n",
    "    5: 137 kt or higher\n",
    "    \"\"\"\n",
    "    T = copy.deepcopy(targets)\n",
    "#     T[T < 50] = -1\n",
    "#     T[(T >= 50) & (T <= 63)] = 0\n",
    "#     T[(T >= 64) & (T <= 82)] = 1\n",
    "#     T[(T >= 83) & (T <= 95)] = 2\n",
    "#     T[(T >= 96) & (T <= 112)] = 3\n",
    "#     T[(T >= 113) & (T <= 136)] = 4\n",
    "#     T[T >= 137] = 5\n",
    "    T[T < 50] = -1\n",
    "    T[(T >= 50) & (T <= 63)] = 0\n",
    "    T[(T >= 64) & (T <= 95)] = 1\n",
    "    T[T > 95] = 3\n",
    "    return T"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "ab07ed0f",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-01-04T18:16:23.836054Z",
     "start_time": "2023-01-04T18:16:23.210336Z"
    }
   },
   "outputs": [],
   "source": [
    "Ttrain = categorize(Ttrain)\n",
    "inds = np.where(Ttrain == -1)\n",
    "Xtrain = np.delete(Xtrain, inds, 0)\n",
    "Ttrain = np.delete(Ttrain, inds, 0).astype(int)\n",
    "\n",
    "Tval = categorize(Tval)\n",
    "inds = np.where(Tval == -1)\n",
    "Xval = np.delete(Xval, inds, 0)\n",
    "Tval = np.delete(Tval, inds, 0).astype(int)\n",
    "\n",
    "Ttest = categorize(Ttest)\n",
    "inds = np.where(Ttest == -1)\n",
    "Xtest = np.delete(Xtest, inds, 0)\n",
    "Ttest = np.delete(Ttest, inds, 0).astype(int)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "6792ca9e",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-01-04T18:16:23.839592Z",
     "start_time": "2023-01-04T18:16:23.837123Z"
    }
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "((22263, 1, 128, 128),\n",
       " (22263, 1),\n",
       " (5719, 1, 128, 128),\n",
       " (5719, 1),\n",
       " (15180, 1, 128, 128),\n",
       " (15180, 1))"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "Xtrain.shape, Ttrain.shape, Xval.shape, Tval.shape, Xtest.shape, Ttest.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "d31c42a1",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2023-01-04T18:16:24.069461Z",
     "start_time": "2023-01-04T18:16:23.840315Z"
    }
   },
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 648x216 with 3 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, axs = plt.subplots(1,3, figsize=(9,3))\n",
    "axs[0].hist(Ttrain, density=True, stacked=True, bins=30)\n",
    "axs[1].hist(Tval, density=True, stacked=True, bins=30)\n",
    "axs[2].hist(Ttest, density=True, stacked=True, bins=30);\n",
    "\n",
    "fig.tight_layout()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7a7ae7b8",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.8.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
