All Downloads are FREE. Search and download functionalities are using the official Maven repository.

net.finmath.marketdata.model.volatility.caplet.CapletVolBootstrapping Maven / Gradle / Ivy

package net.finmath.marketdata.model.volatility.caplet;

import java.time.LocalDate;

import net.finmath.exception.CalculationException;
import net.finmath.marketdata.model.AnalyticModel;
import net.finmath.marketdata.model.curves.DiscountCurve;
import net.finmath.marketdata.model.curves.ForwardCurve;
import net.finmath.marketdata.model.volatilities.VolatilitySurface;
import net.finmath.marketdata.model.volatilities.VolatilitySurface.QuotingConvention;
import net.finmath.marketdata.model.volatility.caplet.tenorconversion.CorrelationProvider;
import net.finmath.marketdata.model.volatility.caplet.tenorconversion.TenorConverter;
import net.finmath.rootfinder.BisectionSearch;
import net.finmath.time.Schedule;
import net.finmath.time.ScheduleGenerator;
import net.finmath.time.ScheduleGenerator.DaycountConvention;
import net.finmath.time.ScheduleGenerator.Frequency;
import net.finmath.time.ScheduleGenerator.ShortPeriodConvention;
import net.finmath.time.businessdaycalendar.BusinessdayCalendar;
import net.finmath.time.businessdaycalendar.BusinessdayCalendar.DateRollConvention;
import net.finmath.time.businessdaycalendar.BusinessdayCalendarExcludingTARGETHolidays;
import net.finmath.time.businessdaycalendar.BusinessdayCalendarExcludingWeekends;

/**
 * This class implements a caplet volatility bootstrapper. Given an object of type CapVolMarketData
 * which contains market volatilities for caps it calculates the caplet volatilities.
 *
 * @author Daniel Willhalm (initial version)
 * @author Christian Fries (review and fixes)
 */
public class CapletVolBootstrapping {

	private final CapVolMarketData capVolMarketData;
	private final DiscountCurve discountCurve;
	private final ForwardCurve forwardCurve;

	private double[][] capletVolMatrix;
	private double[] capletFixingTimeVectorInYears;
	private final CorrelationProvider correlationProvider;

	private transient AnalyticModel analyticModel;

	/**
	 * The constructor of the caplet bootstrapping class.
	 *
	 * @param correlationProvider The correlationProvider which is necessary only if the underlying cap data changes its tenor (common for EUR cap data).
	 * @param capVolMarketData The market data for the caps.
	 * @param parsedModel The analytic model for forward and discount curves.
	 */
	public CapletVolBootstrapping(final CorrelationProvider correlationProvider, final CapVolMarketData capVolMarketData, final AnalyticModel parsedModel) {
		super();
		this.capVolMarketData = capVolMarketData;
		this.correlationProvider = correlationProvider;
		this.analyticModel = parsedModel;
		String currency = null;
		switch (capVolMarketData.getCapTenorStructure()) {
		case EUR:
			currency = "EUR";
			break;

		case USD:
			currency = "USD";
			break;
		default:
			throw new IllegalArgumentException("Unknown currency " + capVolMarketData.getCapTenorStructure() + ".");
		}
		forwardCurve = parsedModel.getForwardCurve("Forward_" + currency + "_" + capVolMarketData.getIndex());
		discountCurve = parsedModel.getDiscountCurve(currency + "_" + capVolMarketData.getDiscountIndex());
	}

	/**
	 * Overloaded constructor of the caplet bootstrapping class if a correlation provider isn't necessary.
	 *
	 * @param capVolMarketData The market data for the caps.
	 * @param parsedModel The analytic model for forward and discount curves.
	 */
	public CapletVolBootstrapping(final CapVolMarketData capVolMarketData, final AnalyticModel parsedModel) {
		this(null, capVolMarketData, parsedModel);
	}

	/**
	 * Method that bootstraps the caplet volatilities from the cap volatility data.
	 * It is assumed that the caplet volatilities between available cap volatility data is constant.
	 * The bisection method is used as a root finder to align the cap price and the sum of caplets price.
	 *
	 * @return The bootstrapped caplet volatility matrix.
	 * @throws CalculationException Thrown if calculation fails arithmetically.
	 */
	public double[][] getCapletVolMatrix() throws CalculationException {
		//initialize the caplet volatility matrix. It has 2 * highest cap expiry - 1 entries since the first caplet has value 0
		capletVolMatrix = new double[capVolMarketData.getMaxExpiryInMonths()/capVolMarketData.getUnderlyingTenorInMonths()-1][capVolMarketData.getNumberOfStrikes()];
		capletFixingTimeVectorInYears = new double[capVolMarketData.getMaxExpiryInMonths()/capVolMarketData.getUnderlyingTenorInMonths()-1];
		for (int i = 0; i < capletFixingTimeVectorInYears.length; i++) {
			capletFixingTimeVectorInYears[i] = (i+1)*capVolMarketData.getUnderlyingTenorInMonths()/12.0;
		}

		Frequency frequency = null;
		switch (capVolMarketData.getUnderlyingTenorInMonths()) {
		case 1:
			frequency = Frequency.MONTHLY;
			break;
		case 3:
			frequency = Frequency.QUARTERLY;
			break;
		case 6:
			frequency = Frequency.SEMIANNUAL;
			break;
		case 12:
			frequency = Frequency.ANNUAL;
			break;
		default:
			throw new IllegalArgumentException("Unknown tenor " + capVolMarketData.getUnderlyingTenorInMonths() + ".");
		}

		//Conversion if underlying tenors are different; not tested yet
		//analytic model has to contain two forward curves
		if (capVolMarketData.getUnderlyingTenorInMonths() != capVolMarketData.getUnderlyingTenorInMonthsBeforeChange()) {

			int numberOfExpiriesBeforeChange = 0;
			while(capVolMarketData.getExpiryInMonths(numberOfExpiriesBeforeChange) <= capVolMarketData.getTenorChangeTimeInMonths()) {
				numberOfExpiriesBeforeChange++;
			}

			final int[] expiryVectorInMonthsBeforeChange = new int[numberOfExpiriesBeforeChange];
			for (int i = 0; i < numberOfExpiriesBeforeChange; i++) {
				expiryVectorInMonthsBeforeChange[i] = capVolMarketData.getExpiryInMonths(i);
			}
			final double[][] capVolMatrixBeforeChange = new double[numberOfExpiriesBeforeChange][capVolMarketData.getNumberOfStrikes()];
			for (int j = 0; j < capVolMarketData.getNumberOfStrikes(); j++) {
				for (int i = 0; i < numberOfExpiriesBeforeChange; i++) {
					capVolMatrixBeforeChange[i][j] = capVolMarketData.getVolMatrix()[i][j];
				}
			}
			final double[] capletFixingTimeVectorInYearsBeforeChange = new double[capVolMarketData.getMaxExpiryInMonths()/capVolMarketData.getUnderlyingTenorInMonths()-1];
			for (int i = 0; i < capletFixingTimeVectorInYearsBeforeChange.length; i++) {
				capletFixingTimeVectorInYearsBeforeChange[i] = (i+1)*capVolMarketData.getUnderlyingTenorInMonthsBeforeChange()/12.0;
			}

			final CapVolMarketData capVolMarketDataBeforeChange = new CapVolMarketData(
					capVolMarketData.getIndexBeforeChange(),
					capVolMarketData.getDiscountIndex(),
					capVolMarketData.getCapTenorStructure(),
					expiryVectorInMonthsBeforeChange, capVolMarketData.getStrikeVector(),
					capVolMatrixBeforeChange, capVolMarketData.getShift(),
					capVolMarketData.getUnderlyingTenorInMonthsBeforeChange());

			final CapletVolBootstrapping capletVolBootstrapperBeforeChange = new CapletVolBootstrapping(capVolMarketDataBeforeChange, analyticModel);

			final TenorConverter tenorConverterBeforeChange = new TenorConverter(
					correlationProvider,
					capVolMarketDataBeforeChange.getUnderlyingTenorInMonthsBeforeChange(),
					capVolMarketData.getUnderlyingTenorInMonths(),
					capletFixingTimeVectorInYearsBeforeChange,
					capVolMarketDataBeforeChange.getStrikeVector(),
					capletVolBootstrapperBeforeChange.getCapletVolMatrix(),
					capVolMarketDataBeforeChange.getCapTenorStructure(),
					analyticModel, capVolMarketData.getDiscountIndex(),
					capVolMarketDataBeforeChange.getIndex(),
					capVolMarketData.getIndex());

			final double[][] capVolMatrixBeforeChangeNewTenor = capletVolBootstrapperBeforeChange.calculateCapVolsFromCapletVols(tenorConverterBeforeChange.convertTenor());

			for (int j = 0; j < capVolMarketData.getNumberOfStrikes(); j++) {
				for (int i = 0; i < numberOfExpiriesBeforeChange; i++) {
					capVolMarketData.setCapVolMatrixEntry(i, j, capVolMatrixBeforeChangeNewTenor[i][j]);
				}
			}
		}

		//loop over the number of strikes of the market cap volatilities
		for (int j = 0; j < capVolMarketData.getNumberOfStrikes(); j++) {
			int lastExpiryInMonths = 0;
			int currentExpiryInMonths = capVolMarketData.getExpiryInMonths(0);
			int lastCaplet = 0;
			boolean isFirstCap = true;

			//loop over the number of caplets
			for (int i = 0; i < capVolMarketData.getMaxExpiryInMonths()/capVolMarketData.getUnderlyingTenorInMonths(); i++) {
				if (currentExpiryInMonths != (i+1)*capVolMarketData.getUnderlyingTenorInMonths()) {
					continue;
				}
				if (isFirstCap) {
					for (int k = 0; k < i; k++) {
						capletVolMatrix[k][j] = capVolMarketData.getCapVolData(0, j);
						//next line due to numerical errors from strike interpolation
						if (j+1 < capVolMarketData.getNumberOfStrikes()) {
							capletVolMatrix[k][j+1] = capletVolMatrix[k][j];
						}
					}
					isFirstCap = false;
				}
				else {
					//rootfinder
					//bisection method
					final int currentExpiryRow = capVolMarketData.getRowIndex(currentExpiryInMonths);
					final BusinessdayCalendar businessdayCalendar = new BusinessdayCalendarExcludingTARGETHolidays(new BusinessdayCalendarExcludingWeekends());
					final LocalDate localDate = discountCurve.getReferenceDate();
					final LocalDate startDate = businessdayCalendar.getRolledDate(localDate, 2);
					//Quoting convention k�nnte auch �bergeben werden
					VolatilitySurface capletVolatilities = new CapletVolatilitySurface("Cap volatility surface", localDate, capVolMarketData.getCapVolData(currentExpiryRow, j), capletFixingTimeVectorInYears, capVolMarketData.getStrikeVector(), forwardCurve, QuotingConvention.VOLATILITYLOGNORMAL, discountCurve);
					//Schedule k�nnte auch �bergeben werden
					final Schedule schedule = ScheduleGenerator.createScheduleFromConventions(localDate, startDate, localDate.plusMonths(currentExpiryInMonths), frequency, DaycountConvention.ACT_365, ShortPeriodConvention.FIRST, DateRollConvention.MODIFIED_FOLLOWING, businessdayCalendar, -2, 0);
					final CapShiftedVol cap = new CapShiftedVol(schedule, forwardCurve.getName(), capVolMarketData.getStrike(j), false, discountCurve.getName(), capletVolatilities.getName(), capVolMarketData.getShift());
					final double capPrice = cap.getValueAsPrice(0, analyticModel.addVolatilitySurfaces(capletVolatilities));
					double sumCapletPrices = Double.MAX_VALUE;
					double leftPoint = capVolMarketData.getCapVolData(currentExpiryRow, j)*2;
					while (capPrice - sumCapletPrices <= 0) {
						leftPoint /= 1.5;
						for (int l = i; l > lastCaplet; l--) {
							capletVolMatrix[l-1][j] = leftPoint;
							if (j+1 < capVolMarketData.getNumberOfStrikes()) {
								capletVolMatrix[l-1][j+1] = capletVolMatrix[l-1][j];
							}
						}
						capletVolatilities = new CapletVolatilitySurface("Cap volatility surface", localDate, capletVolMatrix, capletFixingTimeVectorInYears, capVolMarketData.getStrikeVector(), forwardCurve, QuotingConvention.VOLATILITYLOGNORMAL, discountCurve);
						sumCapletPrices = cap.getValueAsPrice(0, analyticModel.addVolatilitySurfaces(capletVolatilities));
						if(capPrice - sumCapletPrices == 0) {
							break;
						}
					}
					final double leftValue = capPrice - sumCapletPrices;
					double rightPoint = leftPoint;
					while (capPrice - sumCapletPrices >= 0) {
						rightPoint += 0.1;
						for (int l = i; l > lastCaplet; l--) {
							capletVolMatrix[l-1][j] = rightPoint;
							if (j+1 < capVolMarketData.getNumberOfStrikes()) {
								capletVolMatrix[l-1][j+1] = capletVolMatrix[l-1][j];
							}
						}
						capletVolatilities = new CapletVolatilitySurface("Cap volatility surface", localDate, capletVolMatrix, capletFixingTimeVectorInYears, capVolMarketData.getStrikeVector(), forwardCurve, QuotingConvention.VOLATILITYLOGNORMAL, discountCurve);
						sumCapletPrices = cap.getValueAsPrice(0, analyticModel.addVolatilitySurfaces(capletVolatilities));
						if(capPrice - sumCapletPrices == 0) {
							break;
						}
					}
					final double rightValue = capPrice - sumCapletPrices;
					final BisectionSearch bisectionSearch = new BisectionSearch(leftPoint, rightPoint);
					bisectionSearch.setValue(leftValue);
					bisectionSearch.setValue(rightValue);
					while (bisectionSearch.isDone() != true) {
						for (int l = i; l > lastCaplet; l--) {
							capletVolMatrix[l-1][j] = bisectionSearch.getNextPoint();
							if (j+1 < capVolMarketData.getNumberOfStrikes()) {
								capletVolMatrix[l-1][j+1] = capletVolMatrix[l-1][j];
							}
						}
						capletVolatilities = new CapletVolatilitySurface("Cap volatility surface", localDate, capletVolMatrix, capletFixingTimeVectorInYears, capVolMarketData.getStrikeVector(), forwardCurve, QuotingConvention.VOLATILITYLOGNORMAL, discountCurve);
						sumCapletPrices = cap.getValueAsPrice(0, analyticModel.addVolatilitySurfaces(capletVolatilities));
						bisectionSearch.setValue(capPrice - sumCapletPrices);
					}
					final double volaBestPoint = bisectionSearch.getBestPoint();

					for (int m = i; i - m < (currentExpiryInMonths-lastExpiryInMonths)/capVolMarketData.getUnderlyingTenorInMonths(); m--) {
						capletVolMatrix[m-1][j] = volaBestPoint;
					}
				}
				lastExpiryInMonths = currentExpiryInMonths;
				lastCaplet = i;
				if (capVolMarketData.getRowIndex(currentExpiryInMonths)+1 < capVolMarketData.getNumberOfExpiryDates()) {
					currentExpiryInMonths = capVolMarketData.getExpiryInMonths(capVolMarketData.getRowIndex(currentExpiryInMonths)+1);
				}
			}
		}
		return capletVolMatrix;
	}

