/** * Copyright (C) 2013 - present by OpenGamma Inc. and the OpenGamma group of companies * * Please see distribution for license. */ package com.opengamma.financial.analytics.model.discounting; import static com.opengamma.engine.value.ValueRequirementNames.SWAP_PAY_LEG_DETAILS; import static com.opengamma.engine.value.ValueRequirementNames.SWAP_RECEIVE_LEG_DETAILS; import java.util.Collections; import java.util.List; import java.util.Set; import org.threeten.bp.Clock; import org.threeten.bp.Instant; import org.threeten.bp.LocalDate; import org.threeten.bp.ZonedDateTime; import com.google.common.collect.Iterables; import com.opengamma.OpenGammaRuntimeException; import com.opengamma.analytics.financial.forex.method.FXMatrix; import com.opengamma.analytics.financial.instrument.annuity.AnnuityDefinition; import com.opengamma.analytics.financial.instrument.payment.PaymentDefinition; import com.opengamma.analytics.financial.instrument.swap.SwapDefinition; import com.opengamma.analytics.financial.interestrate.AnnuityAccrualDatesVisitor; import com.opengamma.analytics.financial.interestrate.AnnuityFixedRatesVisitor; import com.opengamma.analytics.financial.interestrate.AnnuityFixingDatesVisitor; import com.opengamma.analytics.financial.interestrate.AnnuityFixingYearFractionsVisitor; import com.opengamma.analytics.financial.interestrate.AnnuityGearingsVisitor; import com.opengamma.analytics.financial.interestrate.AnnuityIndexTenorsVisitor; import com.opengamma.analytics.financial.interestrate.AnnuityNotionalsVisitor; import com.opengamma.analytics.financial.interestrate.AnnuityPaymentAmountsVisitor; import com.opengamma.analytics.financial.interestrate.AnnuityPaymentDatesVisitor; import com.opengamma.analytics.financial.interestrate.AnnuityPaymentFractionsVisitor; import com.opengamma.analytics.financial.interestrate.AnnuityPaymentTimesVisitor; import com.opengamma.analytics.financial.interestrate.AnnuitySpreadsVisitor; import com.opengamma.analytics.financial.interestrate.InstrumentDerivative; import com.opengamma.analytics.financial.interestrate.annuity.derivative.Annuity; import com.opengamma.analytics.financial.interestrate.payments.derivative.Payment; import com.opengamma.analytics.financial.interestrate.swap.derivative.Swap; import com.opengamma.analytics.financial.interestrate.swap.provider.AnnuityDiscountFactorsVisitor; import com.opengamma.analytics.financial.interestrate.swap.provider.AnnuityForwardRatesVisitor; import com.opengamma.analytics.financial.interestrate.swap.provider.AnnuityProjectedPaymentsVisitor; import com.opengamma.analytics.financial.provider.description.interestrate.MulticurveProviderInterface; import com.opengamma.core.security.Security; import com.opengamma.engine.ComputationTarget; import com.opengamma.engine.function.CompiledFunctionDefinition; import com.opengamma.engine.function.FunctionCompilationContext; import com.opengamma.engine.function.FunctionExecutionContext; import com.opengamma.engine.function.FunctionInputs; import com.opengamma.engine.value.ComputedValue; import com.opengamma.engine.value.ValueRequirement; import com.opengamma.engine.value.ValueRequirementNames; import com.opengamma.engine.value.ValueSpecification; import com.opengamma.financial.analytics.model.fixedincome.FixedSwapLegDetails; import com.opengamma.financial.analytics.model.fixedincome.FloatingSwapLegDetails; import com.opengamma.financial.security.swap.FixedInterestRateLeg; import com.opengamma.financial.security.swap.SwapSecurity; import com.opengamma.util.money.CurrencyAmount; import com.opengamma.util.time.Tenor; import com.opengamma.util.tuple.Pair; /** * Produces information about cash-flows for a swap leg. */ public class DiscountingSwapLegDetailFunction extends DiscountingFunction { /** Whether this function returns details for the pay leg or the receive leg */ private final boolean _payLeg; /** * Sets the value requirement name to {@link ValueRequirementNames#SWAP_PAY_LEG_DETAILS} or {@link ValueRequirementNames#SWAP_RECEIVE_LEG_DETAILS} * * @param payLeg True if the details to be returned are for the pay leg; false returns details for receive legs. */ public DiscountingSwapLegDetailFunction(final String payLeg) { this(Boolean.parseBoolean(payLeg)); } /** * Sets the value requirement name to {@link ValueRequirementNames#SWAP_PAY_LEG_DETAILS} or {@link ValueRequirementNames#SWAP_RECEIVE_LEG_DETAILS} * * @param payLeg True if the details to be returned are for the pay leg; false returns details for receive legs. */ public DiscountingSwapLegDetailFunction(final boolean payLeg) { super(payLeg ? SWAP_PAY_LEG_DETAILS : SWAP_RECEIVE_LEG_DETAILS); _payLeg = payLeg; } @Override public CompiledFunctionDefinition compile(final FunctionCompilationContext context, final Instant atInstant) { return new DiscountingCompiledFunction(getTargetToDefinitionConverter(context), getDefinitionToDerivativeConverter(context), false) { @Override public boolean canApplyTo(final FunctionCompilationContext compilationContext, final ComputationTarget target) { final Security security = target.getTrade().getSecurity(); return security instanceof SwapSecurity; } @SuppressWarnings("synthetic-access") @Override protected Set<ComputedValue> getValues(final FunctionExecutionContext executionContext, final FunctionInputs inputs, final ComputationTarget target, final Set<ValueRequirement> desiredValues, final InstrumentDerivative derivative, final FXMatrix fxMatrix) { final MulticurveProviderInterface data = getMergedProviders(inputs, fxMatrix); final SwapSecurity security = (SwapSecurity) target.getTrade().getSecurity(); final Clock snapshotClock = executionContext.getValuationClock(); final ZonedDateTime now = ZonedDateTime.now(snapshotClock); final SwapDefinition definition = (SwapDefinition) getDefinitionFromTarget(target); if (definition == null) { throw new OpenGammaRuntimeException("Definition for security " + security + " was null"); } @SuppressWarnings("unchecked") final Swap<? extends Payment, ? extends Payment> swap = (Swap<? extends Payment, ? extends Payment>) derivative; final AnnuityDefinition<? extends PaymentDefinition> legDefinition; final Annuity<? extends Payment> legDerivative; final boolean isFixed; if (_payLeg) { isFixed = security.getPayLeg() instanceof FixedInterestRateLeg; final boolean payFirstLeg = definition.getFirstLeg().getNthPayment(0).getReferenceAmount() < 0; legDefinition = payFirstLeg ? definition.getFirstLeg() : definition.getSecondLeg(); legDerivative = payFirstLeg ? swap.getFirstLeg() : swap.getSecondLeg(); } else { isFixed = security.getReceiveLeg() instanceof FixedInterestRateLeg; final boolean payFirstLeg = definition.getFirstLeg().getNthPayment(0).getReferenceAmount() < 0; legDefinition = payFirstLeg ? definition.getSecondLeg() : definition.getFirstLeg(); legDerivative = payFirstLeg ? swap.getSecondLeg() : swap.getFirstLeg(); } final ValueRequirement desiredValue = Iterables.getOnlyElement(desiredValues); final ValueSpecification spec = new ValueSpecification(getValueRequirementNames()[0], target.toSpecification(), desiredValue.getConstraints()); final CurrencyAmount[] notionals = legDefinition.accept(AnnuityNotionalsVisitor.getInstance(), now); final Pair<LocalDate[], LocalDate[]> accrualDates = legDefinition.accept(AnnuityAccrualDatesVisitor.getInstance(), now); final double[] paymentTimes = legDerivative.accept(AnnuityPaymentTimesVisitor.getInstance()); final double[] paymentFractions = legDerivative.accept(AnnuityPaymentFractionsVisitor.getInstance()); final CurrencyAmount[] paymentAmounts = legDerivative.accept(AnnuityPaymentAmountsVisitor.getInstance()); final Double[] fixedRates = legDerivative.accept(AnnuityFixedRatesVisitor.getInstance()); final double[] discountFactors = legDerivative.accept(AnnuityDiscountFactorsVisitor.getInstance(), data); if (isFixed) { final FixedSwapLegDetails details = new FixedSwapLegDetails(accrualDates.getFirst(), accrualDates.getSecond(), discountFactors, paymentTimes, paymentFractions, paymentAmounts, notionals, fixedRates); return Collections.singleton(new ComputedValue(spec, details)); } final Pair<LocalDate[], LocalDate[]> fixingDates = legDefinition.accept(AnnuityFixingDatesVisitor.getInstance(), now); final Double[] fixingYearFractions = legDefinition.accept(AnnuityFixingYearFractionsVisitor.getInstance(), now); final Double[] forwardRates = legDerivative.accept(AnnuityForwardRatesVisitor.getInstance(), data); final LocalDate[] paymentDates = legDefinition.accept(AnnuityPaymentDatesVisitor.getInstance(), now); final CurrencyAmount[] projectedAmounts = legDerivative.accept(AnnuityProjectedPaymentsVisitor.getInstance(), data); final double[] spreads = legDefinition.accept(AnnuitySpreadsVisitor.getInstance(), now); final double[] gearings = legDefinition.accept(AnnuityGearingsVisitor.getInstance(), now); final List<Set<Tenor>> indexTenors = legDefinition.accept(AnnuityIndexTenorsVisitor.getInstance(), now); final FloatingSwapLegDetails details = new FloatingSwapLegDetails(accrualDates.getFirst(), accrualDates.getSecond(), paymentFractions, fixingDates.getFirst(), fixingDates.getSecond(), fixingYearFractions, forwardRates, fixedRates, paymentDates, paymentTimes, discountFactors, paymentAmounts, projectedAmounts, notionals, spreads, gearings, indexTenors); return Collections.singleton(new ComputedValue(spec, details)); } }; } }