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

com.opengamma.strata.loader.csv.RatesCalibrationCsvLoader Maven / Gradle / Ivy

There is a newer version: 2.12.46
Show newest version
/*
 * Copyright (C) 2015 - present by OpenGamma Inc. and the OpenGamma group of companies
 *
 * Please see distribution for license.
 */
package com.opengamma.strata.loader.csv;

import static com.opengamma.strata.collect.Guavate.toImmutableList;
import static com.opengamma.strata.collect.Guavate.toImmutableMap;
import static java.util.stream.Collectors.toList;

import java.time.Period;
import java.time.YearMonth;
import java.time.format.DateTimeFormatter;
import java.time.format.DateTimeFormatterBuilder;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Set;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import com.google.common.annotations.VisibleForTesting;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.io.CharSource;
import com.google.common.math.DoubleMath;
import com.opengamma.strata.basics.StandardId;
import com.opengamma.strata.basics.date.SequenceDate;
import com.opengamma.strata.basics.date.Tenor;
import com.opengamma.strata.collect.io.CharSources;
import com.opengamma.strata.collect.io.CsvFile;
import com.opengamma.strata.collect.io.CsvRow;
import com.opengamma.strata.collect.io.ResourceLocator;
import com.opengamma.strata.collect.io.UnicodeBom;
import com.opengamma.strata.collect.result.ParseFailureException;
import com.opengamma.strata.data.FieldName;
import com.opengamma.strata.loader.LoaderUtils;
import com.opengamma.strata.market.curve.CurveDefinition;
import com.opengamma.strata.market.curve.CurveGroupName;
import com.opengamma.strata.market.curve.CurveName;
import com.opengamma.strata.market.curve.CurveNode;
import com.opengamma.strata.market.curve.CurveNodeClashAction;
import com.opengamma.strata.market.curve.CurveNodeDate;
import com.opengamma.strata.market.curve.CurveNodeDateOrder;
import com.opengamma.strata.market.curve.RatesCurveGroupDefinition;
import com.opengamma.strata.market.curve.SeasonalityDefinition;
import com.opengamma.strata.market.curve.node.FixedIborSwapCurveNode;
import com.opengamma.strata.market.curve.node.FixedInflationSwapCurveNode;
import com.opengamma.strata.market.curve.node.FixedOvernightSwapCurveNode;
import com.opengamma.strata.market.curve.node.FraCurveNode;
import com.opengamma.strata.market.curve.node.FxSwapCurveNode;
import com.opengamma.strata.market.curve.node.IborFixingDepositCurveNode;
import com.opengamma.strata.market.curve.node.IborFutureCurveNode;
import com.opengamma.strata.market.curve.node.IborIborSwapCurveNode;
import com.opengamma.strata.market.curve.node.OvernightFutureCurveNode;
import com.opengamma.strata.market.curve.node.OvernightIborSwapCurveNode;
import com.opengamma.strata.market.curve.node.TermDepositCurveNode;
import com.opengamma.strata.market.curve.node.ThreeLegBasisSwapCurveNode;
import com.opengamma.strata.market.curve.node.XCcyIborIborSwapCurveNode;
import com.opengamma.strata.market.curve.node.XCcyOvernightOvernightSwapCurveNode;
import com.opengamma.strata.market.observable.QuoteId;
import com.opengamma.strata.product.deposit.type.IborFixingDepositConvention;
import com.opengamma.strata.product.deposit.type.IborFixingDepositTemplate;
import com.opengamma.strata.product.deposit.type.TermDepositConvention;
import com.opengamma.strata.product.deposit.type.TermDepositTemplate;
import com.opengamma.strata.product.fra.type.FraConvention;
import com.opengamma.strata.product.fra.type.FraTemplate;
import com.opengamma.strata.product.fx.type.FxSwapConvention;
import com.opengamma.strata.product.fx.type.FxSwapTemplate;
import com.opengamma.strata.product.index.type.IborFutureContractSpec;
import com.opengamma.strata.product.index.type.IborFutureTemplate;
import com.opengamma.strata.product.index.type.OvernightFutureContractSpec;
import com.opengamma.strata.product.index.type.OvernightFutureTemplate;
import com.opengamma.strata.product.swap.type.FixedIborSwapConvention;
import com.opengamma.strata.product.swap.type.FixedIborSwapTemplate;
import com.opengamma.strata.product.swap.type.FixedInflationSwapConvention;
import com.opengamma.strata.product.swap.type.FixedInflationSwapTemplate;
import com.opengamma.strata.product.swap.type.FixedOvernightSwapConvention;
import com.opengamma.strata.product.swap.type.FixedOvernightSwapTemplate;
import com.opengamma.strata.product.swap.type.IborIborSwapConvention;
import com.opengamma.strata.product.swap.type.IborIborSwapTemplate;
import com.opengamma.strata.product.swap.type.OvernightIborSwapConvention;
import com.opengamma.strata.product.swap.type.OvernightIborSwapTemplate;
import com.opengamma.strata.product.swap.type.ThreeLegBasisSwapConvention;
import com.opengamma.strata.product.swap.type.ThreeLegBasisSwapTemplate;
import com.opengamma.strata.product.swap.type.XCcyIborIborSwapConvention;
import com.opengamma.strata.product.swap.type.XCcyIborIborSwapTemplate;
import com.opengamma.strata.product.swap.type.XCcyOvernightOvernightSwapConvention;
import com.opengamma.strata.product.swap.type.XCcyOvernightOvernightSwapTemplate;

/**
 * Loads a set of definitions to calibrate rates curves by reading from CSV resources.
 * 

* There are three type of CSV files. *

* The first file is the curve group metadata file. * This file has the following header row:
* {@code Group Name, Curve Type, Reference, Curve Name}. *

    *
  • The 'Group Name' column is the name of the group of curves. *
  • The 'Curve Type' column is the type of the curve, "forward" or "discount". *
  • The 'Reference' column is the reference the curve is used for, such as "USD" or "USD-LIBOR-3M". *
  • The 'Curve Name' column is the name of the curve. *
*

* The second file is the curve settings metadata file. * This file has the following header row:
* {@code Curve Name, Value Type, Day Count, Interpolator, Left Extrapolator, Right Extrapolator}. *

    *
  • The 'Curve Name' column is the name of the curve. *
  • The 'Value Type' column is the type of data in the curve, "zero" for zero rates, or "df" for discount factors. *
  • The 'Day Count' column is the name of the day count, such as "Act/365F". *
  • The 'Interpolator' and extrapolator columns define the interpolator to use. *
*

* The third file is the curve calibration nodes file. * This file has the following header row:
* {@code Curve Name,Label,Symbology,Ticker,Field Name,Type,Convention,Time,Date,Min Gap,Clash Action,Spread}. *

    *
  • The 'Curve Name' column is the name of the curve. *
  • The 'Label' column is the label used to refer to the node. *
  • The 'Symbology' column is the symbology scheme applicable to the ticker used for the market price. *
  • The 'Ticker' column is the identifier within the symbology used for the market price. *
  • The 'Field Name' column is the field name used for the market price, defaulted to "MarketValue", allowing * fields such as 'Bid' or 'Ask' to be specified. *
  • The 'Type' column is the type of the instrument, such as "FRA" or "OIS". *
  • The 'Convention' column is the name of the convention to use. *
  • The 'Time' column is the description of the time, such as "1Y" for a 1 year swap, or "3Mx6M" for a FRA. *
  • The optional 'Date' column is the date to use for the node, defaults to "End", but can be * set to "LastFixing" or a yyyy-MM-dd date. *
  • The optional 'Min Gap' column is the minimum gap between this node and the adjacent nodes. *
  • The optional 'Clash Action' column is the action to perform if the nodes are closer than the minimum gap * or in the wrong order, defaults to "Exception", but can be set to "DropThis" or "DropOther". *
  • The optional 'Spread' column is the spread to add to the instrument. *
*

* Each curve must be contained entirely within a single file, but each file may contain more than * one curve. The curve points do not need to be ordered. *

* CSV files sometimes contain a Unicode Byte Order Mark. * Callers are responsible for handling this, such as by using {@link UnicodeBom}. */ public final class RatesCalibrationCsvLoader { // CSV column headers private static final String CURVE_NAME = "Curve Name"; private static final String CURVE_LABEL = "Label"; private static final String CURVE_SYMBOLOGY_QUOTE = "Symbology"; private static final String CURVE_TICKER_QUOTE = "Ticker"; private static final String CURVE_FIELD_QUOTE = "Field Name"; private static final String CURVE_TYPE = "Type"; private static final String CURVE_CONVENTION = "Convention"; private static final String CURVE_TIME = "Time"; private static final String CURVE_DATE = "Date"; private static final String CURVE_SPREAD = "Spread"; private static final String CURVE_MIN_GAP = "Min Gap"; private static final String CURVE_CLASH_ACTION = "Clash Action"; // these regexes use possessive ?+ and ++ to prevent unnecessary backtracking // Regex to parse FRA time string private static final Pattern FRA_TIME_REGEX = Pattern.compile("P?+([0-9]++)M?+ ?+X ?+P?+([0-9]++)M?+"); // Regex to parse future time string private static final Pattern FUT_TIME_REGEX = Pattern.compile("P?+((?:[0-9]++D)?+(?:[0-9]++W)?+(?:[0-9]++M)?+) ?+[+] ?+([0-9]++)([BF]?)"); // Regex to parse future month string private static final Pattern FUT_MONTH_REGEX = Pattern.compile("([A-Z]{3}[0-9]{2})"); // Regex to parse simple time string with years, months and days private static final Pattern SIMPLE_YMD_TIME_REGEX = Pattern.compile("P?+(([0-9]++Y)?+([0-9]++M)?+([0-9]++W)?+([0-9]++D)?+)"); // Regex to parse simple time string with years and months private static final Pattern SIMPLE_YM_TIME_REGEX = Pattern.compile("P?+(([0-9]++Y)?+([0-9]++M)?+)"); // Regex to parse simple time string with days private static final Pattern SIMPLE_DAYS_REGEX = Pattern.compile("P?+([0-9]++D)?+"); // parse year-month private static final DateTimeFormatter YM_FORMATTER = new DateTimeFormatterBuilder() .parseCaseInsensitive().appendPattern("MMMuu").toFormatter(Locale.ENGLISH); //------------------------------------------------------------------------- /** * Loads one or more CSV format curve calibration files. *

* If the files contain a duplicate entry an exception will be thrown. * * @param groupsResource the curve groups CSV resource * @param settingsResource the curve settings CSV resource * @param curveNodeResources the CSV resources for curve nodes * @return the group definitions, mapped by name * @throws IllegalArgumentException if the files contain a duplicate entry */ public static ImmutableMap load( ResourceLocator groupsResource, ResourceLocator settingsResource, ResourceLocator... curveNodeResources) { return load(groupsResource, settingsResource, ImmutableList.copyOf(curveNodeResources)); } /** * Loads one or more CSV format curve calibration files. *

* If the files contain a duplicate entry an exception will be thrown. * * @param groupsResource the curve groups CSV resource * @param settingsResource the curve settings CSV resource * @param curveNodeResources the CSV resources for curve nodes * @return the group definitions, mapped by name * @throws IllegalArgumentException if the files contain a duplicate entry */ public static ImmutableMap load( ResourceLocator groupsResource, ResourceLocator settingsResource, Collection curveNodeResources) { Collection curveNodeCharSources = curveNodeResources.stream().map(r -> r.getCharSource()).collect(toList()); return parse(groupsResource.getCharSource(), settingsResource.getCharSource(), curveNodeCharSources); } /** * Loads one or more CSV format curve calibration files with seasonality. *

* If the files contain a duplicate entry an exception will be thrown. * * @param groupsResource the curve groups CSV resource * @param settingsResource the curve settings CSV resource * @param seasonalityResource the curve seasonality CSV resource * @param curveNodeResources the CSV resources for curve nodes * @return the group definitions, mapped by name * @throws IllegalArgumentException if the files contain a duplicate entry */ public static ImmutableMap loadWithSeasonality( ResourceLocator groupsResource, ResourceLocator settingsResource, ResourceLocator seasonalityResource, Collection curveNodeResources) { Collection curveNodeCharSources = curveNodeResources.stream().map(r -> r.getCharSource()).collect(toList()); return parseWithSeasonality( groupsResource.getCharSource(), settingsResource.getCharSource(), seasonalityResource.getCharSource(), curveNodeCharSources); } //------------------------------------------------------------------------- /** * Parses one or more CSV format curve calibration files. *

* If the files contain a duplicate entry an exception will be thrown. * * @param groupsCharSource the curve groups CSV character source * @param settingsCharSource the curve settings CSV character source * @param curveNodeCharSources the CSV character sources for curve nodes * @return the group definitions, mapped by name * @throws IllegalArgumentException if the files contain a duplicate entry */ public static ImmutableMap parse( CharSource groupsCharSource, CharSource settingsCharSource, Collection curveNodeCharSources) { return parse0(groupsCharSource, settingsCharSource, ImmutableMap.of(), curveNodeCharSources); } /** * Parses one or more CSV format curve calibration files with seasonality. *

* If the files contain a duplicate entry an exception will be thrown. * * @param groupsCharSource the curve groups CSV character source * @param settingsCharSource the curve settings CSV character source * @param seasonalityResource the seasonality CSV character source * @param curveNodeCharSources the CSV character sources for curve nodes * @return the group definitions, mapped by name * @throws IllegalArgumentException if the files contain a duplicate entry */ public static ImmutableMap parseWithSeasonality( CharSource groupsCharSource, CharSource settingsCharSource, CharSource seasonalityResource, Collection curveNodeCharSources) { Map seasonality = SeasonalityDefinitionCsvLoader.parseSeasonalityDefinitions(seasonalityResource); return parse0(groupsCharSource, settingsCharSource, seasonality, curveNodeCharSources); } // parse based on pre-parsed seasonality private static ImmutableMap parse0( CharSource groupsCharSource, CharSource settingsCharSource, Map seasonality, Collection curveNodeCharSources) { try { // load curve groups and settings List curveGroups = RatesCurveGroupDefinitionCsvLoader.parseCurveGroupDefinitions(groupsCharSource); Map settingsMap = RatesCurvesCsvLoader.parseCurveSettings(settingsCharSource); // load curve definitions List curveDefinitions = curveNodeCharSources.stream() .flatMap(res -> parseSingle(res, settingsMap).stream()) .collect(toImmutableList()); // check for any curve setting without a matching definition List defCurveNames = curveDefinitions.stream() .map(def -> def.getName()) .collect(toImmutableList()); Set settingCurveNames = settingsMap.keySet(); for (CurveName settingCurveName : settingCurveNames) { if (!defCurveNames.contains(settingCurveName)) { throw new ParseFailureException( "Error parsing rates calibration CSV files: Missing nodes for curve '{value}'", settingCurveName.getName()); } } // Add the curve definitions to the curve group definitions return curveGroups.stream() .map(groupDefinition -> groupDefinition.withCurveDefinitions(curveDefinitions).withSeasonalityDefinitions(seasonality)) .collect(toImmutableMap(groupDefinition -> groupDefinition.getName())); } catch (ParseFailureException ex) { throw ex; } catch (RuntimeException ex) { throw new ParseFailureException(ex, "Error parsing rates calibration CSV files: {exceptionMessage}", ex.getMessage()); } } //------------------------------------------------------------------------- // loads a single curves CSV file // requestedDate can be null, meaning load all dates private static List parseSingle( CharSource charSource, Map settingsMap) { try { CsvFile csv = CsvFile.of(charSource, true); Map> allNodes = new HashMap<>(); for (CsvRow row : csv.rows()) { String curveNameStr = row.getField(CURVE_NAME); String label = row.getField(CURVE_LABEL); String symbologyQuoteStr = row.getField(CURVE_SYMBOLOGY_QUOTE); String tickerQuoteStr = row.getField(CURVE_TICKER_QUOTE); String fieldQuoteStr = row.getField(CURVE_FIELD_QUOTE); String typeStr = row.getField(CURVE_TYPE); String conventionStr = row.getField(CURVE_CONVENTION); String timeStr = row.getField(CURVE_TIME); String dateStr = row.findField(CURVE_DATE).orElse(""); String minGapStr = row.findField(CURVE_MIN_GAP).orElse(""); String clashActionStr = row.findField(CURVE_CLASH_ACTION).orElse(""); String spreadStr = row.findField(CURVE_SPREAD).orElse(""); CurveName curveName = CurveName.of(curveNameStr); StandardId quoteStandardId = StandardId.of(symbologyQuoteStr, tickerQuoteStr); FieldName quoteField = fieldQuoteStr.isEmpty() ? FieldName.MARKET_VALUE : FieldName.of(fieldQuoteStr); QuoteId quoteId = QuoteId.of(quoteStandardId, quoteField); double spread = spreadStr.isEmpty() ? 0d : Double.parseDouble(spreadStr); CurveNodeDate date = parseDate(dateStr); CurveNodeDateOrder order = parseDateOrder(minGapStr, clashActionStr); List curveNodes = allNodes.computeIfAbsent(curveName, k -> new ArrayList<>()); curveNodes.add(createCurveNode(typeStr, conventionStr, timeStr, label, quoteId, spread, date, order)); } return buildCurveDefinition(settingsMap, allNodes); } catch (RuntimeException ex) { throw new ParseFailureException( ex, "Error parsing curve definition CSV file '{fileName}': {exceptionMessage}", CharSources.extractFileName(charSource), ex.getMessage()); } } // parse date order private static CurveNodeDate parseDate(String dateStr) { if (dateStr.isEmpty()) { return CurveNodeDate.END; } if (dateStr.length() == 10 && dateStr.charAt(4) == '-' && dateStr.charAt(7) == '-') { return CurveNodeDate.of(LoaderUtils.parseDate(dateStr)); } String dateUpper = dateStr.toUpperCase(Locale.ENGLISH); if (dateUpper.equals("END")) { return CurveNodeDate.END; } if (dateUpper.equals("LASTFIXING")) { return CurveNodeDate.LAST_FIXING; } throw new ParseFailureException( "Invalid format for node date, should be date in 'yyyy-MM-dd' format, 'End' or 'LastFixing': '{value}'", dateUpper); } // parse date order private static CurveNodeDateOrder parseDateOrder(String minGapStr, String clashActionStr) { CurveNodeClashAction clashAction = clashActionStr.isEmpty() ? CurveNodeClashAction.EXCEPTION : CurveNodeClashAction.of(clashActionStr); if (minGapStr.isEmpty()) { return CurveNodeDateOrder.of(1, clashAction); } Matcher matcher = SIMPLE_DAYS_REGEX.matcher(minGapStr.toUpperCase(Locale.ENGLISH)); if (!matcher.matches()) { throw new ParseFailureException("Invalid days format for minimum gap, should be 2D or P2D: '{value}'", minGapStr); } Period minGap = Period.parse("P" + matcher.group(1)); return CurveNodeDateOrder.of(minGap.getDays(), clashAction); } // build the curves private static List buildCurveDefinition( Map settingsMap, Map> allNodes) { ImmutableList.Builder results = ImmutableList.builder(); for (Map.Entry> entry : allNodes.entrySet()) { CurveName name = entry.getKey(); LoadedCurveSettings settings = settingsMap.get(name); if (settings == null) { throw new ParseFailureException("Missing settings for curve '{value}'", name); } results.add(settings.createCurveDefinition(entry.getValue())); } return results.build(); } //------------------------------------------------------------------------- // create the curve node private static CurveNode createCurveNode( String typeStr, String conventionStr, String timeStr, String label, QuoteId quoteId, double spread, CurveNodeDate date, CurveNodeDateOrder order) { if ("DEP".equalsIgnoreCase(typeStr) || "TermDeposit".equalsIgnoreCase(typeStr)) { return curveTermDepositCurveNode(conventionStr, timeStr, label, quoteId, spread, date, order); } if ("FIX".equalsIgnoreCase(typeStr) || "IborFixingDeposit".equalsIgnoreCase(typeStr)) { return curveIborFixingDepositCurveNode(conventionStr, label, quoteId, spread, date, order); } if ("FRA".equalsIgnoreCase(typeStr)) { return curveFraCurveNode(conventionStr, timeStr, label, quoteId, spread, date, order); } if ("IFU".equalsIgnoreCase(typeStr) || "IborFuture".equalsIgnoreCase(typeStr)) { return curveIborFutureCurveNode(conventionStr, timeStr, label, quoteId, spread, date, order); } if ("ONF".equalsIgnoreCase(typeStr) || "OvernightFuture".equalsIgnoreCase(typeStr)) { return curveOvernightFutureCurveNode(conventionStr, timeStr, label, quoteId, spread, date, order); } if ("OIS".equalsIgnoreCase(typeStr) || "FixedOvernightSwap".equalsIgnoreCase(typeStr)) { return curveFixedOvernightCurveNode(conventionStr, timeStr, label, quoteId, spread, date, order); } if ("IRS".equalsIgnoreCase(typeStr) || "FixedIborSwap".equalsIgnoreCase(typeStr)) { return curveFixedIborCurveNode(conventionStr, timeStr, label, quoteId, spread, date, order); } if ("BAS".equalsIgnoreCase(typeStr) || "IborIborSwap".equalsIgnoreCase(typeStr)) { return curveIborIborCurveNode(conventionStr, timeStr, label, quoteId, spread, date, order); } if ("BS3".equalsIgnoreCase(typeStr) || "ThreeLegBasisSwap".equalsIgnoreCase(typeStr)) { return curveThreeLegBasisCurveNode(conventionStr, timeStr, label, quoteId, spread, date, order); } if ("ONI".equalsIgnoreCase(typeStr) || "OvernightIborBasisSwap".equalsIgnoreCase(typeStr)) { return curveOvernightIborCurveNode(conventionStr, timeStr, label, quoteId, spread, date, order); } if ("XCS".equalsIgnoreCase(typeStr) || "XCcyIborIborSwap".equalsIgnoreCase(typeStr)) { return curveXCcyIborIborCurveNode(conventionStr, timeStr, label, quoteId, spread, date, order); } if ("XCO".equalsIgnoreCase(typeStr) || "XCcyOvernightOvernightSwap".equalsIgnoreCase(typeStr)) { return curveXCcyOvernightOvernightCurveNode(conventionStr, timeStr, label, quoteId, spread, date, order); } if ("FXS".equalsIgnoreCase(typeStr) || "FxSwap".equalsIgnoreCase(typeStr)) { return curveFxSwapCurveNode(conventionStr, timeStr, label, quoteId, spread, date, order); } if ("INF".equalsIgnoreCase(typeStr) || "FixedInflationSwap".equalsIgnoreCase(typeStr)) { return curveFixedInflationCurveNode(conventionStr, timeStr, label, quoteId, spread, date, order); } throw new ParseFailureException("Invalid curve node type '{type}'", typeStr); } private static CurveNode curveTermDepositCurveNode( String conventionStr, String timeStr, String label, QuoteId quoteId, double spread, CurveNodeDate date, CurveNodeDateOrder order) { Matcher matcher = SIMPLE_YMD_TIME_REGEX.matcher(timeStr.toUpperCase(Locale.ENGLISH)); if (!matcher.matches()) { throw new ParseFailureException("Invalid time format for Term Deposit: '{value}'", timeStr); } Period periodToEnd = Period.parse("P" + matcher.group(1)); TermDepositConvention convention = TermDepositConvention.of(conventionStr); TermDepositTemplate template = TermDepositTemplate.of(periodToEnd, convention); return TermDepositCurveNode.builder() .template(template) .rateId(quoteId) .additionalSpread(spread) .label(label) .date(date) .dateOrder(order) .build(); } private static CurveNode curveIborFixingDepositCurveNode( String conventionStr, String label, QuoteId quoteId, double spread, CurveNodeDate date, CurveNodeDateOrder order) { IborFixingDepositConvention convention = IborFixingDepositConvention.of(conventionStr); IborFixingDepositTemplate template = IborFixingDepositTemplate.of( convention.getIndex().getTenor().getPeriod(), convention); return IborFixingDepositCurveNode.builder() .template(template) .rateId(quoteId) .additionalSpread(spread) .label(label) .date(date) .dateOrder(order) .build(); } private static CurveNode curveFraCurveNode( String conventionStr, String timeStr, String label, QuoteId quoteId, double spread, CurveNodeDate date, CurveNodeDateOrder order) { Matcher matcher = FRA_TIME_REGEX.matcher(timeStr.toUpperCase(Locale.ENGLISH)); if (!matcher.matches()) { throw new ParseFailureException("Invalid time format for FRA: '{value}'", timeStr); } Period periodToStart = Period.parse("P" + matcher.group(1) + "M"); Period periodToEnd = Period.parse("P" + matcher.group(2) + "M"); FraConvention convention = FraConvention.of(conventionStr); FraTemplate template = FraTemplate.of(periodToStart, periodToEnd, convention); return FraCurveNode.builder() .template(template) .rateId(quoteId) .additionalSpread(spread) .label(label) .date(date) .dateOrder(order) .build(); } //------------------------------------------------------------------------- @VisibleForTesting static IborFutureCurveNode curveIborFutureCurveNode( String conventionStr, String timeStr, String label, QuoteId quoteId, double spread, CurveNodeDate date, CurveNodeDateOrder order) { Matcher matcher = FUT_TIME_REGEX.matcher(timeStr.toUpperCase(Locale.ENGLISH)); Matcher matcher2 = FUT_MONTH_REGEX.matcher(timeStr.toUpperCase(Locale.ENGLISH)); SequenceDate seqDate = null; if (matcher.matches()) { Period periodToStart = Period.parse("P" + matcher.group(1)); int sequenceNumber = Integer.parseInt(matcher.group(2)); String seqType = matcher.group(3); if ("F".equals(seqType)) { seqDate = SequenceDate.full(periodToStart, sequenceNumber); } else { seqDate = SequenceDate.base(periodToStart, sequenceNumber); } } else if (matcher2.matches()) { YearMonth yearMonth = YearMonth.parse(matcher2.group(1), YM_FORMATTER); seqDate = SequenceDate.base(yearMonth); } else { throw new ParseFailureException("Invalid time format for Ibor Future: '{value}'", timeStr); } IborFutureContractSpec contractSpec = IborFutureContractSpec.of(conventionStr); IborFutureTemplate template = IborFutureTemplate.of(seqDate, contractSpec); return IborFutureCurveNode.builder() .template(template) .rateId(quoteId) .additionalSpread(spread) .label(label) .date(date) .dateOrder(order) .build(); } private static CurveNode curveOvernightFutureCurveNode( String conventionStr, String timeStr, String label, QuoteId quoteId, double spread, CurveNodeDate date, CurveNodeDateOrder order) { Matcher matcher = FUT_TIME_REGEX.matcher(timeStr.toUpperCase(Locale.ENGLISH)); Matcher matcher2 = FUT_MONTH_REGEX.matcher(timeStr.toUpperCase(Locale.ENGLISH)); SequenceDate seqDate = null; if (matcher.matches()) { Period periodToStart = Period.parse("P" + matcher.group(1)); int sequenceNumber = Integer.parseInt(matcher.group(2)); String seqType = matcher.group(3); if ("F".equals(seqType)) { seqDate = SequenceDate.full(periodToStart, sequenceNumber); } else { seqDate = SequenceDate.base(periodToStart, sequenceNumber); } } else if (matcher2.matches()) { YearMonth yearMonth = YearMonth.parse(matcher2.group(1), YM_FORMATTER); seqDate = SequenceDate.base(yearMonth); } else { throw new ParseFailureException("Invalid time format for Overnight Future: '{value}'", timeStr); } OvernightFutureContractSpec contractSpec = OvernightFutureContractSpec.of(conventionStr); OvernightFutureTemplate template = OvernightFutureTemplate.of(seqDate, contractSpec); return OvernightFutureCurveNode.builder() .template(template) .rateId(quoteId) .additionalSpread(spread) .label(label) .date(date) .dateOrder(order) .build(); } //------------------------------------------------------------------------- private static CurveNode curveFixedOvernightCurveNode( String conventionStr, String timeStr, String label, QuoteId quoteId, double spread, CurveNodeDate date, CurveNodeDateOrder order) { Matcher matcher = SIMPLE_YMD_TIME_REGEX.matcher(timeStr.toUpperCase(Locale.ENGLISH)); if (!matcher.matches()) { throw new ParseFailureException("Invalid time format for Fixed-Overnight swap: '{value}'", timeStr); } Period periodToEnd = Period.parse("P" + matcher.group(1)); FixedOvernightSwapConvention convention = FixedOvernightSwapConvention.of(conventionStr); FixedOvernightSwapTemplate template = FixedOvernightSwapTemplate.of(Tenor.of(periodToEnd), convention); return FixedOvernightSwapCurveNode.builder() .template(template) .rateId(quoteId) .additionalSpread(spread) .label(label) .date(date) .dateOrder(order) .build(); } private static CurveNode curveFixedIborCurveNode( String conventionStr, String timeStr, String label, QuoteId quoteId, double spread, CurveNodeDate date, CurveNodeDateOrder order) { Matcher matcher = SIMPLE_YMD_TIME_REGEX.matcher(timeStr.toUpperCase(Locale.ENGLISH)); if (!matcher.matches()) { throw new ParseFailureException("Invalid time format for Fixed-Ibor swap: '{value}'", timeStr); } Period periodToEnd = Period.parse("P" + matcher.group(1)); FixedIborSwapConvention convention = FixedIborSwapConvention.of(conventionStr); FixedIborSwapTemplate template = FixedIborSwapTemplate.of(Tenor.of(periodToEnd), convention); return FixedIborSwapCurveNode.builder() .template(template) .rateId(quoteId) .additionalSpread(spread) .label(label) .date(date) .dateOrder(order) .build(); } private static CurveNode curveIborIborCurveNode( String conventionStr, String timeStr, String label, QuoteId quoteId, double spread, CurveNodeDate date, CurveNodeDateOrder order) { Matcher matcher = SIMPLE_YM_TIME_REGEX.matcher(timeStr.toUpperCase(Locale.ENGLISH)); if (!matcher.matches()) { throw new ParseFailureException("Invalid time format for Ibor-Ibor swap: '{value}'", timeStr); } Period periodToEnd = Period.parse("P" + matcher.group(1)); IborIborSwapConvention convention = IborIborSwapConvention.of(conventionStr); IborIborSwapTemplate template = IborIborSwapTemplate.of(Tenor.of(periodToEnd), convention); return IborIborSwapCurveNode.builder() .template(template) .rateId(quoteId) .additionalSpread(spread) .label(label) .date(date) .dateOrder(order) .build(); } private static CurveNode curveThreeLegBasisCurveNode( String conventionStr, String timeStr, String label, QuoteId quoteId, double spread, CurveNodeDate date, CurveNodeDateOrder order) { Matcher matcher = SIMPLE_YM_TIME_REGEX.matcher(timeStr.toUpperCase(Locale.ENGLISH)); if (!matcher.matches()) { throw new ParseFailureException("Invalid time format for Three legs basis swap: '{value}'", timeStr); } Period periodToEnd = Period.parse("P" + matcher.group(1)); ThreeLegBasisSwapConvention convention = ThreeLegBasisSwapConvention.of(conventionStr); ThreeLegBasisSwapTemplate template = ThreeLegBasisSwapTemplate.of(Tenor.of(periodToEnd), convention); return ThreeLegBasisSwapCurveNode.builder() .template(template) .rateId(quoteId) .additionalSpread(spread) .label(label) .date(date) .dateOrder(order) .build(); } private static CurveNode curveXCcyIborIborCurveNode( String conventionStr, String timeStr, String label, QuoteId quoteId, double spread, CurveNodeDate date, CurveNodeDateOrder order) { Matcher matcher = SIMPLE_YM_TIME_REGEX.matcher(timeStr.toUpperCase(Locale.ENGLISH)); if (!matcher.matches()) { throw new ParseFailureException("Invalid time format for Cross Currency Swap: '{value}'", timeStr); } Period periodToEnd = Period.parse("P" + matcher.group(1)); XCcyIborIborSwapConvention convention = XCcyIborIborSwapConvention.of(conventionStr); XCcyIborIborSwapTemplate template = XCcyIborIborSwapTemplate.of(Tenor.of(periodToEnd), convention); return XCcyIborIborSwapCurveNode.builder() .template(template) .spreadId(quoteId) .additionalSpread(spread) .label(label) .date(date) .dateOrder(order) .build(); } private static CurveNode curveXCcyOvernightOvernightCurveNode( String conventionStr, String timeStr, String label, QuoteId quoteId, double spread, CurveNodeDate date, CurveNodeDateOrder order) { Matcher matcher = SIMPLE_YM_TIME_REGEX.matcher(timeStr.toUpperCase(Locale.ENGLISH)); if (!matcher.matches()) { throw new ParseFailureException("Invalid time format for Cross Currency Swap: '{value}'", timeStr); } Period periodToEnd = Period.parse("P" + matcher.group(1)); XCcyOvernightOvernightSwapConvention convention = XCcyOvernightOvernightSwapConvention.of(conventionStr); XCcyOvernightOvernightSwapTemplate template = XCcyOvernightOvernightSwapTemplate.of(Tenor.of(periodToEnd), convention); return XCcyOvernightOvernightSwapCurveNode.builder() .template(template) .spreadId(quoteId) .additionalSpread(spread) .label(label) .date(date) .dateOrder(order) .build(); } private static CurveNode curveOvernightIborCurveNode( String conventionStr, String timeStr, String label, QuoteId quoteId, double spread, CurveNodeDate date, CurveNodeDateOrder order) { Matcher matcher = SIMPLE_YMD_TIME_REGEX.matcher(timeStr.toUpperCase(Locale.ENGLISH)); if (!matcher.matches()) { throw new ParseFailureException("Invalid time format for Overnight-Ibor swap: '{value}'", timeStr); } Period periodToEnd = Period.parse("P" + matcher.group(1)); OvernightIborSwapConvention convention = OvernightIborSwapConvention.of(conventionStr); OvernightIborSwapTemplate template = OvernightIborSwapTemplate.of(Tenor.of(periodToEnd), convention); return OvernightIborSwapCurveNode.builder() .template(template) .rateId(quoteId) .additionalSpread(spread) .label(label) .date(date) .dateOrder(order) .build(); } private static CurveNode curveFxSwapCurveNode( String conventionStr, String timeStr, String label, QuoteId quoteId, double spread, CurveNodeDate date, CurveNodeDateOrder order) { if (!DoubleMath.fuzzyEquals(spread, 0d, 1e-10d)) { throw new ParseFailureException("Additional spread must be zero for FX swaps"); } Matcher matcher = SIMPLE_YMD_TIME_REGEX.matcher(timeStr.toUpperCase(Locale.ENGLISH)); if (!matcher.matches()) { throw new ParseFailureException("Invalid time format for FX swap: '{value}'", timeStr); } Period periodToEnd = Period.parse("P" + matcher.group(1)); FxSwapConvention convention = FxSwapConvention.of(conventionStr); FxSwapTemplate template = FxSwapTemplate.of(periodToEnd, convention); return FxSwapCurveNode.builder() .template(template) .farForwardPointsId(quoteId) .label(label) .date(date) .dateOrder(order) .build(); } private static CurveNode curveFixedInflationCurveNode( String conventionStr, String timeStr, String label, QuoteId quoteId, double spread, CurveNodeDate date, CurveNodeDateOrder order) { Matcher matcher = SIMPLE_YM_TIME_REGEX.matcher(timeStr.toUpperCase(Locale.ENGLISH)); if (!matcher.matches()) { throw new ParseFailureException("Invalid time format for Fixed-Inflation swap: '{value}'", timeStr); } Period periodToEnd = Period.parse("P" + matcher.group(1)); FixedInflationSwapConvention convention = FixedInflationSwapConvention.of(conventionStr); FixedInflationSwapTemplate template = FixedInflationSwapTemplate.of(Tenor.of(periodToEnd), convention); return FixedInflationSwapCurveNode.builder() .template(template) .rateId(quoteId) .additionalSpread(spread) .label(label) .date(date) .dateOrder(order) .build(); } //------------------------------------------------------------------------- /** * Restricted constructor. */ private RatesCalibrationCsvLoader() { } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy