{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "from torch.nn.functional import relu\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import os\n",
    "\n",
    "from policy_selection_for_inventories.simulator import Simulator\n",
    "\n",
    "from policy_selection_for_inventories.algorithms.gapsi_special import GAPSI_Special\n",
    "\n",
    "from policy_selection_for_inventories.algorithms.mpc import MPC\n",
    "from policy_selection_for_inventories.algorithms.base_stock import Base_Stock\n",
    "import policy_selection_for_inventories\n",
    "\n",
    "\n",
    "from policy_selection_for_inventories.environments.ls_fifop_product import LS_FIFOP_Product\n",
    "from policy_selection_for_inventories.environments.infinite_warehouse import Infinite_Warehouse\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "lifetime = 2\n",
    "leadtime = 0\n",
    "\n",
    "MPC_horizon = 7"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Category demands"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Loading the demands and initializing the environment"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "df = pd.read_csv(\n",
    "    os.path.join(policy_selection_for_inventories.__path__[0], \"..\", \"..\", \"data\", \"m5_processed\", \"demands_category.csv\"),\n",
    "    index_col=[\"cat_id\",\"ds\"]\n",
    ")\n",
    "df = df.join(df.assign().groupby(\"cat_id\")[\"y\"].mean().rename(\"mean\"), on=\"cat_id\", how=\"left\")\n",
    "df[\"normalized_y\"] = df[\"y\"]/df[\"mean\"]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Categories ordered by normalized standard deviation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "cat_id\n",
       "FOODS        0.210176\n",
       "HOBBIES      0.222655\n",
       "HOUSEHOLD    0.290718\n",
       "Name: norm_std, dtype: float64"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "ordered_norm_std = df.assign().groupby(\"cat_id\")[\"normalized_y\"].std().rename(\"norm_std\").sort_values()\n",
    "ordered_norm_std"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Median index"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Median index: HOBBIES\n"
     ]
    }
   ],
   "source": [
    "median_index = ordered_norm_std.index.values[(len(ordered_norm_std)-1)//2]\n",
    "print(\"Median index: {}\".format(median_index))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "demands = torch.FloatTensor(df.loc[median_index, \"y\"].values)\n",
    "#demands /= demands.mean()\n",
    "T = len(demands)\n",
    "\n",
    "purchase_cost = 1.0*torch.ones(T)\n",
    "holding_cost = 1.0*torch.ones(T)\n",
    "selling_price = 0.0*torch.ones(T)\n",
    "outdating_cost = 1.0*torch.ones(T)\n",
    "penalty_cost = 10.0*torch.ones(T)\n",
    "\n",
    "product = LS_FIFOP_Product(lifetime, leadtime, demands, purchase_cost, holding_cost, selling_price, outdating_cost, penalty_cost, 10*T*demands.numpy().max())\n",
    "environment = Infinite_Warehouse([product])"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Computing forecasts"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "NB_SAMPLE = 10"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "true_leadtime_demand_forecasts = torch.cat([torch.zeros(7),demands,torch.zeros(leadtime)]).unfold(0,leadtime+1,1).sum(dim=1)[:T]\n",
    "\n",
    "SEED = 1\n",
    "torch.manual_seed(SEED)\n",
    "noise_theoretical_std = demands.std()/2\n",
    "noise = noise_theoretical_std*torch.randn(NB_SAMPLE,T)\n",
    "\n",
    "leadtime_demand_forecasts = []\n",
    "fake_environments = []\n",
    "\n",
    "for sample_idx in range(NB_SAMPLE) :\n",
    "    noisy_demands = relu(demands+noise[sample_idx])\n",
    "    leadtime_demand_forecasts.append(\n",
    "        torch.cat([torch.zeros(7),noisy_demands,torch.zeros(leadtime)]).unfold(0,leadtime+1,1).sum(dim=1)[:T]   \n",
    "    )\n",
    "\n",
    "    fake_product = LS_FIFOP_Product(\n",
    "        lifetime, leadtime, torch.cat([torch.zeros(MPC_horizon), relu(demands+noise[sample_idx])]), purchase_cost, holding_cost, selling_price, outdating_cost, penalty_cost, 10*T*demands.numpy().max()\n",
    "    )\n",
    "    fake_environments.append(Infinite_Warehouse([fake_product]))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Computing the best base-stock level $S_T^*$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:11<00:00,  1.40it/s]\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Final cost 4586.16259765625\n"
     ]
    }
   ],
   "source": [
    "levels_1 = policy_selection_for_inventories.utils.find_best_S_policy(\n",
    "    environment=environment,\n",
    "    T=T,\n",
    "    seasonality=[1],\n",
    "    initial_levels=[demands[:T].unfold(0,leadtime+1,1).sum(dim=1).mean().reshape(1)],\n",
    "    nb_iterations=100,\n",
    "    learning_rate=500.0,\n",
    "    show_progress_bar=True\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Running the algorithms"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "algorithms = [Base_Stock(environment, [levels_1[0]*torch.ones(T)])]\n",
    "algorithms += [Base_Stock(environment, [leadtime_demand_forecasts[sample_idx]]) for sample_idx in range(NB_SAMPLE)]\n",
    "algorithms += [MPC(fake_environments[sample_idx], MPC_horizon) for sample_idx in range(NB_SAMPLE)]\n",
    "algorithms += [GAPSI_Special(environment, [1], 0.1, 50, torch.zeros(1), torch.zeros(1), torch.ones(1),\n",
    "    [(leadtime+1)*demands.max()*torch.ones(T,1)])\n",
    "]\n",
    "algorithms += [GAPSI_Special(environment, [2], 0.1, 50, torch.zeros(2), torch.zeros(2), torch.ones(2),\n",
    "    [torch.stack([2*leadtime_demand_forecasts[sample_idx], (leadtime+1)*demands.max()*torch.ones(T)], dim=1)])\n",
    "    for sample_idx in range(NB_SAMPLE)\n",
    "]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "32it [14:09, 26.56s/it]\n"
     ]
    }
   ],
   "source": [
    "states, controls, costs, parameters, computation_times = Simulator(environment,algorithms, True, False).run(T)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Comparing the algorithms"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "alg_names = [r\"$S^*_T$\", r\"$\\hat{S}_t$\", \"MPC\", \"GAPSI without forecasts\", \"GAPSI with forecasts\"]\n",
    "alg_mean = []\n",
    "alg_std = []\n",
    "alg_time = []\n",
    "\n",
    "alg_mean.append(costs[0].mean())\n",
    "alg_std.append(0.0)\n",
    "alg_time.append(round(computation_times[0], 2))\n",
    "\n",
    "alg_mean.append(torch.stack(costs)[1:1+NB_SAMPLE].mean(dim=1).mean())\n",
    "alg_std.append(torch.stack(costs)[1:1+NB_SAMPLE].mean(dim=1).std())\n",
    "alg_time.append(round(np.array(computation_times[1:1+NB_SAMPLE]).mean(), 2))\n",
    "\n",
    "alg_mean.append(torch.stack(costs)[1+NB_SAMPLE:1+2*NB_SAMPLE].mean(dim=1).mean())\n",
    "alg_std.append(torch.stack(costs)[1+NB_SAMPLE:1+2*NB_SAMPLE].mean(dim=1).std())\n",
    "alg_time.append(round(np.array(computation_times[1+NB_SAMPLE:1+2*NB_SAMPLE]).mean(), 2))\n",
    "\n",
    "alg_mean.append(costs[1+2*NB_SAMPLE].mean())\n",
    "alg_std.append(0.0)\n",
    "alg_time.append(round(computation_times[1+2*NB_SAMPLE], 2))\n",
    "\n",
    "alg_mean.append(torch.stack(costs)[2+2*NB_SAMPLE:2+3*NB_SAMPLE].mean(dim=1).mean())\n",
    "alg_std.append(torch.stack(costs)[2+2*NB_SAMPLE:2+3*NB_SAMPLE].mean(dim=1).std())\n",
    "alg_time.append(round(np.array(computation_times[2+2*NB_SAMPLE:2+3*NB_SAMPLE]).mean(), 2))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Ratio of losses\n",
      "\n",
      "Algorithm: $S^*_T$\n",
      "\tMean: 1.0\n",
      "\tStd.: 0.0\n",
      "\tAvg. time: 0.228\n",
      "Algorithm: $\\hat{S}_t$\n",
      "\tMean: 1.219\n",
      "\tStd.: 0.009\n",
      "\tAvg. time: 0.222\n",
      "Algorithm: MPC\n",
      "\tMean: 1.219\n",
      "\tStd.: 0.009\n",
      "\tAvg. time: 83.983\n",
      "Algorithm: GAPSI without forecasts\n",
      "\tMean: 0.944\n",
      "\tStd.: 0.0\n",
      "\tAvg. time: 0.672\n",
      "Algorithm: GAPSI with forecasts\n",
      "\tMean: 0.904\n",
      "\tStd.: 0.002\n",
      "\tAvg. time: 0.677\n"
     ]
    }
   ],
   "source": [
    "ratio_of_losses_mean = [round((alg_mean[i]/alg_mean[0]).item(), 3) for i in range(1,len(alg_mean))]\n",
    "ratio_of_losses_std = [round((alg_std[i]/alg_mean[0]).item(), 3) for i in range(1,len(alg_mean))]\n",
    "\n",
    "print(\"Ratio of losses\\n\")\n",
    "for i in range(len(alg_names)) :\n",
    "    print(\"Algorithm: {}\\n\\tMean: {}\\n\\tStd.: {}\\n\\tAvg. time: {}\".format(\n",
    "        alg_names[i],\n",
    "        1.000 if (i==0) else ratio_of_losses_mean[i-1],\n",
    "        0.000 if (i==0) else ratio_of_losses_std[i-1],\n",
    "        alg_time[i]\n",
    "    ))"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "policy_selection_for_inventories",
   "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.6"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
