Pop-up shop#
# install dependencies and select solver
%pip install -q amplpy numpy pandas
SOLVER = "cbc"
from amplpy import AMPL, ampl_notebook
ampl = ampl_notebook(
modules=["cbc"], # modules to install
license_uuid="default", # license to use
) # instantiate AMPL object and register magics
import numpy as np
import pandas as pd
The problem: Maximizing the net profit of a pop-up shop#
There is an opportunity to operate a pop-up shop to sell a unique commemorative item for events held at a famous location. The items cost 12 € each when bought from the supplier and will sell for 40 €. Unsold items can be returned to the supplier at a value of only 2 € due to their commemorative nature.
Parameter |
Symbol |
Value |
---|---|---|
sales price |
\(r\) |
40 € |
unit cost |
\(c\) |
12 € |
salvage value |
\(w\) |
2 € |
Demand for these items, however, will be high only if the weather is good. Historical data suggests three typical scenarios, namely \(S=\{\text{sunny skies, good weather, poor weather}\}\), as detailed in the following table.
Scenario (\(s\)) |
Demand (\(d_s\)) |
Probability (\(p_s\)) |
---|---|---|
Sunny Skies |
650 |
0.10 |
Good Weather |
400 |
0.60 |
Poor Weather |
200 |
0.30 |
The problem is to determine how many items to order for the pop-up shop.
The dilemma is that the weather will not be known until after the order is placed. Ordering enough items to meet demand for a good weather day results in a financial penalty on returned goods if the weather is poor. On the other hand, ordering just enough items to satisfy demand on a poor weather day leaves “money on the table” if the weather is good.
How many items should be ordered for sale?
Expected value for the mean scenario (EVM)#
A naive solution to this problem is to place an order equal to the expected demand, which can be calculated as
Choosing an order size \(\hat{x} = \mathbb E[D] = 365\) results in an expected profit we call the expected value of the mean scenario (EVM). The resulting expected profit is given by
where \(f_s\) is the net profit in scenario \(s\) assuming that we ordered \(\hat{x}\) items.
These calculations can be executed using operations on the pandas dataframe. First, we create a pandas DataFrame object to store the scenario data and calculate the expected demand.
# price information
r = 40
c = 12
w = 2
# scenario information
scenarios = {
"sunny skies": {"probability": 0.10, "demand": 650},
"good weather": {"probability": 0.60, "demand": 400},
"poor weather": {"probability": 0.30, "demand": 200},
}
df = pd.DataFrame.from_dict(scenarios).T
display(df)
expected_demand = sum(df["probability"] * df["demand"])
print(f"Expected demand = {expected_demand}")
probability | demand | |
---|---|---|
sunny skies | 0.1 | 650.0 |
good weather | 0.6 | 400.0 |
poor weather | 0.3 | 200.0 |
Expected demand = 365.0
Subsequent calculations to obtain the EVM can be done directly within the pandas dataframe holding the scenario data.
df["order"] = expected_demand
df["sold"] = df[["demand", "order"]].min(axis=1)
df["salvage"] = df["order"] - df["sold"]
df["profit"] = r * df["sold"] + w * df["salvage"] - c * df["order"]
EVM = sum(df["probability"] * df["profit"])
display(df)
print(f"Expected value of the mean demand (EVM) = {EVM}")
probability | demand | order | sold | salvage | profit | |
---|---|---|---|---|---|---|
sunny skies | 0.1 | 650.0 | 365.0 | 365.0 | 0.0 | 10220.0 |
good weather | 0.6 | 400.0 | 365.0 | 365.0 | 0.0 | 10220.0 |
poor weather | 0.3 | 200.0 | 365.0 | 200.0 | 165.0 | 3950.0 |
Expected value of the mean demand (EVM) = 8339.0
No scenario shows a profit loss, which appears to be a satisfactory outcome. However, can we find an order resulting in a higher expected profit?
Value of the stochastic solution (VSS)#
In order to answer this question, let us formulate the problem in mathematical terms. Let \(x\) be a non-negative number representing the number of items that will be ordered, and \(y_s\) be the non-negative variable describing the number of items sold in scenario \(s\) in the set \(S\) comprising all scenarios under consideration. The number \(y_s\) of sold items is the lesser of the demand \(d_s\) and the order size \(x\), that is
Any unsold inventory \(x - y_s\) remaining after the event will be sold at the salvage price \(w\). Taking into account the revenue from sales \(r y_s\), the salvage value of the unsold inventory \(w(x - y_s)\), and the cost of the order \(c x\), the profit \(f_s\) for scenario \(s\) is given by
Using the constants introduced earlier, the profit \(f_s\) for scenario \(s \in S\) can then be written as
The expected profit is given by \(\mathbb E(F) = \sum_s p_s f_s\). Operationally, \(y_s\) can be no larger the number of items ordered, \(x\), or the demand under scenario \(s\), \(d_s\). The optimization problem is to find the order size \(x\) that maximizes expected profit subject to operational constraints on the decision variables. The variables \(x\) and \(y_s\) are non-negative integers, while \(f_s\) is a real number that can take either positive or negative values. Putting these facts together, the optimization problem to be solved is
where \(S\) is the set of all scenarios under consideration.
We can implement this problem in AMPL as follows.
%%writefile pop.mod
param r;
param c;
param w;
# set of scenarios
set S;
param p{S};
param d{S};
# decision variables
var x >= 0;
var y{S} >= 0;
var f{S};
# objective
maximize EV: sum{s in S} p[s] * f[s];
# constraints
s.t. profit {s in S}: f[s] == r * y[s] + w * (x - y[s]) - c * x;
s.t. sales_less_than_order {s in S}: y[s] <= x;
s.t. sales_less_than_demand {s in S}: y[s] <= d[s];
Overwriting pop.mod
# price and scenario information
r = 40
c = 12
w = 2
scenarios = {
"sunny skies": {"demand": 650, "probability": 0.1},
"good weather": {"demand": 400, "probability": 0.6},
"poor weather": {"demand": 200, "probability": 0.3},
}
# create a data frame with the data and rename the columns to match the model
scenarios_df = pd.DataFrame.from_dict(scenarios).T.rename(
columns={"demand": "d", "probability": "p"}
)
# Create AMPL instance and load the model
ampl = AMPL()
ampl.read("pop.mod")
# load the data
ampl.param["r"] = r
ampl.param["c"] = c
ampl.param["w"] = w
ampl.set_data(scenarios_df, "S")
# solve the problem
ampl.option["solver"] = SOLVER
ampl.solve()
print("Solver Termination Condition:", ampl.get_value("solve_result"))
print()
# display solution using Pandas
df = pd.DataFrame.from_dict(scenarios).T
df["order"] = ampl.var["x"].value()
df["sold"] = ampl.var["y"].get_values().toPandas()
df["salvage"] = df["order"] - df["sold"]
df["profit"] = ampl.var["f"].get_values().toPandas()
display(df)
print("Expected Profit:", ampl.obj["EV"].value())
cbc 2.10.7: cbc 2.10.7: optimal solution; objective 8920
0 simplex iterations
Solver Termination Condition: solved
demand | probability | order | sold | salvage | profit | |
---|---|---|---|---|---|---|
sunny skies | 650.0 | 0.1 | 400.0 | 400 | 0.0 | 11200 |
good weather | 400.0 | 0.6 | 400.0 | 400 | 0.0 | 11200 |
poor weather | 200.0 | 0.3 | 400.0 | 200 | 200.0 | 3600 |
Expected Profit: 8920.0
Optimizing over all scenarios provides an expected profit of 8,920 €, an increase of 581 € over the naive strategy of simply ordering the expected number of items sold. The new optimal solution places a larger order, that is \(x=400\). In poor weather conditions, there will be more returns and lower profit that is more than compensated by the increased profits in good weather conditions.
The additional value that results from solve of this planning problem is called the Value of the Stochastic Solution (VSS). The value of the stochastic solution is the additional profit compared to ordering to meet the expected demand. In this case,
Expected value with perfect information (EVPI)#
Maximizing expected profit requires the size of the order be decided before knowing what scenario will unfold. The decision for \(x\) has to be made “here and now” with probablistic information about the future, but without specific information on which future will actually transpire.
Nevertheless, we can perform the hypothetical calculation of what profit would be realized if we could know the future. We are still subject to the variability of weather, what is different is we know what the weather will be at the time the order is placed.
The resulting value for the expected profit is called the Expected Value of Perfect Information (EVPI). The difference EVPI - EV is the extra profit due to having perfect knowledge of the future.
To compute the expected profit with perfect information, we let the order variable \(x\) be indexed by the subsequent scenario that will unfold. Given decision varaible \(x_s\), the model for EVPI becomes
The following implementation is a variation of the prior cell.
%%writefile pop_evpi.mod
param r;
param c;
param w;
# set of scenarios
set S;
param p{S};
param d{S};
# decision variables
var x{S} >= 0;
var y{S} >= 0;
var f{S};
# objective
maximize EV: sum{s in S} p[s] * f[s];
# constraints
s.t. profit {s in S}: f[s] == r * y[s] + w * (x[s] - y[s]) - c * x[s];
s.t. sales_less_than_order {s in S}: y[s] <= x[s];
s.t. sales_less_than_demand {s in S}: y[s] <= d[s];
Overwriting pop_evpi.mod
# Create AMPL instance and load the model
ampl = AMPL()
ampl.read("pop_evpi.mod")
# load the data
ampl.param["r"] = r
ampl.param["c"] = c
ampl.param["w"] = w
ampl.set_data(scenarios_df, "S")
# solve the problem
ampl.option["solver"] = SOLVER
ampl.solve()
print("Solver Termination Condition:", ampl.get_value("solve_result"))
print()
# display solution using Pandas
df = pd.DataFrame.from_dict(scenarios).T
df["order"] = ampl.var["x"].get_values().toPandas()
df["sold"] = ampl.var["y"].get_values().toPandas()
df["salvage"] = df["order"] - df["sold"]
df["salvage"] = df["salvage"].round(2)
df["profit"] = ampl.var["f"].get_values().toPandas()
display(df)
print("Expected Profit:", ampl.obj["EV"].value())
cbc 2.10.7: cbc 2.10.7: optimal solution; objective 10220
0 simplex iterations
Solver Termination Condition: solved
demand | probability | order | sold | salvage | profit | |
---|---|---|---|---|---|---|
sunny skies | 650.0 | 0.1 | 650.0 | 650 | 0.0 | 18200 |
good weather | 400.0 | 0.6 | 400.0 | 400 | 0.0 | 11200 |
poor weather | 200.0 | 0.3 | 200.0 | 200 | -0.0 | 5600 |
Expected Profit: 10220.0
Summary#
To summarize, have computed three different solutions to the problem of order size:
The expected value of the mean solution (EVM) is the expected profit resulting from ordering the number of items expected to sold under all scenarios.
The expected value of the stochastic solution (EVSS) is the expected profit found by solving an two-state optimization problem where the order size was the “here and now” decision without specific knowledge of which future scenario would transpire.
The expected value of perfect information (EVPI) is the result of a hypotherical case where knowledge of the future scenario was somehow available when then order had to be placed.
For this example we found
Solution |
Value (€) |
---|---|
Expected Value of the Mean Solution (EVM) |
8,399.0 |
Expected Value of the Stochastic Solution (EVSS) |
8,920.0 |
Expected Value of Perfect Information (EVPI) |
10,220.0 |
These results verify our expectation that
The value of the stochastic solution
The value of perfect information
As one might expect, there is a cost that results from lack of knowledge about an uncertain future.