{
 "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": [
    "# Total demands"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Loading the demands and initializing the environment"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "demands = torch.FloatTensor(pd.read_csv(os.path.join(policy_selection_for_inventories.__path__[0], \"..\", \"..\", \"data\", \"m5_processed\", \"demands_total.csv\"), index_col=0)[\"y\"])\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": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "NB_SAMPLE = 10"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "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": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:12<00:00,  1.38it/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 48403.77734375\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": 9,
   "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": 19,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "0it [00:00, ?it/s]"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "32it [02:13,  4.18s/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": 31,
   "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": 34,
   "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.256\n",
      "Algorithm: $\\hat{S}_t$\n",
      "\tMean: 1.196\n",
      "\tStd.: 0.008\n",
      "\tAvg. time: 0.216\n",
      "Algorithm: MPC\n",
      "\tMean: 1.199\n",
      "\tStd.: 0.008\n",
      "\tAvg. time: 12.377\n",
      "Algorithm: GAPSI without forecasts\n",
      "\tMean: 0.952\n",
      "\tStd.: 0.0\n",
      "\tAvg. time: 0.674\n",
      "Algorithm: GAPSI with forecasts\n",
      "\tMean: 0.91\n",
      "\tStd.: 0.002\n",
      "\tAvg. time: 0.691\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
}
