Binomial Model for Pricing Options
Contents
Binomial Model for Pricing Options#
This notebooks demonstrates techniques for pricing options using a binomial lattice to model prices of the underlying security or commodity. The notebook makes use of the pandas_datareader library to download pricing information, and the Pyomo modeling library for some example calculations.
!pip install pandas_datareader --upgrade
Requirement already satisfied: pandas_datareader in /Users/jeff/opt/anaconda3/lib/python3.9/site-packages (0.10.0)
Requirement already satisfied: lxml in /Users/jeff/opt/anaconda3/lib/python3.9/site-packages (from pandas_datareader) (4.8.0)
Requirement already satisfied: pandas>=0.23 in /Users/jeff/opt/anaconda3/lib/python3.9/site-packages (from pandas_datareader) (1.4.1)
Requirement already satisfied: requests>=2.19.0 in /Users/jeff/opt/anaconda3/lib/python3.9/site-packages (from pandas_datareader) (2.27.1)
Requirement already satisfied: python-dateutil>=2.8.1 in /Users/jeff/opt/anaconda3/lib/python3.9/site-packages (from pandas>=0.23->pandas_datareader) (2.8.2)
Requirement already satisfied: pytz>=2020.1 in /Users/jeff/opt/anaconda3/lib/python3.9/site-packages (from pandas>=0.23->pandas_datareader) (2021.3)
Requirement already satisfied: numpy>=1.18.5 in /Users/jeff/opt/anaconda3/lib/python3.9/site-packages (from pandas>=0.23->pandas_datareader) (1.21.2)
Requirement already satisfied: six>=1.5 in /Users/jeff/opt/anaconda3/lib/python3.9/site-packages (from python-dateutil>=2.8.1->pandas>=0.23->pandas_datareader) (1.16.0)
Requirement already satisfied: charset-normalizer~=2.0.0 in /Users/jeff/opt/anaconda3/lib/python3.9/site-packages (from requests>=2.19.0->pandas_datareader) (2.0.4)
Requirement already satisfied: idna<4,>=2.5 in /Users/jeff/opt/anaconda3/lib/python3.9/site-packages (from requests>=2.19.0->pandas_datareader) (3.3)
Requirement already satisfied: certifi>=2017.4.17 in /Users/jeff/opt/anaconda3/lib/python3.9/site-packages (from requests>=2.19.0->pandas_datareader) (2022.9.24)
Requirement already satisfied: urllib3<1.27,>=1.21.1 in /Users/jeff/opt/anaconda3/lib/python3.9/site-packages (from requests>=2.19.0->pandas_datareader) (1.26.8)
Historical Data#
The first step is download historical data for a selected security or commodity. For the purposes of this notebook, it is useful to choose security of commodities for which there is an active options trading so the pricing model can be compared to real data. A set of symbols representing commodity prices available on Yahoo Finance is located here.
Programming note: Given a stock market symbol, the function yahoo_finance
defined in the following cell creates a pandas data series object S
of historical prices.
import matplotlib.pyplot as plt
import numpy as np
import pandas as pd
import datetime
import pandas_datareader as pdr
import requests
def yahoo_finance(symbol, nYears = 3):
# end date is today
end = datetime.datetime.today().date()
start = end - datetime.timedelta(nYears*365)
# get stock price data. Give it three trys
for nTry in range(0,3):
try:
S = pdr.data.DataReader(symbol,"yahoo", start, end)['Adj Close']
break
except:
pass
S.name = symbol
# plot data
plt.figure(figsize=(10, 4))
S.plot(title=S.name, ylabel='Adjusted Close', grid=True)
return S
S = yahoo_finance('HOH23.NYM', 0.5)
print(S)
Date
2022-05-19 3.1092
2022-05-20 3.1091
2022-05-23 3.1310
2022-05-24 3.1473
2022-05-25 3.1879
...
2022-11-11 3.2738
2022-11-14 3.2387
2022-11-15 3.3072
2022-11-16 3.2787
2022-11-17 3.2283
Name: HOH23.NYM, Length: 128, dtype: float64
Fitting Historical Data to Geometric Brownian Motion#
A model for Geometric Brownian Motion can be written in discrete time as
where \(Z_t\) is a normal random variate with zero mean and a standard deviation of one. This could also be written as
where \(\nu = \mu -\frac{\sigma^2}{2}\). These are two different models for price. The difference between \(\mu\) and \(\nu\), i.e. \(\frac{\sigma^2}{2}\), is referred to as ‘volatility drag.’
Rearranging each of these models
\begin{align*} r^{lin} & = \frac{S_{t+\Delta t} - S_t}{S_t} = \mu\Delta t + \sigma \sqrt{\Delta t}Z_t \sim \cal{N}(\mu\Delta t, \sigma^2\Delta t)\ r^{log} & = \ln S_{t+\Delta t} - \ln S_t = \nu\Delta t + \sigma \sqrt{\Delta t}Z_t \sim \cal{N}(\nu\Delta t,\sigma^2\Delta t)\ \end{align*}
where \(\cal{N}(\alpha,\beta)\) denotes the normal distribution with mean \(\alpha\) and variance \(\beta\) (i.e., standard deviation \(\sqrt{\beta}\)).
The parameters can be estimated from trading data as
\begin{align*} \hat{\mu} & = \frac{1}{\Delta t} \mbox{mean}(\frac{S_{t+\Delta t} - S_t}{S_t}) \ \hat{\nu} & = \frac{1}{\Delta t} \mbox{mean}(\ln S_{t+\Delta t} - \ln S_t) \ \hat{\sigma} & = \frac{1}{\sqrt{\Delta t}} \mbox{stdev}(\frac{S_{t+\Delta t} - S_t}{S_t}) \approx \frac{1}{\sqrt{\Delta t}} \mbox{stdev}(\ln S_{t+\Delta t} - \ln S_t) \end{align*}
Note that for trading data time is measured in ‘trading days’. On average there are 252 trading days in a year.
Period |
Trading Days |
---|---|
Year |
252 |
Quarter |
63 |
Month |
21 |
Weekly |
4.83 |
Calendar Day |
0.690 |
Using these data and the above formulae, \(\mu\), \(\nu\), and \(\sigma\) can be rescaled to other units of time. For example, if these are estimated from daily trading data, then on an annualized basis
\begin{align*} \mu^{annual} & = 252 \times \mu^{trading\ day} \ \nu^{annual} & = 252 \times \nu^{trading\ day} \ \sigma^{annual} & = \sqrt{252} \times \sigma^{trading\ day} \ \end{align*}
from scipy.stats import norm
# compute linear and log returns
rlin = (S - S.shift(1))/S.shift(1)
rlog = np.log(S) - np.log(S.shift(1))
rlin = rlin.dropna()
rlog = rlog.dropna()
# plot data
plt.figure(figsize=(10,5))
plt.subplot(2,1,1)
rlin.plot()
plt.title('Linear Returns (daily)')
plt.grid()
plt.tight_layout()
plt.subplot(2,1,2)
rlog.plot()
plt.title('Log Returns (daily)')
plt.grid()
plt.tight_layout()
print('\nLinear Returns')
mu,sigma = norm.fit(rlin)
print(' mu = {0:12.8f} (annualized = {1:.2f}%)'.format(mu,100*252*mu))
print('sigma = {0:12.8f} (annualized = {1:.2f}%)'.format(sigma,100*np.sqrt(252)*sigma))
print('\nLog Returns')
nu,sigma = norm.fit(rlog)
print(' nu = {0:12.8f} (annualized = {1:.2f}%)'.format(nu,100*252*nu))
print('sigma = {0:12.8f} (annualized = {1:.2f}%)'.format(sigma,100*np.sqrt(252)*sigma))
Linear Returns
mu = 0.00051426 (annualized = 12.96%)
sigma = 0.02087600 (annualized = 33.14%)
Log Returns
nu = 0.00029599 (annualized = 7.46%)
sigma = 0.02089946 (annualized = 33.18%)
print(mu, sigma)
print(S[-1])
fig, ax = plt.subplots(1, 1)
slast = []
for n in range(10000):
r = nu*dt + sigma*np.sqrt(dt)*np.random.normal(size=84)
s = np.exp(np.log(S[-1] + np.cumsum(r)))
ax.plot(s, alpha=0.02, color='b')
slast.append(s[-1])
0.0005142609680219921 0.020899461334536643
3.228300094604492
plt.hist(slast, bins=100)
print(np.mean(slast), np.median(slast))
3.75837312618468 3.7584105750341843
Binomial Model#
The binomial model provides a means of modeling the statistical distribution of future prices. Given a current price \(S_t\), there are two possible states for the next observed value \(S_{t+\Delta t}\)
where \(u\), \(d\), and \(p\) are chosen to match the statistics of a model based on Geometric Brownian Motion. The following parameter values are derived in Luenberger (2013),
\begin{align*} p & = \frac{1}{2} + \frac{\nu\Delta t}{2\sqrt{\sigma^2\Delta t + (\nu\Delta t)^2}} \ \ln u & = \sqrt{\sigma^2\Delta t + (\nu\Delta t)^2} \ \ln d & = - \sqrt{\sigma^2\Delta t + (\nu\Delta t)^2} \end{align*}
# Time/period
dt = 21
p = 0.5 + nu*dt/2/np.sqrt(dt*sigma**2 + (nu*dt)**2)
u = np.exp(np.sqrt(dt*sigma**2 + (nu*dt)**2))
d = np.exp(-np.sqrt(dt*sigma**2 + (nu*dt)**2))
print('Probability (p) = ', round(p,4))
print(' Up Return (u) = ', round(u,4))
print('Down Return (d) = ', round(d,4))
Probability (p) = 0.5324
Up Return (u) = 1.1007
Down Return (d) = 0.9085
The model extends to multiple time steps in a natural way as shown in this diagram:
Note that each step forward in time introduces an additional state to the set of possible outcomes.
For the purpose of coding, we will use Python dictionaries to store future prices \(S^f\). Future prices are indexed by two subscripts, \(k\) and \(s\), such that Sf[k,s]
corresponds to the price at time \(t + k\Delta t\) in state \(s\).
We start by setting the initial node equal to the last observed price, \(S^f_{0,0} = S_t\). For each \(k\) and \(s\) there are two subsequent nodes
\begin{align*} S^f_{k + 1, s} & = u S^f_{k,s} \ S^f_{k + 1, s + 1} & = d S^f_{k,s} \end{align*}
These two equations can be combined by eliminating the common term \(S^f_{k,s}\) to yield
\begin{align*} S^f_{k + 1, s} & = u S^f_{k,s} \ S^f_{k+1,s+1} & = \frac{d}{u} S^f_{k+1,s} \end{align*}
These formula can be solved explicit to give
which is the formula used below to compute values in the binomial lattice.
N = 5
# initialize Sf
Sf = {}
Sf[0,0] = S[-1]
# compute values
for k in range(1,N+1):
for s in range(0,k+1):
Sf[k,s] = u**(k-s)*d**s*Sf[0,0]
%matplotlib inline
def Sdisplay(Sf):
plt.figure(figsize=(10,6))
for k,s in Sf.keys():
plt.plot(k,Sf[k,s],'.',ms=30,color='b')
plt.text(k,Sf[k,s],' {0:.2f}'.format(Sf[k,s]),ha='left',va='center')
if (k > 0) & (s < k):
plt.plot([k-1,k],[Sf[k-1,s],Sf[k,s]],'b')
plt.plot([k-1,k],[Sf[k-1,s],Sf[k,s+1]],'b')
plt.xlabel('k')
plt.ylabel('Value')
plt.grid()
Sdisplay(Sf)
The probability of reaching state \(s\) at time step \(k\) is denoted by \(P_{k,s}\). This can be computed given probability of preceding states and the conditional probabilities \(p\) and \(1-p\).
\begin{align*} P_{k,s} & = p P_{k-1,s} + (1-p) P_{k-1,s-1} \end{align*}
The following cell evaluates price and probability for a binomial model, and also plot the average price.
P = {}
P[0,0] = 1
for k in range(0,N):
P[k+1,0] = p*P[k,0]
P[k+1,k+1] = (1-p)*P[k,k]
for s in range(1,k+1):
P[k+1,s] = p*P[k,s] + (1-p)*P[k,s-1]
%matplotlib inline
def SPdisplay(Sf,P,D):
plt.figure(figsize=(10,6))
nPeriods = max([k for k,s in Sf.keys()]) + 1
Sfmean = np.zeros(N+1)
Sfvar = np.zeros(N+1)
for k,s in Sf.keys():
Sfmean[k] += Sf[k,s]*P[k,s]
Sfvar[k] += Sf[k,s]**2*P[k,s]
plt.plot(k,Sf[k,s],'.',ms=30,color='b',alpha=np.sqrt(P[k,s]))
if (k > 0) & (s < k):
plt.plot([k-1,k],[Sf[k-1,s],Sf[k,s]],'b',alpha=np.sqrt(P[k-1,s]))
plt.plot([k-1,k],[Sf[k-1,s],Sf[k,s+1]],'b',alpha=np.sqrt(P[k-1,s]))
for k,s in D.keys():
plt.text(k,Sf[k,s],' {0:.2f}'.format(D[k,s]),ha='left',va='center')
plt.plot(range(0,N+1),Sfmean,'r--',lw=3)
Sfstdev = np.sqrt(Sfvar - Sfmean**2)
plt.plot(range(0,N+1),Sfmean + 1.96*Sfstdev,'g--',lw=3)
plt.plot(range(0,N+1),Sfmean - 1.96*Sfstdev,'g--',lw=3)
plt.xlabel('k')
plt.ylabel('Value')
plt.grid()
plt.title('Price Projection with 95% confidence interval')
SPdisplay(Sf,P,P)
European Call Option#
A European Call Option is a contract the provides the holder with the right, but not the obligation, to purchase an asset at a specified price and date in the future. The specified price is generally called the strike price, and the specified date is the expiration date.
The purpose of the call option is to reduce the holder’s exposure to the risk of increasing prices. An airline, for example, might choose to purchase call options on airplance fuels in order to reduce the risk of selling advance tickets.
The value of the call option upon expiration depends on the price of underlying asset. If the asset spot price \(S\) is greater than the strike price \(K\), then the call option is worth the difference \(S-K\) because that is amount needed to fulfill the contract in the spot market.
On the other hand, if the asset price falls below the strike price, then the option contract has no value since the holder could buy the asset on the spot market for less than the strike price.
The next cell demonstrates the terminal value of a european call option where the strike price is equal to the initial price (known as an ‘at the money’ strike).
K = 36
C = {}
for s in range(0,N+1):
C[N,s] = max(0,Sf[N,s] - K)
SPdisplay(Sf,P,C)
plt.plot([0,N],[K,K],'y--',lw=3)
[<matplotlib.lines.Line2D at 0x7f8682c4b1f0>]
To price a call option, consider the construction of a portfolio composed of the underlying asset and cash that would have the same payoff. At node \(k,s\), the portfolio is given by
where \(x_{k,s}\) is the number of units of the underlying asset, and \(y_{k,s}\) is the cash component of the portfolio. The subsequent value of the portfolio at \(k+1\) has two possible values
\begin{align*} C_{k+1,s} & = x_{k,s}uS^f_{k,s} + (1+r)y_{k,s} \ C_{k+1,s+1} & = x_{k,s}dS^f_{k,s} + (1+r)y_{k,s} \end{align*}
where \(r\) is the per period interest rate for cash. Solving for \(x_{k,s}\) and \(y_{k,s}\),
\begin{align*} x_{k,s} & = \frac{C_{k+1,s} - C_{k+1,s+1}}{(u-d)S^f_{k,s}} \ y_{k,s} & = \frac{uC_{k+1,s+1}-dC_{k+1,s}}{(1+r)(u-d)} \end{align*}
Inserting these solutions into the original expression,
This can be expressed in a far more suggestive form
or
This expression has two important consequences. The first is an interpretation as a ‘risk-neutral’ probability such that \(C_{k,s}\) is the ‘expected value’ of the call option. This is not the real-world probability! Instead, it is different measure that provides a very useful means of computing the value of options as illustrated in the following diagram.
The second important consequence is the actual pricing formula. It’s linear, for one thing, is easily implemented as a calculation that proceeds backward in time as shown in the diagram above. As can be seen from the derivation, the computation is quite general and can be applied to any option which can be assigned a terminal value.
r = 0.030/12
q = (1+r-d)/(u-d)
print('q = ',q)
C = {}
x = {}
y = {}
for s in range(0,N+1):
C[N,s] = max(0,Sf[N,s] - K)
for k in reversed(range(0,N)):
for s in range(0,k+1):
C[k,s] = (q*C[k+1,s]+(1-q)*C[k+1,s+1])/(1+r)
x[k,s] = (C[k+1,s]-C[k+1,s+1])/(u-d)/Sf[k,s]
y[k,s] = C[k,s] - x[k,s]*Sf[k,s]
SPdisplay(Sf,P,C)
plt.plot([0,N],[K,K],'y--',lw=3)
plt.title('Price of a Call Option')
SPdisplay(Sf,P,x)
plt.plot([0,N],[K,K],'y--',lw=3)
plt.title('Hedge Ratio')
SPdisplay(Sf,P,y)
plt.plot([0,N],[K,K],'y--',lw=3)
plt.title('Cash Position')
q = 0.4842981134668199
Text(0.5, 1.0, 'Cash Position')
Implementing a Replicating Portfolio with Pyomo#
A replicating portfolio is the key concept that allows use of the binomial model for pricing options. In a nutshell, the value of an option is the money needed to construct a portfolio exactly replicate the option payoff.
This concept can be extended to applications involving ‘real assets’, including processes that convert commodity resources into higher value products. As a first step, let’s see how Pyomo can be used to model a replicating portfolio for a European call option.
We start with an expression for the value of the replicating portfolio at time \(k\) in state \(s\). For the purpose of later generalization, we use the symbol \(W\) to denote wealth,
The portfolio consists of \(x_{k,s}\) units of an underlying asset with a price \(S^f_{k,s}\) subject to statistical variability, and \(y_{k,s}\) units of a ‘bond’ which has a known future price and therefore depends only on \(k\). For example,
where \(r_f\) is a risk-free interest rate. This is the ‘cash’ component of the portfolio.
For a European call option we must have enough wealth on-hand to pay off the value of the call option at \(k=N\). This requirement can be written as inequalities
where \({\cal S}_N\) is the set of possible states at time step \(N\).
At earlier nodes, the value of the portfolio must be sufficient to finance the portfolio as subsequent states. In the binomial model, for each node \((k,s)\) there are two subsequenct states \((k+1,s)\) and \((k+1,s+1)\). This results in two constraints:
\begin{align*} x_{k,s}S^f_{k+1,s} + y_{k,s}B_{k+1} & \geq W_{k+1,s} \ x_{k,s}S^f_{k+1,s+1} + y_{k,s}B_{k+1} & \geq W_{k+1,s+1} \end{align*}
What we seek is the minimum cost portfolio at the initial node, i.e.,
subject to the above constraints.
If everything works as expected, the results of this calculation should be the same as those computed using risk-neutral probabilities.
from pyomo.environ import *
# set of periods and states for each period
Periods = range(0,N+1)
States = range(0,N+1)
# future bond prices
B = [(1+r)**k for k in Periods]
m = ConcreteModel()
# model variables
m.W = Var(Periods,States,domain=Reals)
m.x = Var(Periods,States,domain=Reals)
m.y = Var(Periods,States,domain=Reals)
# objective
m.OBJ = Objective(expr = m.W[0,0], sense=minimize)
# constraint list
m.cons = ConstraintList()
for k in Periods:
for s in range(0,k+1):
m.cons.add(m.W[k,s] == m.x[k,s]*Sf[k,s] + m.y[k,s]*(1+r)**k)
for s in States:
m.cons.add(m.W[N,s] >= max(0,Sf[N,s] - K))
for k in range(0,N):
for s in range(0,k+1):
m.cons.add(m.x[k,s]*Sf[k+1,s] + m.y[k,s]*B[k+1] >= m.W[k+1,s])
m.cons.add(m.x[k,s]*Sf[k+1,s+1] + m.y[k,s]*B[k+1] >= m.W[k+1,s+1])
SolverFactory('glpk').solve(m)
W = {}
for k in range(0,N+1):
for s in range(0,k+1):
W[k,s] = m.W[k,s]()
SPdisplay(Sf,P,W)
plt.title('Value of a European Call Option')
Text(0.5, 1.0, 'Value of a European Call Option')
European Put Option#
A European put option is the right (without obligation) to sell an asset for specified price on a specified date. The analysis is proceeds exactly as for the European call option, except that the value on the terminal date is given by
The value of the put option can be computed using the same risk neutral probabilities as for the call option, or by modifying the Pyomo model as shown in the following cell.
from pyomo.environ import *
# set of periods and states for each period
Periods = range(0,N+1)
States = range(0,N+1)
# future bond prices
B = [(1+r)**k for k in Periods]
m = ConcreteModel()
# model variables
m.W = Var(Periods,States,domain=Reals)
m.x = Var(Periods,States,domain=Reals)
m.y = Var(Periods,States,domain=Reals)
# objective
m.OBJ = Objective(expr = m.W[0,0], sense=minimize)
# constraint list
m.cons = ConstraintList()
for k in Periods:
for s in range(0,k+1):
m.cons.add(m.W[k,s] == m.x[k,s]*Sf[k,s] + m.y[k,s]*(1+r)**k)
for s in States:
m.cons.add(m.W[N,s] >= max(0,K - Sf[N,s]))
for k in range(0,N):
for s in range(0,k+1):
m.cons.add(m.x[k,s]*Sf[k+1,s] + m.y[k,s]*B[k+1] >= m.W[k+1,s])
m.cons.add(m.x[k,s]*Sf[k+1,s+1] + m.y[k,s]*B[k+1] >= m.W[k+1,s+1])
SolverFactory('glpk').solve(m)
W = {}
for k in range(0,N+1):
for s in range(0,k+1):
W[k,s] = m.W[k,s]()
SPdisplay(Sf,P,W)
plt.title('Value of European Put Option')
Text(0.5, 1.0, 'Value of European Put Option')
Early Exercise#
Let’s compare the value of the put option to the value of exercising the option. At any node \((k,s)\), the value of exercising the option is \(K-S^f_{k,s}\). The next cell compares the value of the European put option to hypothetical value of early exercise.
W = {}
for k in range(0,N+1):
for s in range(0,k+1):
W[k,s] = m.W[k,s]()
SPdisplay(Sf,P,W)
plt.title('Value of European Put Option')
E = {}
for k in range(0,N+1):
for s in range(0,k+1):
E[k,s] = K - Sf[k,s]
SPdisplay(Sf,P,E)
plt.title('The hypothetical value of early exercise of European Put Option')
Text(0.5, 1.0, 'The hypothetical value of early exercise of European Put Option')
We see several nodes where the value of exercising the option is worth more than the option itself. These would be opportunities for a risk-free profit. If such a circumstance arises, then
buy the put option at price \(P_{k,s}\)
buy the asset at price \(S^f_{k,s}\)
exercise the option to sell at the strike price \(K\)
If \(K > P_{k,s} + S^f_{k,s}\) then this deal offers a completely risk-free profit.
Arbitrage = {}
for k in range(0,N+1):
for s in range(0,k+1):
Arbitrage[k,s] = max(0,(K - Sf[k,s] - m.W[k,s]()))
SPdisplay(Sf,P,Arbitrage)
plt.title('Arbitrage value for early exercise of a European Put Option')
Text(0.5, 1.0, 'Arbitrage value for early exercise of a European Put Option')
American Put Option#
Compared to a European option, an American option provides an additional right of early exercise. The holder of the option an choose to exercise the option at any point prior to the expiration date. This additional right can (but does not always add value under certain conditions.
The opportunity for early exerise adds some complexity to the calculation of value. In the optimization framework, the holder of an American put could always receive value by exercising the option if the market value of the option is less than the exercise value. Thus the exercise value establishes a lower bound on the value of the option at all nodes (not just the terminal nodes)
This is demonstrated in the following cell.
from pyomo.environ import *
# set of periods and states for each period
Periods = range(0,N+1)
States = range(0,N+1)
# future bond prices
B = [(1+r)**k for k in Periods]
m = ConcreteModel()
# model variables
m.W = Var(Periods,States,domain=Reals)
m.x = Var(Periods,States,domain=Reals)
m.y = Var(Periods,States,domain=Reals)
# objective
m.OBJ = Objective(expr = m.W[0,0], sense=minimize)
# constraint list
m.cons = ConstraintList()
for k in Periods:
for s in range(0,k+1):
m.cons.add(m.W[k,s] == m.x[k,s]*Sf[k,s] + m.y[k,s]*(1+r)**k)
for k in Periods:
for s in range(0,k+1):
m.cons.add(m.W[k,s] >= max(0,K - Sf[k,s]))
for k in range(0,N):
for s in range(0,k+1):
m.cons.add(m.x[k,s]*Sf[k+1,s] + m.y[k,s]*B[k+1] >= m.W[k+1,s])
m.cons.add(m.x[k,s]*Sf[k+1,s+1] + m.y[k,s]*B[k+1] >= m.W[k+1,s+1])
SolverFactory('glpk').solve(m)
W = {}
E = {}
for k in range(0,N+1):
for s in range(0,k+1):
W[k,s] = m.W[k,s]()
E[k,s] = np.abs(K-Sf[k,s] - m.W[k,s]()) < 0.01
SPdisplay(Sf,P,W)
SPdisplay(Sf,P,E)