Skip to content

ArturSepp/GoalBasedAllocation

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

4 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

GoalBasedAllocation

Dynamic Mean-Variance Portfolio Allocation under Regime-Switching Jump-Diffusions

License: MIT PyPI Python Downloads

Companion code to:

Sepp, A. (2026). Dynamic Mean-Variance Portfolio Allocation under Regime-Switching Jump-Diffusions with Absorbing Barriers. SSRN: https://papers.ssrn.com/sol3/papers.cfm?abstract_id=XXXXXXX

Overview

This package provides a fully analytical Laplace-transform framework for dynamic mean-variance (MV) portfolio allocation under a two-state regime-switching model with exponential jumps at regime transitions and an absorbing wealth floor.

The MV-optimal strategy takes the form

$$\omega^_(t) = |\omega^__a| \cdot \left(\frac{\Pi^*(t)}{\Pi_t} - 1\right)$$

where $\Pi^(t)$ is the target wealth trajectory derived from the Riccati ODE system, and $|\omega^_a|$ is the regime-dependent allocation intensity. This produces an endogenous de-risking glide path: early in the horizon the funding gap is large and allocation is aggressive; as the portfolio approaches the target, allocation moderates automatically.

The terminal wealth density has three analytically tractable components:

  1. Survived density — wealth above the floor at horizon, computed via Laplace inversion of the bounded transition density
  2. Floor atom — probability mass from diffusion paths hitting the absorbing barrier
  3. Jump-overshoot density — wealth below the floor from crash jumps that gap through the barrier, computed via the overshoot distribution

All three components are computed semi-analytically using the Abate-Whitt (1995) Euler acceleration method for numerical Laplace inversion. No Monte Carlo simulation is needed for pricing; MC is used only for validation.

Key features

  • Analytical survival probability, conditional moments, and tilted survival via Laplace transforms
  • Riccati ODE system for the MV-optimal policy with regime-dependent coefficients
  • Full terminal wealth density decomposition (survived + floor atom + overshoot)
  • Exact buy-and-hold moments under regime-switching via 2×2 matrix exponential
  • Portfolio mandate construction with deterministic quadrature for jump aggregation
  • Expected allocation glide paths with variance bands
  • Investment opportunity set construction for client-facing portfolio advice
  • Monte Carlo simulator for validation of all analytical results
  • Integration tests and all paper figures reproducible from a single command

Model

The portfolio wealth $\Pi_t$ follows a regime-switching jump-diffusion with two states: growth ($i=1$) and stress ($i=2$).

Notation (paper → code)

Symbol Name Code variable
$\bar{\mu}^{[i]}$ Total expected return (CMA observable) mu_growth, mu_stress
$\mu^{[i]} = \bar{\mu}^{[i]} - \lambda \alpha^{[i]}$ Diffusion drift mu_bar
$r_h = \max(r, c)$ Hurdle rate r_h
$r_c = r_h - c = \max(0, r-c)$ Net floor growth rate r_c
$\omega^*_a = -\tilde{a}^{[i]}/\Sigma^{[i]}$ Risky allocation coefficient w_a

Process specification

Component Description
Diffusion Regime-dependent drift $\mu^{[i]}$ and volatility $\sigma^{[i]}$
Jumps Exponential crash at growth→stress ($\eta^{[1]}$), exponential recovery at stress→growth ($\eta^{[2]}$)
Regime switching Poisson rates $\lambda^{[12]}$ (crash) and $\lambda^{[21]}$ (recovery)
MV-optimal control $\omega^(t) = |\omega_a^| \cdot (\Pi^*(t)/\Pi_t - 1)$, from Riccati system
Absorbing floor Wealth stopped at $L_t = L_0 \cdot e^{r_c t}$, converting to cash

Asset class parameters

Bonds Equity Private Equity
$\bar{\mu}^{[1]}$ / $\bar{\mu}^{[2]}$ 2.5% / 2.0% 4.5% / 0.0% 7.0% / 0.0%
$\sigma^{[1]}$ / $\sigma^{[2]}$ 6% / 9% 15% / 22.5% 20% / 30%
Crash loss / Recovery gain 8% / 5% 25% / 15% 30% / 20%

Shared: $\lambda^{[12]} = 0.1$, $\lambda^{[21]} = 1.0$, $r = 2%$, $T = 10$y, $\Pi_0 = 100$.

Mandate specification

Mandates are defined by bond weight $w$ with equity-PE split $g = 2/3$: $w_{\text{eq}} = g(1-w)$, $w_{\text{pe}} = (1-g)(1-w)$.

Mandate Weights (Bd/Eq/PE) $\sigma_g$ / $\sigma_s$ $\bar{\mu}_g$ / $\bar{\mu}_s$ $\eta^{[1]}$
Income 100/0/0 6.0 / 9.0% 2.50 / 2.00% 0.087
Conservative 65/23/12 7.7 / 11.6% 3.49 / 1.30% 0.161
Balanced 35/43/22 11.1 / 16.7% 4.34 / 0.70% 0.233
Growth 0/67/33 15.8 / 23.8% 5.33 / 0.00% 0.334

MV-optimal results ($c=0%$, $\omega^*(0)=1$, $q_{dd}=2$)

Mandate $r_{\text{impl}}$ $\Pi^*_T$ $\mathbb{E}[\Pi_T]$ Std Survival $r^{\text{BH}}_{\text{impl}}$ $\text{Std}^{\text{BH}}$
Income 2.39% 240 127 24.7 85.5% 2.46% 30.4
Conservative 2.92% 200 134 33.7 83.2% 3.33% 48.3
Balanced 3.30% 221 139 46.1 78.7% 4.10% 74.8
Growth 3.68% 254 144 61.9 73.8% 5.01% 117.6

Floor protection cost ranges from 6bp (income) to 117bp (growth). BH moments are exact under the RS-JD model via 2×2 matrix exponential (Proposition B.7).

Installation

# From PyPI
pip install goal-based-allocation

# Or directly from GitHub
pip install git+https://github.com/ArturSepp/GoalBasedAllocation.git

# Or clone and install in development mode
git clone https://github.com/ArturSepp/GoalBasedAllocation.git
cd GoalBasedAllocation
pip install -e .

Requires Python >= 3.10 with NumPy >= 1.24, SciPy >= 1.10, and Matplotlib >= 3.7.

Package Structure

GoalBasedAllocation/
├── goal_based_allocation/          # Core library
│   ├── regime_switch_paper.py      # Laplace framework: density, survival, overshoot, BH moments
│   ├── riccati_solver.py           # Riccati ODE system + MC simulator
│   ├── laplace_inversion.py        # Abate-Whitt & Stehfest numerical inversion
│   ├── client_solver.py            # Effective asset construction, portfolio eta quadrature
│   ├── mandate_utils.py            # Portfolio mandate construction from assets
│   └── opportunity_set.py          # Investment opportunity set & advisor framework
├── paper_figures/                  # Paper reproduction
│   └── generate_paper_figures.py   # All 10 figures + integration tests
├── figures/                        # Pre-generated figures
├── pyproject.toml
├── LICENSE
└── README.md

Module reference

Module Description Key functions
regime_switch_paper Core Laplace framework for the RS-JD gap process compute_density, compute_survival, compute_tilted_survival, compute_overshoot_density, bh_moments_rsjd
riccati_solver Riccati ODE for MV-optimal policy, gap process, MC validation find_ell, gap_process_asset, simulate_mv_optimal
laplace_inversion Numerical Laplace inversion algorithms laplace_invert_abate_whitt, laplace_invert_stehfest
client_solver Effective single-asset from multi-asset portfolios build_effective_asset, portfolio_sigma_unc, portfolio_eta_quadrature
mandate_utils Named mandates (Income, Conservative, Balanced, Growth) mandate_effective_asset
opportunity_set Two-step advisor framework: opportunity set + client profile AdvisorSpec, compute_opportunity_point, build_opportunity_set

Quick Start

1. Compute survival probability for a single asset

from goal_based_allocation import create_paper_assets, compute_survival

assets = create_paper_assets()
eq = assets['equity']

for T in [1, 2, 5, 10]:
    S = compute_survival(T, eq.x0, eq)
    print(f"T={T:2d}y: survival={S:.4f}, stopping={1-S:.4f}")

2. Solve the MV-optimal allocation and compute the terminal wealth density

import numpy as np
from goal_based_allocation import (
    create_paper_assets, compute_density, compute_survival,
    compute_overshoot_density
)
from goal_based_allocation.riccati_solver import find_ell, gap_process_asset

assets = create_paper_assets()
eq = assets['equity']
T = 10.0

# Solve Riccati ODE for target return of 4%
ell, ric = find_ell(eq, T, target_return=0.04, r=0.02, c=0.02)
gap = gap_process_asset(ric)

# Terminal targets
PiT = ric.derived_at_tau(0)['Pi_star'][0]   # target wealth at T
L_T = eq.pi_floor                            # floor at T (with r=c)
B_T = PiT - L_T                              # buffer

# Bounded gap density -> wealth density
x_grid = np.linspace(0.001, 4.0, 800)
d0, d1 = compute_density(T, x_grid, gap)
density_total = d0 + d1

# Survival and overshoot
S = compute_survival(T, gap.x0, gap)
d_ov = np.linspace(0.001, 8.0, 400)
f_ov = compute_overshoot_density(T, d_ov, gap)

print(f"target wealth = {PiT:.1f}, floor = {L_T:.1f}")
print(f"Survival = {S:.4f}")
print(f"Overshoot mass = {np.trapezoid(f_ov, d_ov):.4f}")
print(f"Floor atom = {1 - S - np.trapezoid(f_ov, d_ov):.4f}")

3. Compare MV-optimal vs buy-and-hold

from goal_based_allocation import (
    build_effective_asset, bh_moments_rsjd, compute_survival
)
from goal_based_allocation.riccati_solver import find_ell, gap_process_asset

# Build balanced mandate (35% bonds, 43% equity, 22% PE)
eff = build_effective_asset(w_eq=0.43, w_pe=0.22, k=3.0)
T, PI0 = 10.0, 100.0

# MV-optimal survival (analytical via Laplace)
ell, ric = find_ell(eff, T, target_return=0.04, r=0.02, c=0.0)
gap = gap_process_asset(ric)
S = compute_survival(T, gap.x0, gap)
print(f"MV survival: {S:.3f}")

# BH moments (exact via matrix exponential)
bh = bh_moments_rsjd(T, PI0, eff, c=0.0)
print(f"BH: E={bh['E']:.0f}, Std={bh['Std']:.1f}, r_impl={bh['r_impl']*100:.2f}%")

4. Build an investment opportunity set

from goal_based_allocation import AdvisorSpec, build_opportunity_set

spec = AdvisorSpec(omega_0=1.0, c=0.0, q=2/3, q_dd=2.0)
opp = build_opportunity_set(spec)

for p in opp:
    print(f"Bonds={p['w_bd']:4.0%}  r_impl={p['r_impl']:5.2%}  "
          f"Surv={p['S']:5.1%}  Median={p['q50']:6.0f}")

Reproducing Paper Results

All figures and integration tests can be reproduced with a single command:

# Run integration tests (9 assertions) + generate all 10 figures
python -m paper_figures.generate_paper_figures --test

# Generate all figures only
python -m paper_figures.generate_paper_figures

# Generate a single figure
python -m paper_figures.generate_paper_figures --figure 10

# Custom output directory
python -m paper_figures.generate_paper_figures --outdir my_figures/

Integration tests

The --test flag runs 7 tests with 9 assertions covering:

Test Description Tolerance
1 Unbounded density normalization (with and without jumps) < 1e-4
2 Barrier density vs analytical survival consistency < 1e-4
3 Survival probability monotonicity across horizons --
4 Three-asset survival comparison --
5 Riccati ODE initial conditions a(0) = [1, 1] < 1e-6
6 Gap-process survival: analytical vs MC (100K paths) < 5 pp
7 Table 1 parameter validation (3 assets) exact

Figure catalog

# Description File
1 Investment opportunity set, c=0% opportunity_set_c0.png
2 Investment opportunity set, c=2.5% opportunity_set_c25.png
3 Expected allocation glide paths, c=0% allocation_paths_c0.png
4 Expected allocation glide paths, c=2.5% allocation_paths_c25.png
5 Allocation ±1σ uncertainty bands risky_allocation_subplots_c0.png
6 Path dynamics: survived vs stopped path_dynamics_balanced.png
7 Three wealth distributions: Growth mandate floor_vs_lipton.png
8 Three wealth distributions: Balanced mandate floor_vs_lipton_balanced.png
9 MV-optimal vs buy-and-hold density overlay (4 mandates) mandate_density_overlay_c0.png
10 Mandate comparison: analytical vs MC (3 mandates) mandate_comparison.png

Selected figures

Left: MV-optimal vs buy-and-hold terminal wealth densities for four mandates. Right: Analytical density (survived + overshoot) vs MC histograms.

Left: Survived and stopped sample paths with MV-optimal allocation. Right: Investment opportunity set with CDF quantiles.

Methodology

The analytical framework proceeds in three steps:

Step 1: Riccati ODE system. The pre-commitment MV problem reduces to a system of coupled Riccati ODEs for the value function coefficients in each regime. The solution yields the optimal allocation intensity $|\omega^_a|$ and the target wealth trajectory $\Pi^(t)$.

Step 2: Gap process. Under the optimal policy, the log-cushion ratio $X_t = \ln(B_t/Z_t)$ is a regime-switching jump-diffusion with an absorbing barrier at zero. Its Laplace-domain transition density satisfies a degree-6 characteristic polynomial with three positive and three negative roots.

Step 3: Laplace inversion. The bounded transition density, survival probability, tilted survival moments, and overshoot density are all expressed as Laplace transforms and inverted numerically using the Abate-Whitt (1995) Euler acceleration algorithm.

Portfolio aggregation. Multi-asset mandates are reduced to a single effective asset using portfolio volatility with full correlation structure, and portfolio jump sizes via deterministic numerical integration (portfolio_eta_quadrature). Buy-and-hold benchmark moments are computed exactly via the 2×2 matrix exponential of Proposition B.7.

Related Packages

Package Description
OptimalPortfolios Optimal portfolio construction and backtesting
StochVolModels Stochastic volatility models for options pricing
QuantInvestStrats Quantitative investment strategies and analytics
BloombergFetch Bloomberg data API wrapper
VanillaOptionPricers Vanilla option pricing models
factorlasso Factor model estimation with LASSO

Citation

If you use this package in your research, please cite:

@article{Sepp2026GoalBased,
  author  = {Sepp, Artur},
  title   = {Dynamic Mean-Variance Portfolio Allocation under Regime-Switching
             Jump-Diffusions with Absorbing Barriers},
  year    = {2026},
  note    = {Available at SSRN: https://papers.ssrn.com/sol3/papers.cfm?abstract_id=6534579}
}

Key References

  • Sepp, A., Ossa, I., and Kastenholz, M. (2026). Robust optimization of strategic and tactical asset allocation for multi-asset portfolios. Journal of Portfolio Management, 52(4), 86-120.
  • Sepp, A., Hansen, E., and Kastenholz, M. (2026). Capital market assumptions and strategic asset allocation using multi-asset tradable factors. Under revision at the Journal of Portfolio Management.
  • Sepp, A. (2004). Analytical pricing of double-barrier options under a double-exponential jump-diffusion process. International Journal of Theoretical and Applied Finance, 7(2), 151-175.
  • Sepp, A. (2006). Extended CreditGrades model with stochastic volatility and jumps. Wilmott Magazine, September, 50-62.
  • Lipton, A. (2001). Mathematical Methods for Foreign Exchange. World Scientific.
  • Lipton, A. (2001). Assets with jumps. Risk, 14(9), 149-153.
  • Cont, R. and Tankov, P. (2009). Constant proportion portfolio insurance in the presence of jumps in asset prices. Mathematical Finance, 19(3), 379-401.
  • Abate, J. and Whitt, W. (1995). Numerical inversion of Laplace transforms of probability distributions. ORSA Journal on Computing, 7(1), 36-43.

License

MIT — see LICENSE for details.

About

Analytical Laplace-transform framework for dynamic mean-variance portfolio allocation under regime-switching jump-diffusions with absorbing wealth floors. Riccati ODE for MV-optimal policy, terminal wealth density decomposition (survived + floor atom + overshoot), exact BH moments, endogenous de-risking glide paths. No Monte Carlo needed.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

 
 
 

Contributors

Languages