net.finmath.singleswaprate.Utils Maven / Gradle / Ivy
package net.finmath.singleswaprate;
import java.time.LocalDate;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import net.finmath.functions.AnalyticFormulas;
import net.finmath.marketdata.model.volatilities.SwaptionDataLattice;
import net.finmath.marketdata.model.volatilities.SwaptionDataLattice.QuotingConvention;
import net.finmath.marketdata.products.Swap;
import net.finmath.singleswaprate.data.DataTable;
import net.finmath.singleswaprate.data.DataTable.TableConvention;
import net.finmath.singleswaprate.model.VolatilityCubeModel;
import net.finmath.time.Schedule;
import net.finmath.time.SchedulePrototype;
/**
* A collection of utility methods for dealing with the {@link net.finmath.singleswaprate} package.
*
* @author Christian Fries
* @author Roland Bachl
*
*/
public class Utils {
/**
* Convert a {@link DataTable} containing swaption data to a {@link SwaptionDataLattice}.
* The table needs to be in {@link TableConvention#MONTHS}.
*
* @param table The table in convention {@link TableConvention#MONTHS} containing swaption data.
* @param quotingConvention The quoting convention of the data.
* @param referenceDate The reference date associated with the swaptions.
* @param discountCurveName The name of the discount curve to be used for the swaptions.
* @param forwardCurveName The name of the forward curve to be used for the swaptions.
* @param fixMetaSchedule The ScheduleMetaData to be used for the fix schedules of the swaptions.
* @param floatMetaSchedule The ScheduleMetaData to be used for the float schedules of the swaptions.
* @return SwaptionDataLattice containing the swaptions of the table.
*/
public static SwaptionDataLattice convertTableToLattice(DataTable table, QuotingConvention quotingConvention, LocalDate referenceDate,
String discountCurveName, String forwardCurveName, SchedulePrototype fixMetaSchedule, SchedulePrototype floatMetaSchedule) {
return convertMapOfTablesToLattice(
new HashMap() {private static final long serialVersionUID = 1L; { put(0, table); }},
quotingConvention,
referenceDate,
discountCurveName,
forwardCurveName,
fixMetaSchedule,
floatMetaSchedule);
}
/**
* Convert a map of {@link DataTable} containing swaption data to a {@link SwaptionDataLattice}.
* The data of the swaptions is arranged in tables by moneyness, which is used as key in the map.
* The tables need to be in {@link TableConvention#MONTHS}.
*
* @param tables A map of tables, containing swaption data in convention {@link TableConvention#MONTHS}, per moneyness.
* @param quotingConvention The quoting convention of the data.
* @param referenceDate The reference date associated with the swaptions.
* @param discountCurveName The name of the discount curve to be used for the swaptions.
* @param forwardCurveName The name of the forward curve to be used for the swaptions.
* @param fixMetaSchedule The ScheduleMetaData to be used for the fix schedules of the swaptions.
* @param floatMetaSchedule The ScheduleMetaData to be used for the float schedules of the swaptions.
* @return SwaptionDataLattice containing the swaptions of the tables.
*/
public static SwaptionDataLattice convertMapOfTablesToLattice(Map tables, QuotingConvention quotingConvention, LocalDate referenceDate,
String discountCurveName, String forwardCurveName, SchedulePrototype fixMetaSchedule, SchedulePrototype floatMetaSchedule) {
List moneynesss = new ArrayList<>();
List maturities = new ArrayList<>();
List tenors = new ArrayList<>();
List values = new ArrayList<>();
for(int moneyness : tables.keySet()) {
DataTable table = tables.get(moneyness);
if(table.getConvention() != TableConvention.MONTHS) {
throw new IllegalArgumentException("This method is only set up to handle tables with convention 'inMONTHS'.");
}
for(int maturity : table.getMaturities()) {
for(int termination : table.getTerminationsForMaturity(maturity)) {
moneynesss.add(moneyness);
maturities.add(maturity);
tenors.add(termination);
values.add(table.getValue(maturity, termination));
}
}
}
return new SwaptionDataLattice(referenceDate, quotingConvention, 0, forwardCurveName, discountCurveName, floatMetaSchedule, fixMetaSchedule,
maturities.stream().mapToInt(Integer::intValue).toArray(),
tenors.stream().mapToInt(Integer::intValue).toArray(),
moneynesss.stream().mapToInt(Integer::intValue).toArray(),
values.stream().mapToDouble(Double::doubleValue).toArray());
}
/**
* Create smiles for physically settled swaptions by shifting the smiles from cash settled swaptions onto atm levels of physically settled swaptions.
*
* @param model Contains curves to translate swaption data to normal volatility. Can be null, if data already in normal volatility.
* @param physicalSwaptions The physically settled atm swaptions.
* @param cashSwaptions The smile points with corresponding atm nodes of cash swaptions.
* @return The lattice containing the shifted physically settled swaption smiles.
*/
public static SwaptionDataLattice shiftCashToPhysicalSmile(VolatilityCubeModel model, SwaptionDataLattice physicalSwaptions, SwaptionDataLattice... cashSwaptions) {
SwaptionDataLattice physicalLattice = physicalSwaptions.convertLattice(QuotingConvention.PAYERVOLATILITYNORMAL, model);
for(SwaptionDataLattice cashLatticeUnconverted : cashSwaptions) {
SwaptionDataLattice cashLattice = convertCashLatticeToNormalVolatility(cashLatticeUnconverted, model);
List smileMoneynesss = new ArrayList<>();
List smileMaturities = new ArrayList<>();
List smileTerminations = new ArrayList<>();
List smileVolatilities = new ArrayList<>();
for(int moneyness : cashLattice.getMoneyness()) {
if(moneyness == 0) {
continue;
}
for(int maturity : cashLattice.getMaturities(moneyness)) {
for(int termination : cashLattice.getTenors(moneyness, maturity)) {
if((!cashLattice.containsEntryFor(maturity, termination, 0)) || (!physicalLattice.containsEntryFor(maturity, termination, 0)) ) {
continue;
}
smileMoneynesss.add(moneyness);
smileMaturities.add(maturity);
smileTerminations.add(termination);
smileVolatilities.add(physicalLattice.getValue(maturity, termination, 0) +
cashLattice.getValue(maturity, termination, moneyness) - cashLattice.getValue(maturity, termination, 0));
}
}
}
SwaptionDataLattice newSwaptions = new SwaptionDataLattice(cashLattice.getReferenceDate(), QuotingConvention.PAYERVOLATILITYNORMAL, cashLattice.getForwardCurveName(),
cashLattice.getDiscountCurveName(), cashLattice.getFloatMetaSchedule(), cashLattice.getFixMetaSchedule(),
smileMaturities.stream().mapToInt(Integer::intValue).toArray(),
smileTerminations.stream().mapToInt(Integer::intValue).toArray(),
smileMoneynesss.stream().mapToInt(Integer::intValue).toArray(),
smileVolatilities.stream().mapToDouble(Double::doubleValue).toArray());
physicalLattice = physicalLattice.append(newSwaptions, model);
}
return physicalLattice;
}
/**
* Convert a lattice containing cash settled swaption prices to payer normal volatilities.
* Conversion assumes put-call-parity.
*
* @param cashLattice The lattice of cash settled swaptions.
* @param model The model containing curves for conversion.
* @return The converted lattice.
*/
public static SwaptionDataLattice convertCashLatticeToNormalVolatility(SwaptionDataLattice cashLattice,
VolatilityCubeModel model) {
SchedulePrototype fixMetaSchedule = cashLattice.getFixMetaSchedule();
SchedulePrototype floatMetaSchedule = cashLattice.getFloatMetaSchedule();
LocalDate referenceDate = cashLattice.getReferenceDate();
List maturities = new ArrayList<>();
List tenors = new ArrayList<>();
List moneynesss = new ArrayList<>();
List values = new ArrayList<>();
boolean isPayer;
if(cashLattice.getQuotingConvention() == QuotingConvention.PAYERPRICE) {
isPayer = true;
} else if(cashLattice.getQuotingConvention() == QuotingConvention.RECEIVERPRICE){
isPayer = false;
} else {
throw new IllegalArgumentException("This conversion assumes a lattice in convention PAYERPRICE or RECEIVERPRICE.");
}
for(int moneyness : cashLattice.getMoneyness()) {
for(int maturity : cashLattice.getMaturities(moneyness)) {
for(int tenor : cashLattice.getTenors(moneyness, maturity)) {
Schedule fixSchedule = fixMetaSchedule.generateSchedule(referenceDate, maturity, tenor);
Schedule floatSchedule = floatMetaSchedule.generateSchedule(referenceDate, maturity, tenor);
double parSwapRate = Swap.getForwardSwapRate(fixSchedule, floatSchedule, model.getForwardCurve(cashLattice.getForwardCurveName()), model);
double strike = parSwapRate + 0.0001 * (isPayer ? moneyness : - moneyness);
double cashAnnuity = cashFunction(parSwapRate, fixSchedule);
double optionValue = cashLattice.getValue(maturity, tenor, moneyness);
maturities.add(maturity);
tenors.add(tenor);
if(isPayer) {
moneynesss.add(moneyness);
values.add(AnalyticFormulas.bachelierOptionImpliedVolatility(parSwapRate, fixSchedule.getFixing(0), strike, cashAnnuity, optionValue));
} else {
moneynesss.add(-moneyness);
values.add(AnalyticFormulas.bachelierOptionImpliedVolatility(parSwapRate, fixSchedule.getFixing(0), strike, cashAnnuity, optionValue
+ 0.0001 * moneyness * cashAnnuity));
}
}
}
}
return new SwaptionDataLattice(referenceDate, QuotingConvention.PAYERVOLATILITYNORMAL, cashLattice.getForwardCurveName(), cashLattice.getDiscountCurveName(),
floatMetaSchedule, fixMetaSchedule,
maturities.stream().mapToInt(Integer::intValue).toArray(),
tenors.stream().mapToInt(Integer::intValue).toArray(),
moneynesss.stream().mapToInt(Integer::intValue).toArray(),
values.stream().mapToDouble(Double::doubleValue).toArray());
}
/**
* Cash function of cash settled swaptions for equidistant tenors.
*
* @param swapRate The swap rate.
* @param schedule The schedule.
* @return The result of the cash function.
*/
private static double cashFunction(double swapRate, Schedule schedule) {
int numberOfPeriods = schedule.getNumberOfPeriods();
double periodLength = 0.0;
for(int index = 0; index < numberOfPeriods; index++) {
periodLength += schedule.getPeriodLength(index);
}
periodLength /= schedule.getNumberOfPeriods();
if(swapRate == 0.0) {
return numberOfPeriods * periodLength;
} else {
return (1 - Math.pow(1 + periodLength * swapRate, - numberOfPeriods)) / swapRate;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy