net.finmath.marketdata.model.volatilities.OptionSurfaceData Maven / Gradle / Ivy
package net.finmath.marketdata.model.volatilities;
import java.time.LocalDate;
import java.util.HashMap;
import net.finmath.marketdata.model.AnalyticModel;
import net.finmath.marketdata.model.curves.DiscountCurve;
import net.finmath.marketdata.model.volatilities.VolatilitySurface.QuotingConvention;
/**
* An option quote surface with the ability to query option quotes for different strikes and maturities.
*
* The surface is constructed as a collection of smiles. The choice of this dimension is convenient in view of calibration via FFT methods.
*
* This class does not perform any interpolation of market quotes. It merely represents a container of information.
*
* The class provides also the ability to perform the conversion among different quoting conventions and hence can be used both for a calibration on prices or implied volatilities.
*
* The class currently does not cover normal volatilities. Lognormal volatilities are more common in the equity space. The extension is not problematic.
*
* @author Alessandro Gnoatto
*/
public class OptionSurfaceData {
private final String underlying;
private final LocalDate referenceDate;
private final DiscountCurve discountCurve; //\exp(-r*T) needed e.g. for application of the B&S formula
private final DiscountCurve equityForwardCurve; //S0*\exp((r-d)*T)
private final QuotingConvention convention; //either price or volatility (lognormal/normal)
private final HashMap surface;
private final double[] maturities;
/**
* This is a very restrictive constructor that assumes that for each maturity we have the same number of option quotes.
*
* @param underlying The name of the underlying of this surface.
* @param referenceDate The reference date for this market data (t=0).
* @param strikes The vector of strikes.
* @param maturities The vector of maturities.
* @param values The matrix of values per (strike, maturity)
* @param convention The quoting convention (@see net.finmath.marketdata.model.volatilities.VolatilitySurfaceInterface.QuotingConvention).
* @param discountCurve A discount curve for discounting (funding/collateral rate).
* @param equityForwardCurve A the discount curve for forwarding (repo rate (e.g. funding minus dividents).
*/
public OptionSurfaceData(String underlying, LocalDate referenceDate, double[] strikes,
double[] maturities, double[][] values, QuotingConvention convention, DiscountCurve discountCurve,
DiscountCurve equityForwardCurve) {
if(strikes.length != values.length || maturities.length != values[0].length ) {
throw new IllegalArgumentException("Inconsistent number of strikes, maturities or values");
}else {
surface = new HashMap();
for(int j = 0; j< maturities.length; j++) {
double[] valuesOfInterest = new double[strikes.length];
for(int i= 0; i< strikes.length; i++) {
valuesOfInterest[i] = values[i][j];
}
OptionSmileData jthSmile = new OptionSmileData(underlying, referenceDate, strikes, maturities[j], valuesOfInterest, convention);
surface.put(maturities[j],jthSmile);
}
this.underlying = underlying;
this.referenceDate = referenceDate;
this.discountCurve = discountCurve;
this.equityForwardCurve = equityForwardCurve;
this.convention = convention;
this.maturities = maturities;
}
}
/**
* Creates an equity option surface from an array of smiles.
*
* @param smiles The option smile data.
* @param discountCurve A discount curve for discounting (funding/collateral rate).
* @param equityForwardCurve A the discount curve for forwarding (repo rate (e.g. funding minus dividents).
*/
public OptionSurfaceData(OptionSmileData[] smiles, DiscountCurve discountCurve,
DiscountCurve equityForwardCurve) {
OptionSmileData firstSmile = smiles[0];
String myUnderlying = firstSmile.getUnderlying();
LocalDate myReferenceDate = firstSmile.getReferenceDate();
QuotingConvention myConvention = firstSmile.getSmile().get(firstSmile.getStrikes()[0]).getConvention();
HashMap mySurface = new HashMap();
double[] mats = new double[smiles.length];
for(int t = 0; t getSurface(){
return this.surface;
}
public double[] getMaturities() {
return this.maturities;
}
public double getValue(double maturity, double strike){
return getValue(maturity, strike, this.convention);
}
public double getValue(double maturity, double strike, QuotingConvention quotingConvention) {
return getValue(null,maturity,strike,quotingConvention);
}
public double getValue(AnalyticModel model, double maturity, double strike,
QuotingConvention quotingConvention) {
if(quotingConvention.equals(this.convention)) {
OptionSmileData relevantSmile = this.surface.get(maturity);
return relevantSmile.getSmile().get(strike).getValue();
} else {
if(quotingConvention == QuotingConvention.PRICE && this.convention == QuotingConvention.VOLATILITYLOGNORMAL) {
double forwardPrice = equityForwardCurve.getValue(maturity);
double discountBond = discountCurve.getValue(maturity);
OptionSmileData relevantSmile = this.surface.get(maturity);
double volatility = relevantSmile.getSmile().get(strike).getValue();
return net.finmath.functions.AnalyticFormulas.blackScholesGeneralizedOptionValue(forwardPrice, volatility, maturity, strike, discountBond);
} else if(quotingConvention == QuotingConvention.VOLATILITYLOGNORMAL && this.convention == QuotingConvention.PRICE) {
double forwardPrice = equityForwardCurve.getValue(maturity);
double discountBond = discountCurve.getValue(maturity);
OptionSmileData relevantSmile = this.surface.get(maturity);
double price = relevantSmile.getSmile().get(strike).getValue();
return net.finmath.functions.AnalyticFormulas.blackScholesOptionImpliedVolatility(forwardPrice,maturity,strike,discountBond,price);
}
return 0.0;
}
}
public OptionSmileData getSmile(double maturity) {
return surface.get(maturity);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy