/** * Copyright (C) 2012 - present by OpenGamma Inc. and the OpenGamma group of companies * * Please see distribution for license. */ package com.opengamma.financial.analytics.model.credit; import java.util.Collections; import java.util.HashSet; import java.util.Set; import org.threeten.bp.Clock; import org.threeten.bp.Period; import org.threeten.bp.ZonedDateTime; import com.google.common.collect.Iterables; import com.opengamma.OpenGammaRuntimeException; import com.opengamma.analytics.financial.credit.isdastandardmodel.ISDACompliantYieldCurve; import com.opengamma.analytics.financial.credit.isdastandardmodel.ISDACompliantYieldCurveBuild; import com.opengamma.analytics.financial.credit.isdastandardmodel.ISDAInstrumentTypes; import com.opengamma.engine.ComputationTarget; import com.opengamma.engine.ComputationTargetSpecification; import com.opengamma.engine.function.AbstractFunction; import com.opengamma.engine.function.FunctionCompilationContext; import com.opengamma.engine.function.FunctionExecutionContext; import com.opengamma.engine.function.FunctionInputs; import com.opengamma.engine.target.ComputationTargetType; import com.opengamma.engine.value.ComputedValue; import com.opengamma.engine.value.ValueProperties; import com.opengamma.engine.value.ValuePropertyNames; import com.opengamma.engine.value.ValueRequirement; import com.opengamma.engine.value.ValueRequirementNames; import com.opengamma.engine.value.ValueSpecification; import com.opengamma.financial.analytics.ircurve.FixedIncomeStripWithSecurity; import com.opengamma.financial.analytics.ircurve.InterpolatedYieldCurveSpecificationWithSecurities; import com.opengamma.financial.analytics.ircurve.YieldCurveData; import com.opengamma.financial.analytics.model.cds.ISDAFunctionConstants; import com.opengamma.financial.convention.businessday.BusinessDayConvention; import com.opengamma.financial.convention.daycount.DayCount; import com.opengamma.financial.convention.daycount.DayCounts; import com.opengamma.financial.convention.frequency.SimpleFrequency; import com.opengamma.financial.security.FinancialSecurity; import com.opengamma.financial.security.cash.CashSecurity; import com.opengamma.financial.security.swap.FixedInterestRateLeg; import com.opengamma.financial.security.swap.SwapSecurity; import com.opengamma.util.async.AsynchronousExecution; import com.opengamma.util.money.Currency; /** * */ public class ISDAYieldCurveFunction extends AbstractFunction.NonCompiledInvoker { // ISDA fixes yield curve daycout to Act/365 private static final DayCount ACT_365 = DayCounts.ACT_365; @Override public Set<ComputedValue> execute(final FunctionExecutionContext executionContext, final FunctionInputs inputs, final ComputationTarget target, final Set<ValueRequirement> desiredValues) throws AsynchronousExecution { final Clock snapshotClock = executionContext.getValuationClock(); final ZonedDateTime now = ZonedDateTime.now(snapshotClock); final ValueRequirement desiredValue = desiredValues.iterator().next(); final String curveName = desiredValue.getConstraint(ValuePropertyNames.CURVE); final String curveCalculationConfig = desiredValue.getConstraint(ValuePropertyNames.CURVE_CALCULATION_CONFIG); final String offsetString = desiredValue.getConstraint(ISDAFunctionConstants.ISDA_CURVE_OFFSET); final int offset = Integer.parseInt(offsetString); final Object dataObject = inputs.getValue(ValueRequirementNames.YIELD_CURVE_DATA); if (dataObject == null) { throw new OpenGammaRuntimeException("Couldn't get yield curve data for " + curveName); } final YieldCurveData yieldCurveData = (YieldCurveData) dataObject; final InterpolatedYieldCurveSpecificationWithSecurities yieldCurveSpec = yieldCurveData.getCurveSpecification(); final int nNodes = yieldCurveSpec.getStrips().size(); final double[] marketDataForCurve = new double[nNodes]; final ISDAInstrumentTypes[] instruments = new ISDAInstrumentTypes[nNodes]; final Period[] tenors = new Period[nNodes]; int k = 0; DayCount cashDCC = null; DayCount fixDCC = null; BusinessDayConvention floatBadDayConv = null; Period paymentTenor = null; for (final FixedIncomeStripWithSecurity strip : yieldCurveSpec.getStrips()) { final String securityType = strip.getSecurity().getSecurityType(); if (!(securityType.equals(CashSecurity.SECURITY_TYPE) || securityType.equals(SwapSecurity.SECURITY_TYPE)/* || securityType.equals(specObject)*/)) { throw new OpenGammaRuntimeException("ISDA curves should only use Libor and swap rates"); } final Double marketValue = yieldCurveData.getDataPoint(strip.getSecurityIdentifier()); if (marketValue == null) { throw new OpenGammaRuntimeException("Could not get market data for " + strip); } final FinancialSecurity financialSecurity = (FinancialSecurity) strip.getSecurity(); instruments[k] = securityType.equals(CashSecurity.SECURITY_TYPE) ? ISDAInstrumentTypes.MoneyMarket : ISDAInstrumentTypes.Swap; if (financialSecurity instanceof SwapSecurity) { SwapSecurity swap = (SwapSecurity) financialSecurity; if (swap.getPayLeg() instanceof FixedInterestRateLeg) { fixDCC = swap.getPayLeg().getDayCount(); paymentTenor = ((SimpleFrequency) swap.getPayLeg().getFrequency()).toPeriodFrequency().getPeriod(); floatBadDayConv = swap.getReceiveLeg().getBusinessDayConvention(); } else { fixDCC = swap.getReceiveLeg().getDayCount(); paymentTenor = ((SimpleFrequency) swap.getReceiveLeg().getFrequency()).toPeriodFrequency().getPeriod(); floatBadDayConv = swap.getPayLeg().getBusinessDayConvention(); } } else { cashDCC = ((CashSecurity) financialSecurity).getDayCount(); } marketDataForCurve[k] = marketValue; tenors[k] = strip.getResolvedTenor().getPeriod(); k++; } //TODO: Check spot date logic final ISDACompliantYieldCurve yieldCurve = ISDACompliantYieldCurveBuild.build(now.toLocalDate(), now.toLocalDate().minusDays(offset), instruments, tenors, marketDataForCurve, cashDCC, fixDCC, paymentTenor, ACT_365, floatBadDayConv); final ValueProperties properties = createValueProperties().with(ValuePropertyNames.CURVE, curveName).with(ISDAFunctionConstants.ISDA_CURVE_OFFSET, offsetString) .with(ValuePropertyNames.CURVE_CALCULATION_CONFIG, curveCalculationConfig).with(ValuePropertyNames.CURVE_CALCULATION_METHOD, ISDAFunctionConstants.ISDA_METHOD_NAME) .with(ISDAFunctionConstants.ISDA_IMPLEMENTATION, ISDAFunctionConstants.ISDA_IMPLEMENTATION_APPROX).get(); final ValueSpecification spec = new ValueSpecification(ValueRequirementNames.YIELD_CURVE, target.toSpecification(), properties); return Collections.singleton(new ComputedValue(spec, yieldCurve)); } @Override public ComputationTargetType getTargetType() { return ComputationTargetType.CURRENCY; } @Override public boolean canApplyTo(final FunctionCompilationContext context, final ComputationTarget target) { return Currency.OBJECT_SCHEME.equals(target.getUniqueId().getScheme()); } @Override public Set<ValueSpecification> getResults(final FunctionCompilationContext context, final ComputationTarget target) { final ValueProperties properties = createValueProperties().withAny(ValuePropertyNames.CURVE).withAny(ValuePropertyNames.CURVE_CALCULATION_CONFIG) .withAny(ISDAFunctionConstants.ISDA_CURVE_OFFSET).with(ISDAFunctionConstants.ISDA_IMPLEMENTATION, ISDAFunctionConstants.ISDA_IMPLEMENTATION_APPROX) .with(ValuePropertyNames.CURVE_CALCULATION_METHOD, ISDAFunctionConstants.ISDA_METHOD_NAME).get(); return Collections.singleton(new ValueSpecification(ValueRequirementNames.YIELD_CURVE, target.toSpecification(), properties)); } @Override public Set<ValueRequirement> getRequirements(final FunctionCompilationContext context, final ComputationTarget target, final ValueRequirement desiredValue) { final ValueProperties constraints = desiredValue.getConstraints(); final Set<String> curveCalculationMethod = constraints.getValues(ValuePropertyNames.CURVE_CALCULATION_METHOD); if (curveCalculationMethod == null) { return null; } if (curveCalculationMethod.size() == 1) { if (!ISDAFunctionConstants.ISDA_METHOD_NAME.equals(Iterables.getOnlyElement(curveCalculationMethod))) { return null; } final Set<String> implementationMethod = constraints.getValues(ISDAFunctionConstants.ISDA_IMPLEMENTATION); if (implementationMethod != null && implementationMethod.size() == 1) { if (!ISDAFunctionConstants.ISDA_IMPLEMENTATION_APPROX.equals(Iterables.getOnlyElement(implementationMethod))) { return null; } } } final Set<String> curveNames = constraints.getValues(ValuePropertyNames.CURVE); if (curveNames == null || curveNames.size() != 1) { return null; } final Set<String> curveCalculationConfigs = constraints.getValues(ValuePropertyNames.CURVE_CALCULATION_CONFIG); if (curveCalculationConfigs == null || curveCalculationConfigs.size() != 1) { return null; } final Set<String> isdaCurveOffsets = constraints.getValues(ISDAFunctionConstants.ISDA_CURVE_OFFSET); if (isdaCurveOffsets == null || isdaCurveOffsets.size() != 1) { return null; } final String curveName = Iterables.getOnlyElement(curveNames); final String curveCalculationConfig = Iterables.getOnlyElement(curveCalculationConfigs); final ValueProperties properties = ValueProperties.builder().with(ValuePropertyNames.CURVE, curveName).with(ValuePropertyNames.CURVE_CALCULATION_CONFIG, curveCalculationConfig) .withOptional(ValuePropertyNames.CURVE_CALCULATION_CONFIG).get(); final ValueProperties curveTSProperties = ValueProperties.builder().with(ValuePropertyNames.CURVE_CALCULATION_CONFIG, curveCalculationConfig).get(); final Set<ValueRequirement> requirements = new HashSet<>(); final ComputationTargetSpecification targetSpec = target.toSpecification(); requirements.add(new ValueRequirement(ValueRequirementNames.YIELD_CURVE_DATA, targetSpec, properties)); requirements.add(new ValueRequirement(ValueRequirementNames.YIELD_CURVE_INSTRUMENT_CONVERSION_HISTORICAL_TIME_SERIES, targetSpec, curveTSProperties)); return requirements; } @Override public boolean canHandleMissingRequirements() { // historical time series likely to be absent return true; } @Override public boolean canHandleMissingInputs() { return true; } }