CrossOvernightFloatingStream.java

  1. package org.drip.sample.ois;

  2. import java.util.*;

  3. import org.drip.analytics.date.JulianDate;
  4. import org.drip.analytics.daycount.Convention;
  5. import org.drip.analytics.support.*;
  6. import org.drip.function.r1tor1.*;
  7. import org.drip.market.definition.OvernightIndex;
  8. import org.drip.market.otc.*;
  9. import org.drip.numerical.common.*;
  10. import org.drip.param.creator.*;
  11. import org.drip.param.market.*;
  12. import org.drip.param.period.*;
  13. import org.drip.param.valuation.*;
  14. import org.drip.product.creator.*;
  15. import org.drip.product.definition.CalibratableComponent;
  16. import org.drip.product.rates.*;
  17. import org.drip.service.env.EnvManager;
  18. import org.drip.spline.basis.PolynomialFunctionSetParams;
  19. import org.drip.spline.params.*;
  20. import org.drip.spline.stretch.*;
  21. import org.drip.state.creator.*;
  22. import org.drip.state.discount.*;
  23. import org.drip.state.estimator.LatentStateStretchBuilder;
  24. import org.drip.state.identifier.*;
  25. import org.drip.state.inference.*;

  26. /*
  27.  * -*- mode: java; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
  28.  */

  29. /*!
  30.  * Copyright (C) 2018 Lakshmi Krishnamurthy
  31.  * Copyright (C) 2017 Lakshmi Krishnamurthy
  32.  * Copyright (C) 2016 Lakshmi Krishnamurthy
  33.  * Copyright (C) 2015 Lakshmi Krishnamurthy
  34.  * Copyright (C) 2014 Lakshmi Krishnamurthy
  35.  *
  36.  *  This file is part of DRIP, a free-software/open-source library for buy/side financial/trading model
  37.  *      libraries targeting analysts and developers
  38.  *      https://lakshmidrip.github.io/DRIP/
  39.  *  
  40.  *  DRIP is composed of four main libraries:
  41.  *  
  42.  *  - DRIP Fixed Income - https://lakshmidrip.github.io/DRIP-Fixed-Income/
  43.  *  - DRIP Asset Allocation - https://lakshmidrip.github.io/DRIP-Asset-Allocation/
  44.  *  - DRIP Numerical Optimizer - https://lakshmidrip.github.io/DRIP-Numerical-Optimizer/
  45.  *  - DRIP Statistical Learning - https://lakshmidrip.github.io/DRIP-Statistical-Learning/
  46.  *
  47.  *  - DRIP Fixed Income: Library for Instrument/Trading Conventions, Treasury Futures/Options,
  48.  *      Funding/Forward/Overnight Curves, Multi-Curve Construction/Valuation, Collateral Valuation and XVA
  49.  *      Metric Generation, Calibration and Hedge Attributions, Statistical Curve Construction, Bond RV
  50.  *      Metrics, Stochastic Evolution and Option Pricing, Interest Rate Dynamics and Option Pricing, LMM
  51.  *      Extensions/Calibrations/Greeks, Algorithmic Differentiation, and Asset Backed Models and Analytics.
  52.  *
  53.  *  - DRIP Asset Allocation: Library for model libraries for MPT framework, Black Litterman Strategy
  54.  *      Incorporator, Holdings Constraint, and Transaction Costs.
  55.  *
  56.  *  - DRIP Numerical Optimizer: Library for Numerical Optimization and Spline Functionality.
  57.  *
  58.  *  - DRIP Statistical Learning: Library for Statistical Evaluation and Machine Learning.
  59.  *
  60.  *  Licensed under the Apache License, Version 2.0 (the "License");
  61.  *      you may not use this file except in compliance with the License.
  62.  *  
  63.  *  You may obtain a copy of the License at
  64.  *      http://www.apache.org/licenses/LICENSE-2.0
  65.  *  
  66.  *  Unless required by applicable law or agreed to in writing, software
  67.  *      distributed under the License is distributed on an "AS IS" BASIS,
  68.  *      WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  69.  *  
  70.  *  See the License for the specific language governing permissions and
  71.  *      limitations under the License.
  72.  */

  73. /**
  74.  * CrossOvernightStream demonstrates the construction, customization, and valuation of Cross-Currency
  75.  *  Overnight Floating Streams.
  76.  *
  77.  * @author Lakshmi Krishnamurthy
  78.  */

  79. public class CrossOvernightFloatingStream {

  80.     private static final FixFloatComponent OTCOISFixFloat (
  81.         final JulianDate dtSpot,
  82.         final String strCurrency,
  83.         final String strMaturityTenor,
  84.         final double dblCoupon)
  85.     {
  86.         FixedFloatSwapConvention ffConv = OvernightFixedFloatContainer.FundConventionFromJurisdiction (
  87.             strCurrency
  88.         );

  89.         return ffConv.createFixFloatComponent (
  90.             dtSpot,
  91.             strMaturityTenor,
  92.             dblCoupon,
  93.             0.,
  94.             1.
  95.         );
  96.     }

  97.     /*
  98.      * Construct the Array of Deposit Instruments from the given set of parameters
  99.      *
  100.      *      USE WITH CARE: This sample ignores errors and does not handle exceptions.
  101.      */

  102.     private static final SingleStreamComponent[] DepositInstrumentsFromMaturityDays (
  103.         final JulianDate dtEffective,
  104.         final String strCurrency,
  105.         final int[] aiDay)
  106.         throws Exception
  107.     {
  108.         SingleStreamComponent[] aDeposit = new SingleStreamComponent[aiDay.length];

  109.         for (int i = 0; i < aiDay.length; ++i)
  110.             aDeposit[i] = SingleStreamComponentBuilder.Deposit (
  111.                 dtEffective,
  112.                 dtEffective.addBusDays (
  113.                     aiDay[i],
  114.                     strCurrency
  115.                 ),
  116.                 OvernightLabel.Create (
  117.                     strCurrency
  118.                 )
  119.             );

  120.         return aDeposit;
  121.     }

  122.     /*
  123.      * Construct the Array of Swap Instruments from the given set of parameters
  124.      *
  125.      *      USE WITH CARE: This sample ignores errors and does not handle exceptions.
  126.      */

  127.     private static final FixFloatComponent[] OISFromMaturityTenor (
  128.         final JulianDate dtSpot,
  129.         final String strCurrency,
  130.         final String[] astrMaturityTenor,
  131.         final double[] adblCoupon)
  132.         throws Exception
  133.     {
  134.         FixFloatComponent[] aOIS = new FixFloatComponent[astrMaturityTenor.length];

  135.         for (int i = 0; i < astrMaturityTenor.length; ++i)
  136.             aOIS[i] = OTCOISFixFloat (
  137.                 dtSpot,
  138.                 strCurrency,
  139.                 astrMaturityTenor[i],
  140.                 adblCoupon[i]
  141.             );

  142.         return aOIS;
  143.     }

  144.     private static final MergedDiscountForwardCurve CustomOISCurveBuilderSample (
  145.         final JulianDate dtSpot,
  146.         final String strCurrency)
  147.         throws Exception
  148.     {
  149.         /*
  150.          * Construct the Array of Deposit Instruments and their Quotes from the given set of parameters
  151.          */

  152.         SingleStreamComponent[] aDepositComp = DepositInstrumentsFromMaturityDays (
  153.             dtSpot,
  154.             strCurrency,
  155.             new int[] {
  156.                 1, 2, 3, 7, 14, 21, 30, 60
  157.             }
  158.         );

  159.         double[] adblDepositQuote = new double[] {
  160.             0.01200, 0.01200, 0.01200, 0.01450, 0.01550, 0.01600, 0.01660, 0.01850, // Cash
  161.         };

  162.         /*
  163.          * Construct the Deposit Instrument Set Stretch Builder
  164.          */

  165.         LatentStateStretchSpec depositStretch = LatentStateStretchBuilder.ForwardFundingStretchSpec (
  166.             "DEPOSIT",
  167.             aDepositComp,
  168.             "ForwardRate",
  169.             adblDepositQuote
  170.         );

  171.         /*
  172.          * Construct the Array of EDF Instruments and their Quotes from the given set of parameters
  173.          */

  174.         SingleStreamComponent[] aEDFComp = SingleStreamComponentBuilder.ForwardRateFuturesPack (
  175.             dtSpot,
  176.             4,
  177.             strCurrency
  178.         );

  179.         double[] adblEDFQuote = new double[] {
  180.             0.01612, 0.01580, 0.01589, 0.01598
  181.         };

  182.         /*
  183.          * Construct the Cash Instrument Set Stretch Builder
  184.          */

  185.         LatentStateStretchSpec edfStretch = LatentStateStretchBuilder.ForwardFundingStretchSpec (
  186.             "EDF",
  187.             aEDFComp,
  188.             "ForwardRate",
  189.             adblEDFQuote
  190.         );

  191.         /*
  192.          * Construct the Array of OIS Instruments and their Quotes from the given set of parameters
  193.          */

  194.         double[] adblOISQuote = new double[] {
  195.             0.02604,    //  4Y
  196.             0.02808,    //  5Y
  197.             0.02983,    //  6Y
  198.             0.03136,    //  7Y
  199.             0.03268,    //  8Y
  200.             0.03383,    //  9Y
  201.             0.03488     // 10Y
  202.         };

  203.         CalibratableComponent[] aOISComp = OISFromMaturityTenor (
  204.             dtSpot,
  205.             strCurrency,
  206.             new java.lang.String[] {
  207.                 "4Y", "5Y", "6Y", "7Y", "8Y", "9Y", "10Y"
  208.             },
  209.             adblOISQuote
  210.         );

  211.         /*
  212.          * Construct the OIS Instrument Set Stretch Builder
  213.          */

  214.         LatentStateStretchSpec oisStretch = LatentStateStretchBuilder.ForwardFundingStretchSpec (
  215.             "SWAP",
  216.             aOISComp,
  217.             "SwapRate",
  218.             adblOISQuote
  219.         );

  220.         LatentStateStretchSpec[] aStretchSpec = new LatentStateStretchSpec[] {
  221.             depositStretch,
  222.             edfStretch,
  223.             oisStretch
  224.         };

  225.         /*
  226.          * Set up the Linear Curve Calibrator using the following parameters:
  227.          *  - Cubic Exponential Mixture Basis Spline Set
  228.          *  - Ck = 2, Segment Curvature Penalty = 2
  229.          *  - Quadratic Rational Shape Controller
  230.          *  - Natural Boundary Setting
  231.          */

  232.         LinearLatentStateCalibrator lcc = new LinearLatentStateCalibrator (
  233.             new SegmentCustomBuilderControl (
  234.                 MultiSegmentSequenceBuilder.BASIS_SPLINE_POLYNOMIAL,
  235.                 new PolynomialFunctionSetParams (4),
  236.                 SegmentInelasticDesignControl.Create (
  237.                     2,
  238.                     2
  239.                 ),
  240.                 new ResponseScalingShapeControl (
  241.                     true,
  242.                     new QuadraticRationalShapeControl (0.)
  243.                 ),
  244.                 null
  245.             ),
  246.             BoundarySettings.NaturalStandard(),
  247.             MultiSegmentSequence.CALIBRATE,
  248.             null,
  249.             null
  250.         );

  251.         /*
  252.          * Construct the Shape Preserving Discount Curve by applying the linear curve calibrator to the array
  253.          *  of Cash and Swap Stretches.
  254.          */

  255.         return ScenarioDiscountCurveBuilder.ShapePreservingDFBuild (
  256.             strCurrency,
  257.             lcc,
  258.             aStretchSpec,
  259.             new ValuationParams (
  260.                 dtSpot,
  261.                 dtSpot,
  262.                 strCurrency
  263.             ),
  264.             null,
  265.             null,
  266.             null,
  267.             1.
  268.         );
  269.     }

  270.     private static final LatentStateFixingsContainer SetFlatOvernightFixings (
  271.         final JulianDate dtStart,
  272.         final JulianDate dtEnd,
  273.         final JulianDate dtValue,
  274.         final ForwardLabel fri,
  275.         final double dblFlatFixing,
  276.         final double dblNotional)
  277.         throws Exception
  278.     {
  279.         LatentStateFixingsContainer lsfc = new LatentStateFixingsContainer();

  280.         double dblAccount = 1.;

  281.         int iPrevDate = dtStart.julian();

  282.         JulianDate dt = dtStart.addDays (1);

  283.         while (dt.julian() <= dtEnd.julian()) {
  284.             lsfc.add (
  285.                 dt,
  286.                 fri,
  287.                 dblFlatFixing
  288.             );

  289.             if (dt.julian() <= dtValue.julian()) {
  290.                 double dblAccrualFraction = Convention.YearFraction (
  291.                     iPrevDate,
  292.                     dt.julian(),
  293.                     "Act/360",
  294.                     false,
  295.                     null,
  296.                     "USD"
  297.                 );

  298.                 dblAccount *= (1. + dblFlatFixing * dblAccrualFraction);
  299.             }

  300.             iPrevDate = dt.julian();

  301.             dt = dt.addBusDays (
  302.                 1,
  303.                 "USD"
  304.             );
  305.         }

  306.         System.out.println ("\tManual Calc Float Accrued (Geometric Compounding): " + (dblAccount - 1.) * dblNotional);

  307.         System.out.println ("\tManual Calc Float Accrued (Arithmetic Compounding): " +
  308.             ((dtValue.julian() - dtStart.julian()) * dblNotional * dblFlatFixing / 360.));

  309.         return lsfc;
  310.     }

  311.     public static final Map<String, Double> CompoundingRun (
  312.         final ForwardLabel fri)
  313.         throws Exception
  314.     {
  315.         double dblOISVol = 0.3;
  316.         double dblUSDFundingVol = 0.3;
  317.         double dblUSDFundingUSDOISCorrelation = 0.3;

  318.         String strCurrency = fri.currency();

  319.         JulianDate dtToday = org.drip.analytics.date.DateUtil.Today().addTenorAndAdjust (
  320.             "0D",
  321.             strCurrency
  322.         );

  323.         MergedDiscountForwardCurve dc = CustomOISCurveBuilderSample (
  324.             dtToday,
  325.             strCurrency
  326.         );

  327.         JulianDate dtCustomOISStart = dtToday.subtractTenor ("2M");

  328.         JulianDate dtCustomOISMaturity = dtToday.addTenor ("4M");

  329.         CompositePeriodSetting cpsFloating = new CompositePeriodSetting (
  330.             360,
  331.             "ON",
  332.             strCurrency,
  333.             null,
  334.             -1.,
  335.             null,
  336.             null,
  337.             null,
  338.             null
  339.         );

  340.         ComposableFloatingUnitSetting cfusFloating = new ComposableFloatingUnitSetting (
  341.             "ON",
  342.             CompositePeriodBuilder.EDGE_DATE_SEQUENCE_OVERNIGHT,
  343.             null,
  344.             fri,
  345.             CompositePeriodBuilder.REFERENCE_PERIOD_IN_ADVANCE,
  346.             0.
  347.         );

  348.         List<Integer> lsFloatingStreamEdgeDate = CompositePeriodBuilder.OvernightEdgeDates (
  349.             dtCustomOISStart,
  350.             dtCustomOISStart.addTenorAndAdjust (
  351.                 "6M",
  352.                 strCurrency
  353.             ),
  354.             strCurrency
  355.         );

  356.         Stream floatStream = new Stream (
  357.             CompositePeriodBuilder.FloatingCompositeUnit (
  358.                 lsFloatingStreamEdgeDate,
  359.                 cpsFloating,
  360.                 cfusFloating
  361.             )
  362.         );

  363.         CurveSurfaceQuoteContainer mktParams = MarketParamsBuilder.Create (
  364.             dc,
  365.             null,
  366.             null,
  367.             null,
  368.             null,
  369.             null,
  370.             SetFlatOvernightFixings (
  371.                 dtCustomOISStart,
  372.                 dtCustomOISMaturity,
  373.                 dtToday,
  374.                 fri,
  375.                 0.003,
  376.                 -1.
  377.             )
  378.         );

  379.         ValuationParams valParams = new ValuationParams (
  380.             dtToday,
  381.             dtToday,
  382.             strCurrency
  383.         );

  384.         FundingLabel fundingLabelUSD = FundingLabel.Standard ("USD");

  385.         mktParams.setForwardVolatility (
  386.             ScenarioDeterministicVolatilityBuilder.FlatForward (
  387.                 dtToday.julian(),
  388.                 VolatilityLabel.Standard (fri),
  389.                 fri.currency(),
  390.                 dblOISVol
  391.             )
  392.         );

  393.         mktParams.setFundingVolatility (
  394.             ScenarioDeterministicVolatilityBuilder.FlatForward (
  395.                 dtToday.julian(),
  396.                 VolatilityLabel.Standard (fundingLabelUSD),
  397.                 "USD",
  398.                 dblUSDFundingVol
  399.             )
  400.         );

  401.         mktParams.setForwardFundingCorrelation (
  402.             fri,
  403.             fundingLabelUSD,
  404.             new FlatUnivariate (dblUSDFundingUSDOISCorrelation)
  405.         );

  406.         return floatStream.value (
  407.             valParams,
  408.             null,
  409.             mktParams,
  410.             null
  411.         );
  412.     }

  413.     public static final void main (
  414.         final String[] astrArgs)
  415.         throws Exception
  416.     {
  417.         /*
  418.          * Initialize the Credit Analytics Library
  419.          */

  420.         EnvManager.InitEnv ("");

  421.         String strCurrency = "USD";

  422.         Map<String, Double> mapArithmeticOutput = CompoundingRun (
  423.             OvernightLabel.Create (
  424.                 strCurrency
  425.             )
  426.         );

  427.         Map<String, Double> mapGeometricOutput = CompoundingRun (
  428.             ForwardLabel.Create (
  429.                 new OvernightIndex (
  430.                     strCurrency + "OIS",
  431.                     "OIS",
  432.                     strCurrency,
  433.                     "Act/360",
  434.                     strCurrency,
  435.                     "ON",
  436.                     0,
  437.                     CompositePeriodBuilder.ACCRUAL_COMPOUNDING_RULE_GEOMETRIC
  438.                 ),
  439.                 "ON"
  440.             )
  441.         );

  442.         System.out.println ("\n\t-----------------------------------");

  443.         System.out.println ("\t  GEOMETRIC |  ARITHMETIC | CHECK");

  444.         System.out.println ("\t-----------------------------------\n");

  445.         for (Map.Entry<String, Double> meGeometric : mapGeometricOutput.entrySet()) {
  446.             String strKey = meGeometric.getKey();

  447.             double dblGeometricMeasure = meGeometric.getValue();

  448.             double dblArithmeticMeasure = mapArithmeticOutput.get (strKey);

  449.             String strMatch = NumberUtil.WithinTolerance (
  450.                 dblGeometricMeasure,
  451.                 dblArithmeticMeasure,
  452.                 1.e-08,
  453.                 1.e-04
  454.             ) ?
  455.             "MATCH " :
  456.             "DIFFER";

  457.             System.out.println ("\t" +
  458.                 FormatUtil.FormatDouble (dblGeometricMeasure, 1, 8, 1.) + " | " +
  459.                 FormatUtil.FormatDouble (dblArithmeticMeasure, 1, 8, 1.) + " | " +
  460.                 strMatch + " <= " + strKey
  461.             );
  462.         }
  463.     }
  464. }