diff --git a/Finacial Domain/Dynamic Hedging Strategies using Reinforcement Learning/Dynamic_Hedging_StrategiesReinforcementLear.ipynb b/Finacial Domain/Dynamic Hedging Strategies using Reinforcement Learning/Dynamic_Hedging_StrategiesReinforcementLear.ipynb new file mode 100644 index 00000000..bb229aba --- /dev/null +++ b/Finacial Domain/Dynamic Hedging Strategies using Reinforcement Learning/Dynamic_Hedging_StrategiesReinforcementLear.ipynb @@ -0,0 +1,1122 @@ +{ + "cells": [ + { + "cell_type": "code", + "execution_count": 19, + "metadata": { + "id": "okgB-pZMZ5vH" + }, + "outputs": [], + "source": [] + }, + { + "cell_type": "code", + "execution_count": 20, + "metadata": { + "id": "lQMYLRlFY7e_" + }, + "outputs": [], + "source": [ + "import pandas as pd\n", + "data=pd.read_csv(\"/content/HistoricalData_1719635817381 .csv\")" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "yAxlj88ZoT9r" + }, + "source": [ + "**Preprocessing the data**" + ] + }, + { + "cell_type": "code", + "execution_count": 21, + "metadata": { + "colab": { + "base_uri": "https://localhost:8080/" + }, + "id": "GNvzBTbgZ6tz", + "outputId": "a939d7b3-2744-4465-fc03-bd3d0cb36071" + }, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Index(['Date', 'Close/Last', 'Open', 'High', 'Low'], dtype='object')\n", + " Date Close/Last Open High Low Daily Return\n", + "1 2024-06-27 0.983463 0.897188 0.875375 0.986483 0.099511\n", + "2 2024-06-26 0.963705 0.847029 0.846259 0.930446 -0.020090\n", + "3 2024-06-25 0.929517 0.847106 0.807311 0.911458 -0.035476\n", + "4 2024-06-24 0.844325 0.842628 0.874805 0.915141 -0.091652\n", + "5 2024-06-21 0.910912 0.870629 0.827924 0.931018 0.078864\n", + "Index(['Date', 'Close/Last', 'Open', 'High', 'Low', 'Daily Return'], dtype='object')\n" + ] + }, + { + "name": "stderr", + "output_type": "stream", + "text": [ + "/usr/local/lib/python3.10/dist-packages/ipykernel/ipkernel.py:283: DeprecationWarning: `should_run_async` will not call `transform_cell` automatically in the future. Please pass the result to `transformed_cell` argument and any exception that happen during thetransform in `preprocessing_exc_tuple` in IPython 7.17 and above.\n", + " and should_run_async(code)\n" + ] + } + ], + "source": [ + "import pandas as pd\n", + "from sklearn.preprocessing import MinMaxScaler\n", + "\n", + "# Convert the 'Date' column to datetime format\n", + "data['Date'] = pd.to_datetime(data['Date'], format='%m/%d/%Y')\n", + "\n", + "# Check column names\n", + "print(data.columns)\n", + "\n", + "# Initialize the scaler\n", + "scaler = MinMaxScaler()\n", + "\n", + "# Normalize the price columns\n", + "data[['Close/Last', 'Open', 'High', 'Low']] = scaler.fit_transform(data[['Close/Last', 'Open', 'High', 'Low']])\n", + "\n", + "# Create a column for daily returns\n", + "data['Daily Return'] = data['Close/Last'].pct_change()\n", + "\n", + "# Drop the first row with NaN value in 'Daily Return'\n", + "data = data.dropna()\n", + "\n", + "# Verify 'Daily Return' column is created\n", + "print(data.head())\n", + "print(data.columns)\n" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "GOlb7u9qoghx" + }, + "source": [ + "**Define the Environment**" + ] + }, + { + "cell_type": "code", + "execution_count": 22, + "metadata": { + "colab": { + "base_uri": "https://localhost:8080/" + }, + "id": "ZLWxYZ4-jqgE", + "outputId": "17232637-d854-4b74-a72f-d2817de04b23" + }, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "[0.9834625322997432 0.8971882545369603 0.8753748623922881\n", + " 0.9864826205120885 0.09951111111111066]\n" + ] + }, + { + "name": "stderr", + "output_type": "stream", + "text": [ + "/usr/local/lib/python3.10/dist-packages/ipykernel/ipkernel.py:283: DeprecationWarning: `should_run_async` will not call `transform_cell` automatically in the future. Please pass the result to `transformed_cell` argument and any exception that happen during thetransform in `preprocessing_exc_tuple` in IPython 7.17 and above.\n", + " and should_run_async(code)\n" + ] + } + ], + "source": [ + "import numpy as np\n", + "import gym\n", + "from gym import spaces\n", + "\n", + "class HedgingEnv(gym.Env):\n", + " def __init__(self, data):\n", + " super(HedgingEnv, self).__init__()\n", + " self.data = data\n", + " self.current_step = 0\n", + "\n", + " # State space: normalized prices and daily returns\n", + " self.observation_space = spaces.Box(\n", + " low=-np.inf, high=np.inf, shape=(5,), dtype=np.float32\n", + " )\n", + "\n", + " # Action space: increase, decrease, or maintain hedge ratio\n", + " self.action_space = spaces.Discrete(3)\n", + "\n", + " # Initial hedge ratio\n", + " self.hedge_ratio = 0.5\n", + "\n", + " def reset(self):\n", + " self.current_step = 0\n", + " self.hedge_ratio = 0.5\n", + " return self._next_observation()\n", + "\n", + " def _next_observation(self):\n", + " obs = self.data.iloc[self.current_step][['Close/Last', 'Open', 'High', 'Low', 'Daily Return']].values\n", + " return obs\n", + "\n", + " def step(self, action):\n", + " self.current_step += 1\n", + "\n", + " if self.current_step >= len(self.data) - 1:\n", + " done = True\n", + " else:\n", + " done = False\n", + "\n", + " # Update hedge ratio based on action\n", + " if action == 0:\n", + " self.hedge_ratio -= 0.1\n", + " elif action == 1:\n", + " self.hedge_ratio += 0.1\n", + " self.hedge_ratio = np.clip(self.hedge_ratio, 0, 1)\n", + "\n", + " # Calculate reward (simplified example)\n", + " reward = -abs(self.data.iloc[self.current_step]['Daily Return'] - self.hedge_ratio)\n", + "\n", + " return self._next_observation(), reward, done, {}\n", + "\n", + " def render(self, mode='human'):\n", + " pass\n", + "\n", + "# Example usage\n", + "env = HedgingEnv(data)\n", + "initial_observation = env.reset()\n", + "print(initial_observation)\n" + ] + }, + { + "cell_type": "code", + "execution_count": 23, + "metadata": { + "colab": { + "base_uri": "https://localhost:8080/" + }, + "id": "Wy8HuOTdqBAs", + "outputId": "beeca121-241c-4f17-aa7c-a651c5b085d0" + }, + "outputs": [ + { + "name": "stderr", + "output_type": "stream", + "text": [ + "/usr/local/lib/python3.10/dist-packages/ipykernel/ipkernel.py:283: DeprecationWarning: `should_run_async` will not call `transform_cell` automatically in the future. Please pass the result to `transformed_cell` argument and any exception that happen during thetransform in `preprocessing_exc_tuple` in IPython 7.17 and above.\n", + " and should_run_async(code)\n" + ] + } + ], + "source": [ + "import numpy as np\n", + "import gym\n", + "from gym import spaces\n", + "from collections import deque\n", + "import random\n", + "import tensorflow as tf\n", + "from tensorflow.keras.models import Sequential\n", + "from tensorflow.keras.layers import Dense\n", + "from tensorflow.keras.optimizers import Adam\n", + "\n", + "class DQNAgent:\n", + " def __init__(self, state_size, action_size):\n", + " self.state_size = state_size\n", + " self.action_size = action_size\n", + " self.memory = deque(maxlen=2000)\n", + " self.gamma = 0.95 # discount rate\n", + " self.epsilon = 1.0 # exploration rate\n", + " self.epsilon_min = 0.01\n", + " self.epsilon_decay = 0.995\n", + " self.learning_rate = 0.001\n", + " self.model = self._build_model()\n", + "\n", + " def _build_model(self):\n", + " # Neural Net for Deep Q-learning Model\n", + " model = Sequential()\n", + " model.add(Dense(24, input_dim=self.state_size, activation='relu'))\n", + " model.add(Dense(24, activation='relu'))\n", + " model.add(Dense(self.action_size, activation='linear'))\n", + " model.compile(loss='mse',\n", + " optimizer=Adam(learning_rate=self.learning_rate))\n", + " return model\n", + "\n", + " def remember(self, state, action, reward, next_state, done):\n", + " self.memory.append((state, action, reward, next_state, done))\n", + "\n", + " def act(self, state):\n", + " if np.random.rand() <= self.epsilon:\n", + " return random.randrange(self.action_size)\n", + " act_values = self.model.predict(state)\n", + " return np.argmax(act_values[0])\n", + "\n", + " def replay(self, batch_size):\n", + " minibatch = random.sample(self.memory, batch_size)\n", + " for state, action, reward, next_state, done in minibatch:\n", + " target = reward\n", + " if not done:\n", + " target = (reward + self.gamma *\n", + " np.amax(self.model.predict(next_state)[0]))\n", + " target_f = self.model.predict(state)\n", + " target_f[0][action] = target\n", + " self.model.fit(state, target_f, epochs=1, verbose=0)\n", + " if self.epsilon > self.epsilon_min:\n", + " self.epsilon *= self.epsilon_decay\n", + "\n", + "# Initialize the environment and the agent\n", + "state_size = 5\n", + "action_size = 3\n", + "agent = DQNAgent(state_size, action_size)\n" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "AQoKYI0SonyI" + }, + "source": [ + "Train the DQN **Agent**" + ] + }, + { + "cell_type": "code", + "execution_count": 24, + "metadata": { + "colab": { + "base_uri": "https://localhost:8080/" + }, + "id": "FvQgpcsxk2q6", + "outputId": "4d0c2e3b-5bc2-41e5-de6a-09e861cef77b" + }, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "1/1 [==============================] - 0s 84ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 29ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 21ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 29ms/step\n", + "1/1 [==============================] - 0s 29ms/step\n", + "1/1 [==============================] - 0s 21ms/step\n", + "1/1 [==============================] - 0s 21ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 36ms/step\n", + "1/1 [==============================] - 0s 40ms/step\n", + "1/1 [==============================] - 0s 34ms/step\n", + "1/1 [==============================] - 0s 35ms/step\n", + "1/1 [==============================] - 0s 34ms/step\n", + "1/1 [==============================] - 0s 38ms/step\n", + "1/1 [==============================] - 0s 42ms/step\n", + "1/1 [==============================] - 0s 43ms/step\n", + "1/1 [==============================] - 0s 32ms/step\n", + "1/1 [==============================] - 0s 36ms/step\n", + "1/1 [==============================] - 0s 38ms/step\n", + "1/1 [==============================] - 0s 38ms/step\n", + "1/1 [==============================] - 0s 45ms/step\n", + "1/1 [==============================] - 0s 34ms/step\n", + "1/1 [==============================] - 0s 36ms/step\n", + "1/1 [==============================] - 0s 45ms/step\n", + "1/1 [==============================] - 0s 42ms/step\n", + "1/1 [==============================] - 0s 37ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 37ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 39ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 30ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 29ms/step\n", + "1/1 [==============================] - 0s 33ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 31ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 21ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 33ms/step\n", + "1/1 [==============================] - 0s 29ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 34ms/step\n", + "1/1 [==============================] - 0s 36ms/step\n", + "1/1 [==============================] - 0s 35ms/step\n", + "1/1 [==============================] - 0s 36ms/step\n", + "1/1 [==============================] - 0s 41ms/step\n", + "1/1 [==============================] - 0s 42ms/step\n", + "1/1 [==============================] - 0s 39ms/step\n", + "1/1 [==============================] - 0s 36ms/step\n", + "1/1 [==============================] - 0s 37ms/step\n", + "1/1 [==============================] - 0s 33ms/step\n", + "1/1 [==============================] - 0s 32ms/step\n", + "1/1 [==============================] - 0s 35ms/step\n", + "1/1 [==============================] - 0s 59ms/step\n", + "1/1 [==============================] - 0s 34ms/step\n", + "1/1 [==============================] - 0s 35ms/step\n", + "1/1 [==============================] - 0s 36ms/step\n", + "1/1 [==============================] - 0s 38ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 33ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 33ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 30ms/step\n", + "1/1 [==============================] - 0s 30ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 35ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 31ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "episode: 0/2, score: 19, e: 0.96\n", + "1/1 [==============================] - 0s 30ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 33ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 39ms/step\n", + "1/1 [==============================] - 0s 31ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 30ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 37ms/step\n", + "1/1 [==============================] - 0s 37ms/step\n", + "1/1 [==============================] - 0s 39ms/step\n", + "1/1 [==============================] - 0s 41ms/step\n", + "1/1 [==============================] - 0s 41ms/step\n", + "1/1 [==============================] - 0s 40ms/step\n", + "1/1 [==============================] - 0s 40ms/step\n", + "1/1 [==============================] - 0s 40ms/step\n", + "1/1 [==============================] - 0s 44ms/step\n", + "1/1 [==============================] - 0s 36ms/step\n", + "1/1 [==============================] - 0s 35ms/step\n", + "1/1 [==============================] - 0s 40ms/step\n", + "1/1 [==============================] - 0s 41ms/step\n", + "1/1 [==============================] - 0s 38ms/step\n", + "1/1 [==============================] - 0s 34ms/step\n", + "1/1 [==============================] - 0s 44ms/step\n", + "1/1 [==============================] - 0s 37ms/step\n", + "1/1 [==============================] - 0s 29ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 36ms/step\n", + "1/1 [==============================] - 0s 29ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 39ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 29ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 36ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 20ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 38ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 35ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 30ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 41ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 45ms/step\n", + "1/1 [==============================] - 0s 37ms/step\n", + "1/1 [==============================] - 0s 39ms/step\n", + "1/1 [==============================] - 0s 46ms/step\n", + "1/1 [==============================] - 0s 36ms/step\n", + "1/1 [==============================] - 0s 44ms/step\n", + "1/1 [==============================] - 0s 42ms/step\n", + "1/1 [==============================] - 0s 43ms/step\n", + "1/1 [==============================] - 0s 37ms/step\n", + "1/1 [==============================] - 0s 41ms/step\n", + "1/1 [==============================] - 0s 37ms/step\n", + "1/1 [==============================] - 0s 43ms/step\n", + "1/1 [==============================] - 0s 35ms/step\n", + "1/1 [==============================] - 0s 41ms/step\n", + "1/1 [==============================] - 0s 35ms/step\n", + "1/1 [==============================] - 0s 39ms/step\n", + "1/1 [==============================] - 0s 42ms/step\n", + "1/1 [==============================] - 0s 33ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 29ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 21ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 30ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 33ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 21ms/step\n", + "1/1 [==============================] - 0s 40ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 21ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 31ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 34ms/step\n", + "1/1 [==============================] - 0s 30ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 35ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 40ms/step\n", + "1/1 [==============================] - 0s 37ms/step\n", + "1/1 [==============================] - 0s 36ms/step\n", + "1/1 [==============================] - 0s 33ms/step\n", + "1/1 [==============================] - 0s 43ms/step\n", + "1/1 [==============================] - 0s 38ms/step\n", + "1/1 [==============================] - 0s 36ms/step\n", + "1/1 [==============================] - 0s 40ms/step\n", + "1/1 [==============================] - 0s 36ms/step\n", + "1/1 [==============================] - 0s 37ms/step\n", + "1/1 [==============================] - 0s 36ms/step\n", + "1/1 [==============================] - 0s 34ms/step\n", + "1/1 [==============================] - 0s 45ms/step\n", + "1/1 [==============================] - 0s 41ms/step\n", + "1/1 [==============================] - 0s 36ms/step\n", + "1/1 [==============================] - 0s 36ms/step\n", + "1/1 [==============================] - 0s 37ms/step\n", + "1/1 [==============================] - 0s 46ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 21ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 31ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 31ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 34ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 21ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 39ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 21ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 35ms/step\n", + "1/1 [==============================] - 0s 38ms/step\n", + "1/1 [==============================] - 0s 37ms/step\n", + "1/1 [==============================] - 0s 34ms/step\n", + "1/1 [==============================] - 0s 36ms/step\n", + "1/1 [==============================] - 0s 39ms/step\n", + "1/1 [==============================] - 0s 37ms/step\n", + "1/1 [==============================] - 0s 43ms/step\n", + "1/1 [==============================] - 0s 37ms/step\n", + "1/1 [==============================] - 0s 33ms/step\n", + "1/1 [==============================] - 0s 40ms/step\n", + "1/1 [==============================] - 0s 43ms/step\n", + "1/1 [==============================] - 0s 44ms/step\n", + "1/1 [==============================] - 0s 45ms/step\n", + "1/1 [==============================] - 0s 37ms/step\n", + "1/1 [==============================] - 0s 38ms/step\n", + "1/1 [==============================] - 0s 35ms/step\n", + "1/1 [==============================] - 0s 36ms/step\n", + "1/1 [==============================] - 0s 39ms/step\n", + "1/1 [==============================] - 0s 38ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 30ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 30ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 32ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 28ms/step\n", + "1/1 [==============================] - 0s 29ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 29ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 31ms/step\n", + "1/1 [==============================] - 0s 34ms/step\n", + "1/1 [==============================] - 0s 30ms/step\n", + "1/1 [==============================] - 0s 29ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 23ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 31ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 39ms/step\n", + "1/1 [==============================] - 0s 24ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 22ms/step\n", + "1/1 [==============================] - 0s 34ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 25ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 26ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 27ms/step\n", + "1/1 [==============================] - 0s 42ms/step\n", + "1/1 [==============================] - 0s 51ms/step\n", + "1/1 [==============================] - 0s 35ms/step\n", + "1/1 [==============================] - 0s 34ms/step\n", + "1/1 [==============================] - 0s 35ms/step\n", + "1/1 [==============================] - 0s 35ms/step\n", + "1/1 [==============================] - 0s 34ms/step\n", + "1/1 [==============================] - 0s 51ms/step\n", + "1/1 [==============================] - 0s 38ms/step\n", + "1/1 [==============================] - 0s 36ms/step\n", + "episode: 1/2, score: 19, e: 0.87\n" + ] + } + ], + "source": [ + "env = HedgingEnv(data)\n", + "done = False\n", + "batch_size = 10\n", + "EPISODES = 2\n", + "\n", + "for e in range(EPISODES):\n", + " state = env.reset()\n", + " state = np.reshape(state, [1, state_size]).astype(np.float32) # Ensure the correct data type\n", + " for time in range(500):\n", + " action = agent.act(state)\n", + " next_state, reward, done, _ = env.step(action)\n", + " reward = reward if not done else -10\n", + " next_state = np.reshape(next_state, [1, state_size]).astype(np.float32) # Ensure the correct data type\n", + " agent.remember(state, action, reward, next_state, done)\n", + " state = next_state\n", + " if done:\n", + " print(f\"episode: {e}/{EPISODES}, score: {time}, e: {agent.epsilon:.2}\")\n", + " break\n", + " if len(agent.memory) > batch_size:\n", + " agent.replay(batch_size)\n" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "nIQ9pIgcpEoe" + }, + "source": [ + "**Evaluate Agent**" + ] + }, + { + "cell_type": "code", + "execution_count": 25, + "metadata": { + "colab": { + "base_uri": "https://localhost:8080/", + "height": 527 + }, + "id": "lwmM62mQnfC3", + "outputId": "ee5fa4e7-f070-421a-a0e4-5720baec641e" + }, + "outputs": [ + { + "name": "stderr", + "output_type": "stream", + "text": [ + "/usr/local/lib/python3.10/dist-packages/ipykernel/ipkernel.py:283: DeprecationWarning: `should_run_async` will not call `transform_cell` automatically in the future. Please pass the result to `transformed_cell` argument and any exception that happen during thetransform in `preprocessing_exc_tuple` in IPython 7.17 and above.\n", + " and should_run_async(code)\n" + ] + }, + { + "data": { + "image/png": "iVBORw0KGgoAAAANSUhEUgAAAksAAAHHCAYAAACvJxw8AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/bCgiHAAAACXBIWXMAAA9hAAAPYQGoP6dpAABGwklEQVR4nO3deVxWZf7/8ffNvsjiAq64AE5oWprmjqJZWqlZWaa5zRiW2uZSaqW4lFtmNY5pZWGLZbbYmFtSZlnqjJNLaUKRogaCqAWuqHD9/ujH/e0OON63gnDb6/l43I/xnHOdc3/OBXS/55zrXLfNGGMEAACAYnmUdwEAAAAVGWEJAADAAmEJAADAAmEJAADAAmEJAADAAmEJAADAAmEJAADAAmEJAADAAmEJAADAAmEJAC6TIUOGqH79+qV6zMWLF8tmsyktLa1Uj4vf2Ww2TZ48ubzLQDkjLAEXUPhhVPjy8/NTrVq11K1bN/3zn//U8ePHS9z3m2++0e23367q1avL19dX9evX1wMPPKCDBw8WaTt58mTZbDZVr15dp06dKrK9fv366tGjh0u1t2rVSjabTQsWLHBpv8shIyNDkydP1o4dO1za7+eff9b999+vyMhI+fn5KTg4WO3bt9eLL76o06dPl02xFcD06dP18ccfl3cZdn/+u/Dy8lLt2rU1ZMgQpaenl3d5QKkiLAFOmjp1qt566y0tWLBADz30kCTp0UcfVdOmTfXdd98VaT9v3jzFxsbq+++/10MPPaSXXnpJffr00dKlS3XNNddoy5Ytxb7P4cOHSyXc/PTTT9q6davq16+vJUuWXPLxSltGRoamTJniUlhatWqVmjZtqmXLlqlnz56aN2+eZsyYobp16+qxxx7TI488UnYFl7OSwtLAgQN1+vRp1atX7/IXpf/7u1i4cKFuvvlmvf322+rUqZPOnDlTLvUAZcGrvAsA3MXNN9+sli1b2pcnTJig9evXq0ePHurVq5f27Nkjf39/Sb9fUXr00UfVoUMHrV27VgEBAfb9hg8frvbt2+vOO+/U7t27FRoa6vA+zZo107PPPqsRI0bYj3cx3n77bYWHh+u5555Tnz59lJaWVuq3gC6nffv26Z577lG9evW0fv161axZ075t5MiRSk1N1apVq8qxwvLh6ekpT0/Pcnv/P/5d3HfffapWrZpmzZqlFStW6O677y63upx18uRJBQYGlncZqOC4sgRcgi5dumjixInav3+/3n77bfv6adOmyWaz6Y033nAISpIUFRWl2bNnKyMjQ6+88kqRY06aNElZWVmXfHXpnXfeUZ8+fdSjRw+FhITonXfeKbbdhg0b1LJlS/n5+SkqKkovv/yy/Zbgn7399ttq0aKF/P39VaVKFd1zzz1FbinGxcWpSZMm+uGHH9S5c2cFBASodu3amj17tsN7Xn/99ZKkv//97/ZbOYsXLy7xfGbPnq0TJ07otddecwhKhaKjo+1XltLS0ko83p/HoBSe648//qgBAwYoJCREYWFhmjhxoowxOnjwoG677TYFBwerRo0aeu655xyOV9KYoQ0bNshms2nDhg0lnpMkzZkzR+3atVPVqlXl7++vFi1a6IMPPihS88mTJ/XGG2/Y+2rIkCHFvn+PHj0UGRlZ7Hu1bdvWIfBLzv1MXREbGyvp99ulf5ScnKw+ffqoSpUq8vPzU8uWLbVixQr79t9++02enp765z//aV935MgReXh4qGrVqjLG2NcPHz5cNWrUsC9v3LhRd911l+rWrStfX19FRERo1KhRRW7LDhkyRJUqVdLPP/+sW265RUFBQbr33nslSXl5eRo1apTCwsIUFBSkXr166ZdffrnofsCVhbAEXKKBAwdKktatWydJOnXqlD7//HPFxsaqQYMGxe7Tt29f+fr66pNPPimyLTY2Vl26dNHs2bMvegzOf/7zH6Wmpqpfv37y8fHRHXfcUeytuO3bt6t79+46evSopkyZoqFDh2rq1KnF3u555plnNGjQIDVs2FBz587Vo48+qs8//1wdO3bUb7/95tD2119/Vffu3XXttdfqueeeU0xMjMaNG6c1a9ZIkho1aqSpU6dKkoYNG6a33npLb731ljp27FjiOX3yySeKjIxUu3btLqpPLqRv374qKCjQzJkz1bp1az399NN64YUXdOONN6p27dqaNWuWoqOjNXbsWH311Vel9r4vvviimjdvrqlTp2r69Ony8vLSXXfd5XCV7K233pKvr69iY2PtfXX//feXeB779u3T1q1bHdbv379fW7Zs0T333GNf58rP1FmFoa1y5cr2dbt371abNm20Z88ejR8/Xs8995wCAwPVu3dvLV++XJIUGhqqJk2aOPTt119/LZvNpmPHjumHH36wr9+4caM9lEnS+++/r1OnTmn48OGaN2+eunXrpnnz5mnQoEFF6jt//ry6deum8PBwzZkzR3feeaek36+KvfDCC7rppps0c+ZMeXt769Zbb72oPsAVyACwlJiYaCSZrVu3ltgmJCTENG/e3BhjzI4dO4wk88gjj1ge95prrjFVqlSxLyckJBhJJjs723z55ZdGkpk7d659e7169cytt97qVM0PPvigiYiIMAUFBcYYY9atW2ckme3btzu069mzpwkICDDp6en2dT/99JPx8vIyf/zPQ1pamvH09DTPPPOMw/7ff/+98fLycljfqVMnI8m8+eab9nV5eXmmRo0a5s4777Sv27p1q5FkEhMTL3g+OTk5RpK57bbbnDl9s2/fvhKPLckkJCTYlwv7fdiwYfZ158+fN3Xq1DE2m83MnDnTvv7XX381/v7+ZvDgwfZ1hb8f+/btc3ifL774wkgyX3zxhX3d4MGDTb169RzanTp1ymH57NmzpkmTJqZLly4O6wMDAx3et6T3z8nJMb6+vmbMmDEO7WbPnm1sNpvZv3+/Mca1n2lxCt/3s88+M9nZ2ebgwYPmgw8+MGFhYcbX19ccPHjQ3vaGG24wTZs2NWfOnLGvKygoMO3atTMNGza0rxs5cqSpXr26fXn06NGmY8eOJjw83CxYsMAYY8zRo0eNzWYzL774or3dn/vQGGNmzJjhcL7G/N7/ksz48eMd2hb+zY4YMcJhff/+/Yv8vuCviStLQCmoVKmS/am4wv8NCgqy3CcoKKjEJ+k6duyozp07X9TVpfPnz+u9995T37597bfSunTpovDwcIerS/n5+frss8/Uu3dv1apVy74+OjpaN998s8MxP/roIxUUFOjuu+/WkSNH7K8aNWqoYcOG+uKLLxzaV6pUSQMGDLAv+/j4qFWrVtq7d69L51IoNzdX0oX79FLcd9999n97enqqZcuWMsZo6NCh9vWhoaG66qqrLvo8ivPHcWm//vqrcnJyFBsbq23btl3U8YKDg3XzzTdr2bJlDreu3nvvPbVp00Z169aV5PrPtCRdu3ZVWFiYIiIi1KdPHwUGBmrFihWqU6eOJOnYsWNav3697r77bh0/ftz+PkePHlW3bt30008/2Z+ei42NVVZWllJSUiT9fgWpY8eOio2N1caNGyX9frXJGONwZemPfXjy5EkdOXJE7dq1kzFG27dvL1Lz8OHDHZZX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du3a6t+/v1599dWyqFHp6emKi4tT3bp1NWfOHGVnZyszM9NhXFGjRo3UpEkT+6t27dry8PBQkyZN7GFu+fLliomJse9z0003qXHjxho4cKB27typTz/9VE899ZRGjhzJbOQAAEDSRVxZstlsuuaaa3TNNdfooYce0rfffqt//etfWrJkid577z2HcUSlJSkpSampqUpNTVWdOnUctrlyFzEnJ0cpKSn2ZU9PT61cuVLDhw9X27ZtFRgYqMGDB2vq1KmlVjsAAHBvLo9Z2rZtmzZs2KANGzbo66+/1vHjx9W0aVP7+KXbbrutrGqtsBizBACA+ymzMUutWrVS8+bN1alTJ8XHx6tjx44KCQm5pGIBAAAqKpfD0rFjx7h6AgAA/jJcHuAdHBys3377TYsWLdKECRN07NgxSb/fnktPTy/1AgEAAMqTy1eWvvvuO91www0KDQ1VWlqa4uPjVaVKFX300Uc6cOCA3nzzzbKoEwAAoFy4fGVp9OjR+vvf/66ffvpJfn5+9vW33HKLvvrqq1ItDgAAoLy5HJa2bt2q+++/v8j62rVrW36fGgAAgDtyOSz5+voqNze3yPoff/zR6dm4AQAA3IXLYalXr16aOnWqzp07J+n3SSoPHDigcePG6c477yz1AgEAAMqTy2Hpueee04kTJxQeHq7Tp0+rU6dOio6OVqVKlfTMM8+URY0AAADlxuWn4UJCQpSUlKSvv/5a3333nU6cOKHrrrtOXbt2LYv6AAAAypXLX3dSkm3btmnSpElauXJlaRzOrfB1JwAAuB9nP79dug336aefauzYsXriiSe0d+9eSVJycrJ69+6t66+/XgUFBZdWNQAAQAXj9G241157zT4B5a+//qpFixZp7ty5euihh9S3b1/t2rVLjRo1KstaAQAALjunryy9+OKLmjVrlo4cOaJly5bpyJEjeumll/T9999r4cKFBCUAAHBFcnrMUmBgoHbv3q369evLGCNfX1998cUXat++fVnXWOExZgkAAPdT6mOWTp8+rYCAAEm/z63k6+urmjVrXnqlAAAAFZhLUwcsWrRIlSpVkiSdP39eixcvVrVq1RzaPPzww6VXHQAAQDlz+jZc/fr1ZbPZrA9ms9mfkvsr4TYcAADux9nPb6evLKWlpZVGXQAAAG7F5a87AQAA+CshLAEAAFggLAEAAFggLAEAAFgg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suTt7V3idi8vL2VnZ5dKUQAAABWF02Gpdu3a2rVrV4nbv/vuO9WsWbNUigIAAKgonA5Lt9xyiyZOnKgzZ84U2Xb69GklJCSoR48epVocAABAebOZP88uWYKsrCxdd9118vT01IMPPqirrrpKkpScnKz58+crPz9f27ZtU/Xq1cu04IooNzdXISEhysnJKXYAPAAAqHic/fx2ep6l6tWra9OmTRo+fLgmTJhgn8HbZrOpW7dumj9//l8yKAEAgCubS5NS1qtXT6tXr9avv/6q1NRUGWPUsGFDVa5cuazqAwAAKFcuz+AtSZUrV9b1119f2rUAAABUOE4P8AYAAPgrIiwBAABYICwBAABYICwBAABYICwBAABYICwBAABYICwBAABYICwBAABYICwBAABYICwBAABYICwBAABYICwBAABYICwBAABYICwBAABYICwBAABYICwBAABYICwBAABYICwBAABYICwBAABYICwBAABYICwBAABYICwBAABYICwBAABYcIuwlJaWpqFDh6pBgwby9/dXVFSUEhISdPbsWYc2NputyGvLli2Wxz5w4IBuvfVWBQQEKDw8XI899pjOnz9f1qcEAADchFd5F+CM5ORkFRQU6OWXX1Z0dLR27dql+Ph4nTx5UnPmzHFo+9lnn+nqq6+2L1etWrXE4+bn5+vWW29VjRo1tGnTJh06dEiDBg2St7e3pk+fXmbnAwAA3IfNGGPKu4iL8eyzz2rBggXau3evpN+vLDVo0EDbt29Xs2bNnDrGmjVr1KNHD2VkZKh69eqSpIULF2rcuHHKzs6Wj4+PU8fJzc1VSEiIcnJyFBwcfFHnAwAALi9nP7/d4jZccXJyclSlSpUi63v16qXw8HB16NBBK1assDzG5s2b1bRpU3tQkqRu3bopNzdXu3fvLnG/vLw85ebmOrwAAMCVyS3DUmpqqubNm6f777/fvq5SpUp67rnn9P7772vVqlXq0KGDevfubRmYMjMzHYKSJPtyZmZmifvNmDFDISEh9ldERMQlnhEAAKioyjUsjR8/vthB2X98JScnO+yTnp6u7t2766677lJ8fLx9fbVq1TR69Gi1bt1a119/vWbOnKkBAwbo2WefLfW6J0yYoJycHPvr4MGDpf4eAACgYijXAd5jxozRkCFDLNtERkba/52RkaHOnTurXbt2euWVVy54/NatWyspKanE7TVq1NB///tfh3VZWVn2bSXx9fWVr6/vBd8fAAC4v3INS2FhYQoLC3OqbXp6ujp37qwWLVooMTFRHh4Xvii2Y8cO1axZs8Ttbdu21TPPPKPDhw8rPDxckpSUlKTg4GA1btzYuZMAAABXNLeYOiA9PV1xcXGqV6+e5syZo+zsbPu2witAb7zxhnx8fNS8eXNJ0kcffaTXX39dixYtsrddvny5JkyYYL+1d9NNN6lx48YaOHCgZs+erczMTD311FMaOXIkV44AAIAkNwlLSUlJSk1NVWpqqurUqeOw7Y8zH0ybNk379++Xl5eXYmJi9N5776lPnz727Tk5OUpJSbEve3p6auXKlRo+fLjatm2rwMBADR48WFOnTi37kwIAAG7BbedZqkiYZwkAAPdzxc+zBAAAcDkQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACwQlgAAACy4RVhKS0vT0KFD1aBBA/n7+ysqKkoJCQk6e/asQxubzVbktWXLFstjF7fP0qVLy/qUAACAm/Aq7wKckZycrIKCAr388suKjo7Wrl27FB8fr5MnT2rOnDkObT/77DNdffXV9uWqVate8PiJiYnq3r27fTk0NLTUagcAAO7NLcJS9+7dHcJMZGSkUlJStGDBgiJhqWrVqqpRo4ZLxw8NDXV5HwAA8NfgFrfhipOTk6MqVaoUWd+rVy+Fh4erQ4cOWrFihVPHGjlypKpVq6ZWrVrp9ddflzGmtMsFAABuyi2uLP1Zamqq5s2b53BVqVKlSnruuefU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4soSALfm6+urGjVqOLwqV64sSbLZbFqwYIFuvvlm+fv7KzIyUh988IF93z/fhvv111917733KiwsTP7+/mrYsKFDEPv+++/VpUsX+fv7q2rVqho2bJhOnDhh356fn6/Ro0crNDRUVatW1eOPP17k2+YLCgo0Y8YMNWjQQP7+/rr22msdagJQ8RCWAFzRJk6cqDvvvFM7d+7Uvffeq3vuuUd79uwpse0PP/ygNWvWaM+ePVqwYIGqVasmSTp58qS6deumypUra+vWrXr//ff12Wef6cEHH7Tv/9xzz2nx4sV6/fXX9fXXX+vYsWNavny5w3vMmDFDb775phYuXKjdu3dr1KhRGjBggL788suy6wQAl6ZMv6YXAMrQ4MGDjaenpwkMDHR4PfPMM8YYYySZBx54wGGf1q1bm+HDhxtjjNm3b5+RZLZv326MMaZnz57m73//e7Hv9corr5jKlSubEydO2NetWrXKeHh4mMzMTGOMMTVr1jSzZ8+2bz937pypU6eOue2224wxxpw5c8YEBASYTZs2ORx76NChpl+/fhffEQDKFGOWALi1zp07a8GCBQ7rqlSpYv9327ZtHba1bdu2xKffhg8frjvvvFPbtm3TTTfdpN69e6tdu3aSpD179ujaa69VYGCgvX379u1VUFCglJQU+fn56dChQ2rdurV9u5eXl1q2bGm/FZeamqpTp07pxhtvdHjfs2fPqnnz5q6fPIDLgrAEwK0FBgYqOjq6VI518803a//+/Vq9erWSkpJ0ww03aOTIkZozZ06pHL9wfNOqVatUu3Zth23ODkoHcPkxZgnAFW3Lli1Flhs1alRi+7CwMA0ePFhvv/22XnjhBb3yyiuSpEaNGmnnzp06efKkve0333wjDw8PXXXVVQoJCVHNmjX1n//8x779/Pnz+vbbb+3LjRs3lq+vrw4cOKDo6GiHV0RERGmdMoBSxpUlAG4tLy9PmZmZDuu8vLzsA7Pff/99tWzZUh06dNCSJUv03//+V6+99lqxx5o0aZJatGihq6++Wnl5eVq5cqU9WN17771KSEjQ4MGDNXnyZGVnZ+uhhx7SwIEDVb16dUnSI488opkzZ6phw4aKiYnR3Llz9dtvv9mPHxQUpLFjx2rUqFEqKChQhw4dlJOTo2+++UbBwcEaPHhwGfQQgEtFWALg1tauXauaNWs6rLvqqquUnJwsSZoyZYqWLl2qESNGqGbNmnr33XfVuHHjYo/l4+OjCRMmKC0tTf7+/oqNjdXSpUslSQEBAfr000/1yCOP6Prrr1dAQIDuvPNOzZ07177/mDFjdOjQIQ0ePFgeHh76xz/+odtvv105OTn2NtOmTVNYWJhmzJihvXv3KjQ0VNddd52eeOKJ0u4aAKXEZsyfJgEBgCuEzWbT8uXL+boRAJeEMUsAAAAWCEsAAAAWGLME4IrFKAMApYErSwAAABYISwAAABYISwAAABYISwAAABYISwAAABYISwAAABYISwAAABYISwAAABYISwAAABb+H44e8Z8JwvEwAAAAAElFTkSuQmCC", + "text/plain": [ + "
" + ] + }, + "metadata": {}, + "output_type": "display_data" + } + ], + "source": [ + "\n", + "import matplotlib.pyplot as plt\n", + "\n", + "# Plot the cumulative reward for each episode\n", + "episodes = range(EPISODES)\n", + "rewards = [0] * EPISODES\n", + "for i in range(EPISODES):\n", + " rewards[i] = sum([reward for _, _, reward, _, _ in agent.memory if reward != -10])\n", + "\n", + "plt.plot(episodes, rewards)\n", + "plt.xlabel('Episode')\n", + "plt.ylabel('Cumulative Reward')\n", + "plt.title('DQN Agent Cumulative Reward')\n", + "plt.show()\n" + ] + }, + { + "cell_type": "code", + "execution_count": 26, + "metadata": { + "colab": { + "base_uri": "https://localhost:8080/", + "height": 472 + }, + "id": "IRN2dYAKniV_", + "outputId": "41d121ef-75e3-4e0f-d8bb-134e25f025ff" + }, + "outputs": [ + { + "data": { + "image/png": "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eDDBD1wKAQjAZbkw+/F7u3fvLrIguUmTJho1apRGjRql7777TuHh4XrhhRe0cOHCEn1Ow4YNJUm7du0q8t7OnTtVt25dp9mf8rRgwQLZbDbdcsstkn47N0kKCgpSTEzMRfdr3LixJBW5S+6P3n77bcXHx+uFF15wtOXn5+vkyZNF+t52220KDAzUokWLFBkZqTNnzmjAgAGlPSXAOPxfBACX5d1339WhQ4ccr7du3aotW7aoZ8+ekqQzZ84oPz/faZ8mTZrIz89PBQUFJf6ckJAQhYeHa/78+U5B4JtvvtGHH36oXr16Xd6JlNCUKVP04Ycfqm/fvo7Lf7GxsfL399fkyZN17ty5IvscO3ZMkhQYGKg//elPmjNnjg4cOODU5/ezUp6enkVmqaZNm6bCwsIix65WrZruv/9+vfXWW5o3b57atm2rdu3aXfZ5Alc6ZoAAFOuDDz7Qzp07i7RHR0c7ZjKk3y4Dde3aVY8++qgKCgqUlpamOnXq6Mknn5T022zQzTffrD59+qhVq1aqVq2aVqxYoSNHjui+++4rVU3PPfecevbsqaioKD300EOO2+ADAgI0fvz4yzrfPzp//rxjdio/P1/79+/XypUr9dVXX+nGG2/U7NmzHX39/f01c+ZMDRgwQB07dtR9992nwMBAHThwQKtWrVKXLl00ffp0SdK//vUvde3aVR07dtTgwYPVqFEj7du3T6tWrdL27dslSXfccYcWLFiggIAAtWrVSps3b9b69euL/ATBBQMHDtS//vUvffzxx5d9iz9gCgIQgGKNGzeu2Pa5c+c6BaCBAwfKw8NDaWlpOnr0qDp37qzp06crJCREkhQaGqr7779f6enpWrBggapVq6YWLVrorbfeUu/evUtVU0xMjNasWaPk5GSNGzdO1atXV/fu3TV16tRiH9txOQoKChyXkmrUqKGgoCB16tRJ48aN0z333FNkjU2/fv1Uv359TZkyRc8995wKCgrUoEEDdevWTQkJCY5+7du312effaakpCTNnDlT+fn5atiwofr06ePo89JLL8nT01OLFi1Sfn6+unTpovXr1ys2NrbYWjt16qTWrVsrKytL/fv3L9NxAK5UNutyVwMCMNK+ffvUqFEjPffcc/r73//u7nKM16FDB1199dVKT093dylAlcAaIACo4r744gtt375dAwcOdHcpQJXBJTAA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", + "text/plain": [ + "
" + ] + }, + "metadata": {}, + "output_type": "display_data" + } + ], + "source": [ + "# Plot the epsilon decay over episodes\n", + "plt.plot(episodes, [agent.epsilon for i in range(EPISODES)])\n", + "plt.xlabel('Episode')\n", + "plt.ylabel('Epsilon')\n", + "plt.title('Epsilon Decay')\n", + "plt.show()\n" + ] + }, + { + "cell_type": "code", + "execution_count": 27, + "metadata": { + "colab": { + "base_uri": "https://localhost:8080/" + }, + "id": "5PNJWRBmnlBt", + "outputId": "7abb130f-3299-4ef8-e67c-e857fc16aacf" + }, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "1/1 [==============================] - 0s 36ms/step\n", + "1/1 [==============================] - 0s 37ms/step\n" + ] + } + ], + "source": [ + "# Evaluate the agent\n", + "state = env.reset()\n", + "state = np.reshape(state, [1, state_size]).astype(np.float32)\n", + "for time in range(200):\n", + " action = agent.act(state)\n", + " next_state, reward, done, _ = env.step(action)\n", + " next_state = np.reshape(next_state, [1, state_size]).astype(np.float32)\n", + " state = next_state\n", + " if done:\n", + " break\n" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "Ffmk3EcXpLzL" + }, + "source": [ + "**Saving the model**" + ] + }, + { + "cell_type": "code", + "execution_count": 28, + "metadata": { + "colab": { + "base_uri": "https://localhost:8080/" + }, + "id": "z_2HH4SRlhXY", + "outputId": "01aa3b0d-8262-45a0-f821-2895e1a97833" + }, + "outputs": [ + { + "name": "stderr", + "output_type": "stream", + "text": [ + "/usr/local/lib/python3.10/dist-packages/ipykernel/ipkernel.py:283: DeprecationWarning: `should_run_async` will not call `transform_cell` automatically in the future. Please pass the result to `transformed_cell` argument and any exception that happen during thetransform in `preprocessing_exc_tuple` in IPython 7.17 and above.\n", + " and should_run_async(code)\n", + "/usr/local/lib/python3.10/dist-packages/keras/src/engine/training.py:3103: UserWarning: You are saving your model as an HDF5 file via `model.save()`. This file format is considered legacy. We recommend using instead the native Keras format, e.g. `model.save('my_model.keras')`.\n", + " saving_api.save_model(\n" + ] + } + ], + "source": [ + "# prompt: after doing the evaluation of agent what can we do here\n", + "\n", + "# Save the model\n", + "agent.model.save('dqn_agent.h5')\n", + "\n", + "# Load the model\n", + "agent.model.load_weights('dqn_agent.h5')\n", + "\n", + "# Use the model to make predictions on new data\n", + "new_state = env.reset()\n", + "new_state = np.reshape(new_state, [1, state_size]).astype(np.float32)\n", + "action = agent.act(new_state)\n", + "\n" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "0ipH2zFcpU3c" + }, + "source": [ + "**Testing the Model**" + ] + }, + { + "cell_type": "code", + "execution_count": 29, + "metadata": { + "colab": { + "base_uri": "https://localhost:8080/" + }, + "id": "7NesUUCNjwsG", + "outputId": "d57bd4f7-c2eb-4102-c9d9-87a981e00aad" + }, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Predicted action: 0\n" + ] + }, + { + "name": "stderr", + "output_type": "stream", + "text": [ + "/usr/local/lib/python3.10/dist-packages/ipykernel/ipkernel.py:283: DeprecationWarning: `should_run_async` will not call `transform_cell` automatically in the future. Please pass the result to `transformed_cell` argument and any exception that happen during thetransform in `preprocessing_exc_tuple` in IPython 7.17 and above.\n", + " and should_run_async(code)\n" + ] + } + ], + "source": [ + "\n", + "import numpy as np\n", + "\n", + "# Load the trained model\n", + "agent.model.load_weights('dqn_agent.h5')\n", + "\n", + "# Create a new state\n", + "new_state = np.array([0.5, 0.6, 0.7, 0.8, 0.9])\n", + "\n", + "# Reshape the state to match the model input\n", + "new_state = np.reshape(new_state, [1, state_size]).astype(np.float32)\n", + "\n", + "# Predict the action using the loaded model\n", + "action = agent.act(new_state)\n", + "\n", + "# Print the predicted action\n", + "print(\"Predicted action:\", action)\n" + ] + } + ], + "metadata": { + "colab": { + "provenance": [] + }, + "kernelspec": { + "display_name": "Python 3", + "name": "python3" + }, + "language_info": { + "name": "python" + } + }, + "nbformat": 4, + "nbformat_minor": 0 +} diff --git a/Finacial Domain/Dynamic Hedging Strategies using Reinforcement Learning/dqn_agent.h5 b/Finacial Domain/Dynamic Hedging Strategies using Reinforcement Learning/dqn_agent.h5 new file mode 100644 index 00000000..2d7539b7 Binary files /dev/null and b/Finacial Domain/Dynamic Hedging Strategies using Reinforcement Learning/dqn_agent.h5 differ diff --git a/Finacial Domain/Dynamic Hedging Strategies using Reinforcement Learning/readme.md b/Finacial Domain/Dynamic Hedging Strategies using Reinforcement Learning/readme.md new file mode 100644 index 00000000..6db05e3a --- /dev/null +++ b/Finacial Domain/Dynamic Hedging Strategies using Reinforcement Learning/readme.md @@ -0,0 +1,23 @@ +# Reinforcement Learning Model for Dynamic Hedging Strategies + +## Project Description- + +This project involves implementing a reinforcement learning (RL) model to develop dynamic hedging strategies for financial derivatives. The objective is to create an adaptive hedging strategy that optimizes the portfolio performance by dynamically adjusting positions based on market conditions. + +## Key Objectives- + +1. Data Collection and Preprocessing: Gather historical market data and preprocess it for analysis. +2. Environment Setup: Create a financial environment that simulates the market for training the RL model. +3. Model Development: Develop and train the RL model using stable-baselines3. +4. Evaluation and Validation: Evaluate the model's performance and validate its effectiveness. + +## Project Structure + +- `data/`: Contains historical market data used for training and testing. +- `notebooks/`: Jupyter notebooks for data analysis, model development, and evaluation. +- `src/`: Source code for data preprocessing, environment setup, and model implementation. +- `models/`: Saved RL models. +- `results/`: Evaluation results and visualizations. +- `README.md`: Project overview and instructions. +- `requirements.txt`: List of dependencies. + diff --git a/Finacial Domain/Dynamic Hedging Strategies using Reinforcement Learning/requirement.txt b/Finacial Domain/Dynamic Hedging Strategies using Reinforcement Learning/requirement.txt new file mode 100644 index 00000000..ea92a31b --- /dev/null +++ b/Finacial Domain/Dynamic Hedging Strategies using Reinforcement Learning/requirement.txt @@ -0,0 +1,12 @@ +### Dependencies to install before implementation +numpy +pandas +matplotlib +seaborn +scikit-learn +stable-baselines3[extra] +gym +tensorflow +torch +plotly +