{
 "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",
    "import scipy\n",
    "from math import ceil\n",
    "\n",
    "from policy_selection_for_inventories.simulator import Simulator\n",
    "\n",
    "from policy_selection_for_inventories.algorithms.gapsi_special_constrained import GAPSI_Special_Constrained\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",
    "\n",
    "import policy_selection_for_inventories\n",
    "import policy_selection_for_inventories.utils\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"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "df = pd.read_csv(\n",
    "    os.path.join(policy_selection_for_inventories.__path__[0], \"..\", \"..\", \"data\", \"m5_processed\", \"demands_product.csv\"),\n",
    "    index_col=[\"item_id\",\"ds\"]\n",
    ")\n",
    "product_names = df.index.levels[0]\n",
    "datestamps = df.index.levels[1]\n",
    "demands = [ torch.FloatTensor(df.loc[p_name].values.reshape(len(datestamps))) for p_name in product_names ]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "lifetime = 3\n",
    "leadtime = 0\n",
    "T = len(demands[0])\n",
    "K = len(demands)\n",
    "V = torch.stack(demands)[:K,:T].sum(dim=0).quantile(0.9).item()\n",
    "\n",
    "purchase_cost = 1*torch.ones(T)\n",
    "holding_cost = 1*torch.ones(T)\n",
    "selling_price = 0*torch.ones(T)\n",
    "outdating_cost = 1*torch.ones(T)\n",
    "penalty_cost = 10*torch.ones(T)\n",
    "#overflow_cost = 5*torch.ones(T)\n",
    "\n",
    "products = [\n",
    "    LS_FIFOP_Product(lifetime, leadtime, demands[k], purchase_cost, holding_cost, selling_price, outdating_cost, penalty_cost, 10*T*demands[k].numpy().max())\n",
    "    for k in range(K)\n",
    "]\n",
    "environment = Infinite_Warehouse(products)# torch.ones(K), V, [overflow_cost]*K)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "algorithms_name = [\"GAPSI-1\"]\n",
    "algorithms = [None]*len(algorithms_name)\n",
    "features = [None]*len(algorithms_name)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "NB_PARAMS_PER_PROD = 1\n",
    "initial_parameter = torch.zeros(NB_PARAMS_PER_PROD*K)#torch.cat([(leadtime+1)*demands[k].mean().reshape(1) for k in range(K)])\n",
    "min_value = torch.zeros(K)\n",
    "max_value = torch.ones(K)\n",
    "features[0] = [(leadtime+1)*demands[k].max()*torch.ones(T,1) for k in range(K)]\n",
    "\n",
    "algorithms[0] = GAPSI_Special(environment, [NB_PARAMS_PER_PROD]*K, 0.1, 10, initial_parameter, min_value, max_value, features[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 1969/1969 [1:47:34<00:00,  3.28s/it]\n"
     ]
    }
   ],
   "source": [
    "states, controls, costs, parameters, computation_times = Simulator(environment, algorithms, False, True).run(T)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Final metrics"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Lost sales percentage of GAPSI-1 : 4.860000133514404\n",
      "Outdating percentage of GAPSI-1 : 4.740000247955322\n"
     ]
    }
   ],
   "source": [
    "for alg_idx, alg_name in enumerate(algorithms_name) :\n",
    "    local_lost_sales = torch.zeros(K,T)\n",
    "    local_outdating = torch.zeros(K,T)\n",
    "    for t in range(T) :\n",
    "        for p_idx, product in enumerate(environment.products_list) :\n",
    "            post_control_state = torch.cat([\n",
    "                states[alg_idx][t][environment.state_indexes[p_idx]:environment.state_indexes[p_idx]+product.state_dim],\n",
    "                controls[alg_idx][t][p_idx:p_idx+1]\n",
    "            ])\n",
    "            local_outdating[p_idx,t] = relu(post_control_state[0]-product.demands[t])\n",
    "            local_lost_sales[p_idx,t] = relu(product.demands[t] - post_control_state[:product.lifetime].sum())\n",
    "    print(\"Lost sales percentage of {} : {}\".format(\n",
    "        alg_name, \n",
    "        100*(local_lost_sales.sum()/torch.stack(demands)[:K,:T].sum()).round(decimals=4)\n",
    "    ))\n",
    "    print(\"Outdating percentage of {} : {}\".format(\n",
    "        alg_name,\n",
    "        100*(local_outdating.sum()/controls[alg_idx].sum()).round(decimals=4)\n",
    "    ))"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "inventory_policy_selection",
   "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
}
