-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathhull_white.py
186 lines (146 loc) · 8.94 KB
/
hull_white.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
import numpy as np
import pandas as pd
from term_structure import calculate_instantaneous_forward_rate, calculate_zero_coupon_price
def calculate_hull_white_theta(mean_reversion_rate: float, volatility: float, function_zero_coupon_price: callable, tolerance: float) -> callable:
"""
Calculates the theta value for the Hull-White model
using a numeric approximation of the instantaneous forward rate
and the spot rate.
Args:
mean_reversion_rate (float): Mean reversion rate parameter a.
volatility (float): Volatility parameter sigma.
function_zero_coupon_price (function handle): Function that calculates the price of a
zero-coupon bond as a function of time.
tolerance (float): Increment of time used in the numeric calculation of the
derivative of the instantaneous forward rate.
Returns:
theta (function): Function that returns the parameter theta of
Hull-White model at the time t implied by the calibration.
Implemented by Gregor Fabjan from Open-Source Modelling on 13/04/2024.
"""
def theta(t:float)->float:
insta_forward_term = (calculate_instantaneous_forward_rate(t+tolerance, function_zero_coupon_price, tolerance)
-calculate_instantaneous_forward_rate(t-tolerance,function_zero_coupon_price,tolerance))/(2.0*tolerance)
forward_term = mean_reversion_rate*calculate_instantaneous_forward_rate(t, function_zero_coupon_price, tolerance)
variance_term = volatility**2/(2.0*mean_reversion_rate)*(1.0-np.exp(-2.0*mean_reversion_rate*t))
return insta_forward_term + forward_term + variance_term
return theta
def calculate_hull_white_paths(num_paths: int, num_steps: int, end_time: int, function_zero_coupon_price: callable, mean_reversion_rate: float, volatility: float, tolerance: float)->dict:
"""
Simulates a series of stochastic interest rate paths using the Hull-White model.
Args:
num_paths (int): number of paths to simulate.
num_steps (int): number of time steps per path.
end_time (float): end of the modelling window (in years).
(Ex. a modelling window of 50 years means T=50).
function_zero_coupon_price (function): function that calculates the price of a
zero coupon bond issued at time 0 that matures at time t, with a
notional amount 1 and discounted using the assumed term structure.
mean_reversion_rate (float): mean reversion speed parameter a of
the Hull-White model.
sigma (float): volatility parameter sigma of the Hull-White model.
tolerance (float): size of the increment used for finite
difference approximation.
Returns:
dict: A dictionary containing arrays with time steps, interest rate paths,
and bond prices.
time (array): array of time steps.
R (array): array of interest rate paths with
shape (num_paths, num_steps+1).
M (array): array of bond prices with
shape (num_paths, num_steps+1).
Implemented by Gregor Fabjan from Open-Source Modelling on 13/04/2024.
Original inspiration: https://www.youtube.com/watch?v=BIZdwUDbnDo
"""
# Initial instantaneous forward rate at time t-> 0 (also spot rate at time 0).
# r(0) = f(0,0) = - partial derivative of log(P_mkt(0, epsilon) w.r.t epsilon)
r0 = calculate_instantaneous_forward_rate(tolerance, function_zero_coupon_price, tolerance)
# Calculation of theta = 1/a * partial derivative of f(0,t) w.r.t. t
# + f(0,t) + sigma^2/(2 a^2)* (1-exp(-2*a*t)).
theta = calculate_hull_white_theta(mean_reversion_rate, volatility, function_zero_coupon_price, tolerance)
# Generate the single source of random noise.
Z = np.random.normal(0.0, 1.0, [num_paths, num_steps])
# Vector of time moments.
time = np.linspace(0, end_time, num_steps+1)
W = np.zeros([num_paths, num_steps+1])
# Initialize array with interest rate increments
R = np.zeros([num_paths, num_steps+1])
# First interest rate equals the instantaneous forward (spot)
# rate at time 0.
R[:, 0] = r0
dt = end_time/float(num_steps) # Size of increments between two steps
for iTime in range(1, num_steps+1): # For each time increment
# Making sure the samples from the normal distribution have a mean of 0
# and variance 1
if num_paths > 1:
Z[:, iTime-1] = (Z[:, iTime-1]-np.mean(Z[:, iTime-1]))/np.std(Z[:, iTime-1])
# Apply the Euler-Maruyama discretisation scheme for the Hull-White model
# at each time increment.
W[:, iTime] = W[:, iTime-1] + np.power(dt, 0.5)*Z[:, iTime-1]
noise_term = volatility* (W[:, iTime]-W[:, iTime-1])
rate_term = (theta(time[iTime-1])-mean_reversion_rate*R[:, iTime-1])*dt
R[:, iTime] = R[:, iTime-1] + rate_term + noise_term
# Vectorized numeric integration using the Euler integration method .
M = np.exp(-0.5 * (R[:, :-1] + R[:, 1:]) * dt)
M = np.insert(M, 0, 1, axis=1).cumprod(axis=1)
I = 1/M
# Output is a dataframe with time moment, the interest rate path and the price
# of a zero coupon bond issued at time 0 that matures at the selected time
# moment with a notional value of 1.
paths = {"time":time, "R":R, "M":M, "I":I}
return paths
def hull_white_main_calculation(num_paths: int, num_steps: int, end_time: int, mean_reversion_rate: float, volatility:float, function_zero_coupon_price: callable, tolerance: float):
"""
Calculates and plots the prices of zero-coupon bonds (ZCB) calculated
using the Hull-White model`s analytical formula and the Monte Carlo simulation.
Args:
num_paths (int): number of Monte Carlo simulation paths.
num_steps (int): number of time steps per path.
end_time (int): length in years of the modelling window (Ex. 50 years means t=50).
mean_reversion_rate (float): mean reversion rate parameter a of the Hull-White model.
volatility (float): volatility parameter sigma of the Hull-White model.
function_zero_coupon_price (function): function that calculates the price of a zero coupon bond issued.
at time 0 that matures at time t, with a notional amount 1 and discounted using
the assumed term structure.
tolerance (float): the size of the increment used for finite difference approximation.
Returns:
t : time increments.
P : average of the sumulated paths.
implied_term_structure : term structure provided as input into the HW simulation.
Implemented by Gregor Fabjan from Open-Source Modelling on 13/04/2024.
"""
paths = calculate_hull_white_paths(num_paths, num_steps, end_time, function_zero_coupon_price, mean_reversion_rate, volatility, tolerance)
M = paths["M"]
t = paths["time"]
I = paths["I"]
implied_term_structure = function_zero_coupon_price(t)
# Compare the price of an option on a ZCB from Monte Carlo and the analytical expression
P = np.zeros([num_steps+1])
for i in range(0, num_steps+1):
P[i] = np.mean(M[:, i])
return [t, P, implied_term_structure, M, I]
def set_up_hull_white(asset_id: int, modeling_parameters: dict, zero_coupon_price: callable)->pd.DataFrame:
num_paths = modeling_parameters["num_paths"] # Number of stochastic scenarios
num_steps = modeling_parameters["num_steps"] # Number of equidistand discrete modelling points (50*12 = 600)
end_time = modeling_parameters["end_time"] # Time horizon in years (A time horizon of 50 years; T=50)
a = modeling_parameters["a"] # Hull-White mean reversion parameter a
sigma = modeling_parameters["sigma"] # Hull-White volatility parameter sigma
tolerance = modeling_parameters["tolerance"] # Incremental distance used to calculate for numerical approximation
# of for example the instantaneous spot rate (Ex. 0.01 will use an interval
# of 0.01 as a discreete approximation for a derivative)
type = modeling_parameters["curve_type"]
# Final comparison
[t, P, implied_term_structure, M, I] = hull_white_main_calculation(num_paths, num_steps, end_time, a, sigma, zero_coupon_price, tolerance)
if type=="I":
outTmp = I
elif type=="D":
outTmp = M
else:
raise ValueError
run_name = "HW-"+str(asset_id)
multi_index_list = []
for scenario in list(range(0,num_paths)):
multi_index_list.append((run_name,scenario))
multi_index = pd.MultiIndex.from_tuples(multi_index_list, names=('Run', 'Scenario_number'))
scenarios = pd.DataFrame(data = outTmp, columns=t, index=multi_index)
return scenarios