	/**
	 * Method that implements the opposite direction. That means using a caplet volatility matrix
	 * the cap volatility matrix is calculated
	 *
	 * @param inputCapletVolMatrix The caplet volatility matrix.
	 * @return The cap volatility matrix.
	 */
	public double[][] calculateCapVolsFromCapletVols(final double[][] inputCapletVolMatrix) {
		final double[][] capVolMatrix = new double[capVolMarketData.getNumberOfExpiryDates()][capVolMarketData.getNumberOfStrikes()];
		final double[] capletFixingTimeVectorInYears = new double[capVolMarketData.getMaxExpiryInMonths()/capVolMarketData.getUnderlyingTenorInMonths()-1];
		for (int i = 0; i < capletFixingTimeVectorInYears.length; i++) {
			capletFixingTimeVectorInYears[i] = (i+1)*capVolMarketData.getUnderlyingTenorInMonths()/12.0;
		}

		Frequency frequency = null;
		switch (capVolMarketData.getUnderlyingTenorInMonths()) {
		case 1:
			frequency = Frequency.MONTHLY;
			break;
		case 3:
			frequency = Frequency.QUARTERLY;
			break;
		case 6:
			frequency = Frequency.SEMIANNUAL;
			break;
		case 12:
			frequency = Frequency.ANNUAL;
			break;
		default:
			throw new IllegalArgumentException("Unknown tenor " + capVolMarketData.getUnderlyingTenorInMonths() + ".");
		}
		//loop over the number of strikes of the market cap volatilities
		for (int j = 0; j < capVolMarketData.getNumberOfStrikes(); j++) {
			int lastExpiryInMonths = 0;
			int currentExpiryInMonths = capVolMarketData.getExpiryInMonths(0);
			boolean isFirstCaplet = true;
			//loop over the number of caps
			for (int i = 0; i < capVolMarketData.getNumberOfExpiryDates(); i++) {
				if (isFirstCaplet) {
					capVolMatrix[0][j] = inputCapletVolMatrix[0][j];
					//next line due to numerical errors from strike interpolation
					if (j+1 < capVolMarketData.getNumberOfStrikes()) {
						capVolMatrix[0][j+1] = capVolMatrix[0][j];
					}
					isFirstCaplet = false;
				}
				else {
					//rootfinder
					//bisection method
					final LocalDate localDate = discountCurve.getReferenceDate();
					VolatilitySurface capletVolatilities = new CapletVolatilitySurface("Cap volatility surface", localDate, inputCapletVolMatrix, capletFixingTimeVectorInYears, capVolMarketData.getStrikeVector(), forwardCurve, QuotingConvention.VOLATILITYLOGNORMAL, discountCurve);
					final Schedule schedule = ScheduleGenerator.createScheduleFromConventions(localDate, localDate, localDate.plusMonths(currentExpiryInMonths), frequency, DaycountConvention.ACT_365, ShortPeriodConvention.FIRST, DateRollConvention.MODIFIED_FOLLOWING, new BusinessdayCalendarExcludingTARGETHolidays(new BusinessdayCalendarExcludingWeekends()), -2, 0);
					final CapShiftedVol cap = new CapShiftedVol(schedule, forwardCurve.getName(), capVolMarketData.getStrike(j), false, discountCurve.getName(), capletVolatilities.getName(), capVolMarketData.getShift());
					final double sumCapletPrices = cap.getValueAsPrice(0, analyticModel.addVolatilitySurfaces(capletVolatilities));
					double capPrice = Double.MAX_VALUE;
					double leftPoint = inputCapletVolMatrix[lastExpiryInMonths/capVolMarketData.getUnderlyingTenorInMonths()-1][j];
					while (capPrice - sumCapletPrices >= 0) {
						leftPoint /= 1.5;
						capletVolatilities = new CapletVolatilitySurface("Cap volatility surface", localDate, leftPoint, capletFixingTimeVectorInYears, capVolMarketData.getStrikeVector(), forwardCurve, QuotingConvention.VOLATILITYLOGNORMAL, discountCurve);
						capPrice = cap.getValueAsPrice(0, analyticModel.addVolatilitySurfaces(capletVolatilities));
						if(capPrice - sumCapletPrices == 0) {
							break;
						}
					}
					final double leftValue = capPrice - sumCapletPrices;
					double rightPoint = leftPoint;
					while (capPrice - sumCapletPrices <= 0) {
						rightPoint += 0.1;
						capletVolatilities = new CapletVolatilitySurface("Cap volatility surface", localDate, rightPoint, capletFixingTimeVectorInYears, capVolMarketData.getStrikeVector(), forwardCurve, QuotingConvention.VOLATILITYLOGNORMAL, discountCurve);
						capPrice = cap.getValueAsPrice(0, analyticModel.addVolatilitySurfaces(capletVolatilities));
						if(capPrice - sumCapletPrices == 0) {
							break;
						}
					}
					final double rightValue = capPrice - sumCapletPrices;
					final BisectionSearch bisectionSearch = new BisectionSearch(leftPoint, rightPoint);
					bisectionSearch.setValue(leftValue);
					bisectionSearch.setValue(rightValue);
					while (bisectionSearch.isDone() != true) {
						capletVolatilities = new CapletVolatilitySurface("Cap volatility surface", localDate, bisectionSearch.getNextPoint(), capletFixingTimeVectorInYears, capVolMarketData.getStrikeVector(), forwardCurve, QuotingConvention.VOLATILITYLOGNORMAL, discountCurve);
						capPrice = cap.getValueAsPrice(0, analyticModel.addVolatilitySurfaces(capletVolatilities));
						bisectionSearch.setValue(capPrice - sumCapletPrices);
					}
					final double volaBestPoint = bisectionSearch.getBestPoint();
					capVolMatrix[i][j] = volaBestPoint;
				}
				lastExpiryInMonths = currentExpiryInMonths;
				if (capVolMarketData.getRowIndex(currentExpiryInMonths)+1 < capVolMarketData.getNumberOfExpiryDates()) {
					currentExpiryInMonths = capVolMarketData.getExpiryInMonths(capVolMarketData.getRowIndex(currentExpiryInMonths)+1);
				}
			}
		}
		return capVolMatrix;
	}

	public AnalyticModel getParsedModel() {
		return analyticModel;
	}

	public DiscountCurve getDiscountCurve() {
		return discountCurve;
	}

	public ForwardCurve getForwardCurve() {
		return forwardCurve;
	}

	public double[] getCapletFixingTimeVectorInYears() {
		return capletFixingTimeVectorInYears;
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy