YAS.java

  1. package org.drip.sample.bloomberg;

  2. /*
  3.  * Credit Product imports
  4.  */

  5. import org.drip.analytics.cashflow.CompositePeriod;
  6. import org.drip.analytics.date.*;
  7. import org.drip.analytics.daycount.Convention;
  8. import org.drip.analytics.support.*;
  9. import org.drip.market.otc.*;
  10. import org.drip.numerical.common.FormatUtil;
  11. import org.drip.param.definition.*;
  12. import org.drip.param.valuation.*;
  13. import org.drip.product.definition.*;
  14. import org.drip.product.govvie.TreasuryComponent;
  15. import org.drip.product.params.EmbeddedOptionSchedule;

  16. /*
  17.  * Credit Analytics API imports
  18.  */

  19. import org.drip.product.rates.*;
  20. import org.drip.param.creator.*;
  21. import org.drip.param.market.*;
  22. import org.drip.param.quote.*;
  23. import org.drip.product.creator.*;
  24. import org.drip.product.credit.BondComponent;
  25. import org.drip.service.env.EnvManager;
  26. import org.drip.service.template.TreasuryBuilder;
  27. import org.drip.state.creator.*;
  28. import org.drip.state.discount.MergedDiscountForwardCurve;
  29. import org.drip.state.govvie.GovvieCurve;
  30. import org.drip.state.identifier.ForwardLabel;

  31. /*!
  32.  * Copyright (C) 2020 Lakshmi Krishnamurthy
  33.  * Copyright (C) 2019 Lakshmi Krishnamurthy
  34.  * Copyright (C) 2018 Lakshmi Krishnamurthy
  35.  * Copyright (C) 2017 Lakshmi Krishnamurthy
  36.  * Copyright (C) 2016 Lakshmi Krishnamurthy
  37.  * Copyright (C) 2015 Lakshmi Krishnamurthy
  38.  * Copyright (C) 2014 Lakshmi Krishnamurthy
  39.  * Copyright (C) 2013 Lakshmi Krishnamurthy
  40.  *
  41.  *  This file is part of DROP, an open-source library targeting analytics/risk, transaction cost analytics,
  42.  *      asset liability management analytics, capital, exposure, and margin analytics, valuation adjustment
  43.  *      analytics, and portfolio construction analytics within and across fixed income, credit, commodity,
  44.  *      equity, FX, and structured products. It also includes auxiliary libraries for algorithm support,
  45.  *      numerical analysis, numerical optimization, spline builder, model validation, statistical learning,
  46.  *      and computational support.
  47.  *  
  48.  *      https://lakshmidrip.github.io/DROP/
  49.  *  
  50.  *  DROP is composed of three modules:
  51.  *  
  52.  *  - DROP Product Core - https://lakshmidrip.github.io/DROP-Product-Core/
  53.  *  - DROP Portfolio Core - https://lakshmidrip.github.io/DROP-Portfolio-Core/
  54.  *  - DROP Computational Core - https://lakshmidrip.github.io/DROP-Computational-Core/
  55.  *
  56.  *  DROP Product Core implements libraries for the following:
  57.  *  - Fixed Income Analytics
  58.  *  - Loan Analytics
  59.  *  - Transaction Cost Analytics
  60.  *
  61.  *  DROP Portfolio Core implements libraries for the following:
  62.  *  - Asset Allocation Analytics
  63.  *  - Asset Liability Management Analytics
  64.  *  - Capital Estimation Analytics
  65.  *  - Exposure Analytics
  66.  *  - Margin Analytics
  67.  *  - XVA Analytics
  68.  *
  69.  *  DROP Computational Core implements libraries for the following:
  70.  *  - Algorithm Support
  71.  *  - Computation Support
  72.  *  - Function Analysis
  73.  *  - Model Validation
  74.  *  - Numerical Analysis
  75.  *  - Numerical Optimizer
  76.  *  - Spline Builder
  77.  *  - Statistical Learning
  78.  *
  79.  *  Documentation for DROP is Spread Over:
  80.  *
  81.  *  - Main                     => https://lakshmidrip.github.io/DROP/
  82.  *  - Wiki                     => https://github.com/lakshmiDRIP/DROP/wiki
  83.  *  - GitHub                   => https://github.com/lakshmiDRIP/DROP
  84.  *  - Repo Layout Taxonomy     => https://github.com/lakshmiDRIP/DROP/blob/master/Taxonomy.md
  85.  *  - Javadoc                  => https://lakshmidrip.github.io/DROP/Javadoc/index.html
  86.  *  - Technical Specifications => https://github.com/lakshmiDRIP/DROP/tree/master/Docs/Internal
  87.  *  - Release Versions         => https://lakshmidrip.github.io/DROP/version.html
  88.  *  - Community Credits        => https://lakshmidrip.github.io/DROP/credits.html
  89.  *  - Issues Catalog           => https://github.com/lakshmiDRIP/DROP/issues
  90.  *  - JUnit                    => https://lakshmidrip.github.io/DROP/junit/index.html
  91.  *  - Jacoco                   => https://lakshmidrip.github.io/DROP/jacoco/index.html
  92.  *
  93.  *  Licensed under the Apache License, Version 2.0 (the "License");
  94.  *      you may not use this file except in compliance with the License.
  95.  *  
  96.  *  You may obtain a copy of the License at
  97.  *      http://www.apache.org/licenses/LICENSE-2.0
  98.  *  
  99.  *  Unless required by applicable law or agreed to in writing, software
  100.  *      distributed under the License is distributed on an "AS IS" BASIS,
  101.  *      WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  102.  *  
  103.  *  See the License for the specific language governing permissions and
  104.  *      limitations under the License.
  105.  */

  106. /**
  107.  * <i>YAS</i> contains the sample demonstrating the replication of Bloomberg's YAS functionality.
  108.  *  
  109.  * <br><br>
  110.  *  <ul>
  111.  *      <li><b>Module </b> = <a href = "https://github.com/lakshmiDRIP/DROP/tree/master/ProductCore.md">Product Core Module</a></li>
  112.  *      <li><b>Library</b> = <a href = "https://github.com/lakshmiDRIP/DROP/tree/master/FixedIncomeAnalyticsLibrary.md">Fixed Income Analytics</a></li>
  113.  *      <li><b>Project</b> = <a href = "https://github.com/lakshmiDRIP/DROP/tree/master/src/main/java/org/drip/sample/README.md">DROP API Construction and Usage</a></li>
  114.  *      <li><b>Package</b> = <a href = "https://github.com/lakshmiDRIP/DROP/tree/master/src/main/java/org/drip/sample/bloomberg/README.md">Bloomberg CDSO CDSW SWPM YAS</a></li>
  115.  *  </ul>
  116.  * <br><br>
  117.  *
  118.  * @author Lakshmi Krishnamurthy
  119.  */

  120. public class YAS {
  121.     private static final String FIELD_SEPARATOR = "    ";

  122.     private static final FixFloatComponent OTCIRS (
  123.         final JulianDate dtSpot,
  124.         final String strCurrency,
  125.         final String strMaturityTenor,
  126.         final double dblCoupon)
  127.     {
  128.         FixedFloatSwapConvention ffConv = IBORFixedFloatContainer.ConventionFromJurisdiction (
  129.             strCurrency,
  130.             "ALL",
  131.             strMaturityTenor,
  132.             "MAIN"
  133.         );

  134.         return ffConv.createFixFloatComponent (
  135.             dtSpot,
  136.             strMaturityTenor,
  137.             dblCoupon,
  138.             0.,
  139.             1.
  140.         );
  141.     }

  142.     /*
  143.      * Sample demonstrating building of rates curve from cash/future/swaps
  144.      *
  145.      *      USE WITH CARE: This sample ignores errors and does not handle exceptions.
  146.      */

  147.     private static MergedDiscountForwardCurve BuildRatesCurveFromInstruments (
  148.         final JulianDate dtStart,
  149.         final String[] astrCashTenor,
  150.         final double[] adblCashRate,
  151.         final String[] astrIRSTenor,
  152.         final double[] adblIRSRate,
  153.         final double dblBump,
  154.         final String strCurrency)
  155.         throws Exception
  156.     {
  157.         int iNumDCInstruments = astrCashTenor.length + adblIRSRate.length;
  158.         int aiDate[] = new int[iNumDCInstruments];
  159.         double adblRate[] = new double[iNumDCInstruments];
  160.         String astrCalibMeasure[] = new String[iNumDCInstruments];
  161.         double adblCompCalibValue[] = new double[iNumDCInstruments];
  162.         CalibratableComponent aCompCalib[] = new CalibratableComponent[iNumDCInstruments];

  163.         // Cash Calibration

  164.         JulianDate dtCashEffective = dtStart.addBusDays (
  165.             1,
  166.             strCurrency
  167.         );

  168.         for (int i = 0; i < astrCashTenor.length; ++i) {
  169.             astrCalibMeasure[i] = "Rate";
  170.             adblRate[i] = java.lang.Double.NaN;
  171.             adblCompCalibValue[i] = adblCashRate[i] + dblBump;

  172.             aCompCalib[i] = SingleStreamComponentBuilder.Deposit (
  173.                 dtCashEffective,
  174.                 new JulianDate (aiDate[i] = dtCashEffective.addTenor (astrCashTenor[i]).julian()),
  175.                 ForwardLabel.Create (
  176.                     strCurrency,
  177.                     astrCashTenor[i]
  178.                 )
  179.             );
  180.         }

  181.         // IRS Calibration

  182.         JulianDate dtIRSEffective = dtStart.addBusDays (2, strCurrency);

  183.         for (int i = 0; i < astrIRSTenor.length; ++i) {
  184.             astrCalibMeasure[i + astrCashTenor.length] = "SwapRate";
  185.             adblRate[i + astrCashTenor.length] = java.lang.Double.NaN;
  186.             adblCompCalibValue[i + astrCashTenor.length] = adblIRSRate[i] + dblBump;

  187.             aiDate[i + astrCashTenor.length] = dtIRSEffective.addTenor (astrIRSTenor[i]).julian();

  188.             aCompCalib[i + astrCashTenor.length] = OTCIRS (
  189.                 dtIRSEffective,
  190.                 strCurrency,
  191.                 astrIRSTenor[i],
  192.                 0.
  193.             );
  194.         }

  195.         /*
  196.          * Build the IR curve from the components, their calibration measures, and their calibration quotes.
  197.          */

  198.         return ScenarioDiscountCurveBuilder.NonlinearBuild (
  199.             dtStart,
  200.             strCurrency,
  201.             aCompCalib,
  202.             adblCompCalibValue,
  203.             astrCalibMeasure,
  204.             null
  205.         );
  206.     }

  207.     /*
  208.      * Sample demonstrating creation of simple fixed coupon treasury bond
  209.      *
  210.      *      USE WITH CARE: This sample ignores errors and does not handle exceptions.
  211.      */

  212.     private static final TreasuryComponent Treasury (
  213.         final String strName,
  214.         final double dblCoupon,
  215.         final JulianDate dt,
  216.         final String strTenor)
  217.         throws Exception
  218.     {
  219.         return TreasuryBuilder.FromCode (
  220.             "UST",
  221.             dt,
  222.             dt.addTenor (strTenor),
  223.             dblCoupon
  224.         );
  225.     }

  226.     /*
  227.      * Sample demonstrating creation of a set of the on-the-run treasury bonds
  228.      *
  229.      *      USE WITH CARE: This sample ignores errors and does not handle exceptions.
  230.      */

  231.     private static final TreasuryComponent[] OTRTreasurySet (
  232.         final JulianDate dt,
  233.         final String[] astrTenor,
  234.         final double[] adblCoupon)
  235.         throws Exception
  236.     {
  237.         TreasuryComponent aTreasury[] = new TreasuryComponent[astrTenor.length];

  238.         for (int i = 0; i < astrTenor.length; ++i)
  239.             aTreasury[i] = Treasury (
  240.                 "TSY" + astrTenor[i] + "ON",
  241.                 adblCoupon[i],
  242.                 dt,
  243.                 astrTenor[i]
  244.             );

  245.         return aTreasury;
  246.     }

  247.     /*
  248.      * Sample demonstrating building of the treasury discount curve based off the on-the run instruments and their yields
  249.      *
  250.      *      USE WITH CARE: This sample ignores errors and does not handle exceptions.
  251.      */

  252.     private static final GovvieCurve BuildOnTheRunGovvieCurve (
  253.         final JulianDate dt,
  254.         final TreasuryComponent[] aTreasury,
  255.         final double[] adblYield)
  256.         throws Exception
  257.     {
  258.         return ScenarioGovvieCurveBuilder.CubicPolyShapePreserver (
  259.             "UST",
  260.             "UST",
  261.             aTreasury[0].currency(),
  262.             dt.julian(),
  263.             aTreasury,
  264.             adblYield,
  265.             "Yield"
  266.         );
  267.     }

  268.     /*
  269.      * Sample demonstrating creation of treasury quotes map
  270.      *
  271.      *      USE WITH CARE: This sample ignores errors and does not handle exceptions.
  272.      */

  273.     private static final CaseInsensitiveTreeMap<ProductQuote> MakeTSYQuotes (
  274.         final String[] astrTSYTenor,
  275.         final double[] adblTSYYield)
  276.         throws Exception
  277.     {
  278.         CaseInsensitiveTreeMap<ProductQuote> mTSYQuotes = new CaseInsensitiveTreeMap<ProductQuote>();

  279.         for (int i = 0; i < astrTSYTenor.length; ++i) {
  280.             ProductMultiMeasure cmmq = new ProductMultiMeasure();

  281.             cmmq.addQuote (
  282.                 "Yield",
  283.                 new MultiSided (
  284.                     "mid",
  285.                     adblTSYYield[i],
  286.                     Double.NaN
  287.                 ),
  288.                 true
  289.             );

  290.             mTSYQuotes.put (
  291.                 astrTSYTenor[i] + "ON",
  292.                 cmmq
  293.             );
  294.         }

  295.         return mTSYQuotes;
  296.     }

  297.     /*
  298.      * Sample demonstrating generation of all the YAS measures
  299.      *
  300.      *      USE WITH CARE: This sample ignores errors and does not handle exceptions.
  301.      */

  302.     private static final void BondPricerSample()
  303.         throws Exception
  304.     {
  305.         JulianDate dtCurve = DateUtil.Today();

  306.         JulianDate dtSettle = dtCurve.addBusDays (
  307.             3,
  308.             "USD"
  309.         );

  310.         double dblNotional = 1000000.;
  311.         String[] astrCashTenor = new String[] {"3M"};
  312.         double[] adblCashRate = new double[] {0.00276};
  313.         String[] astrIRSTenor = new String[] {   "1Y",    "2Y",    "3Y",    "4Y",    "5Y",    "6Y",    "7Y",
  314.                "8Y",    "9Y",   "10Y",   "11Y",   "12Y",   "15Y",   "20Y",   "25Y",   "30Y",   "40Y",   "50Y"};
  315.         double[] adblIRSRate = new double[]  {0.00367, 0.00533, 0.00843, 0.01238, 0.01609, 0.01926, 0.02191,
  316.             0.02406, 0.02588, 0.02741, 0.02870, 0.02982, 0.03208, 0.03372, 0.03445, 0.03484, 0.03501, 0.03484};
  317.         String[] astrTSYTenor = new String[] {
  318.             "1Y", "2Y", "3Y", "5Y", "7Y", "10Y", "30Y"
  319.         };
  320.         final double[] adblTSYCoupon = new double[] {
  321.             0.0000, 0.00375, 0.00500, 0.0100, 0.01375, 0.01375, 0.02875
  322.         };
  323.         double[] adblTSYYield = new double[] {
  324.             0.00160, 0.00397, 0.00696, 0.01421, 0.01955, 0.02529, 0.03568
  325.         };

  326.         MergedDiscountForwardCurve dc = BuildRatesCurveFromInstruments (
  327.             dtCurve,
  328.             astrCashTenor,
  329.             adblCashRate,
  330.             astrIRSTenor,
  331.             adblIRSRate,
  332.             0.,
  333.             "USD"
  334.         );

  335.         TreasuryComponent[] aTSYBond = OTRTreasurySet (
  336.             dtCurve,
  337.             astrTSYTenor,
  338.             adblTSYCoupon
  339.         );

  340.         /*
  341.          * Create the on-the-run treasury discount curve
  342.          */

  343.         GovvieCurve gc = BuildOnTheRunGovvieCurve (
  344.             dtCurve,
  345.             aTSYBond,
  346.             adblTSYYield
  347.         );

  348.         BondComponent bond = BondBuilder.CreateSimpleFixed (    // Simple Fixed Rate Bond
  349.             "TEST",         // Name
  350.             "USD",          // Currency
  351.             "",             // Empty Credit Curve
  352.             0.054,          // Bond Coupon
  353.             2,              // Frequency
  354.             "30/360",       // Day Count
  355.             DateUtil.CreateFromYMD (
  356.                 2011,
  357.                 4,
  358.                 21
  359.             ), // Effective
  360.             DateUtil.CreateFromYMD (
  361.                 2021,
  362.                 4,
  363.                 15
  364.             ),  // Maturity
  365.             null,       // Principal Schedule
  366.             null
  367.         );

  368.         int[] aiDate = new int[] {
  369.             DateUtil.CreateFromYMD (2016, 3, 1).julian(),
  370.             DateUtil.CreateFromYMD (2017, 3, 1).julian(),
  371.             DateUtil.CreateFromYMD (2018, 3, 1).julian(),
  372.             DateUtil.CreateFromYMD (2019, 3, 1).julian(),
  373.             DateUtil.CreateFromYMD (2020, 3, 1).julian()
  374.         };

  375.         double[] adblFactor = new double[] {
  376.             1.045, 1.03, 1.015, 1., 1.
  377.         };

  378.         EmbeddedOptionSchedule eos = new EmbeddedOptionSchedule (
  379.             aiDate,
  380.             adblFactor,
  381.             false,
  382.             30,
  383.             false,
  384.             Double.NaN,
  385.             "",
  386.             Double.NaN
  387.         );

  388.         bond.setEmbeddedCallSchedule (eos);

  389.         CurveSurfaceQuoteContainer mktParams = MarketParamsBuilder.Create (
  390.             dc,
  391.             gc,
  392.             null,
  393.             null,
  394.             null,
  395.             MakeTSYQuotes (
  396.                 astrTSYTenor,
  397.                 adblTSYYield
  398.             ),
  399.             null
  400.         );

  401.         System.out.println ("\n---- Valuation Details ----");

  402.         System.out.println ("Trade Date   : " + dtCurve);

  403.         System.out.println ("Cash Settle  : " + dtSettle);

  404.         System.out.println ("\n--------");

  405.         ValuationParams valParams = ValuationParams.Spot (
  406.             dtSettle,
  407.             0,
  408.             "",
  409.             Convention.DATE_ROLL_ACTUAL
  410.         );

  411.         double dblPrice = 0.97828;

  412.         double dblAccrued = bond.accrued (
  413.             valParams.valueDate(),
  414.             mktParams
  415.         );

  416.         WorkoutInfo wi = bond.exerciseYieldFromPrice (valParams, mktParams, null, dblPrice);

  417.         double dblTSYSpread = bond.tsySpreadFromPrice (valParams, mktParams, null, wi.date(), wi.factor(), dblPrice);

  418.         double dblGSpread = bond.gSpreadFromPrice (valParams, mktParams, null, wi.date(), wi.factor(), dblPrice);

  419.         double dblISpread = bond.iSpreadFromPrice (valParams, mktParams, null, wi.date(), wi.factor(), dblPrice);

  420.         double dblZSpread = bond.zSpreadFromPrice (valParams, mktParams, null, wi.date(), wi.factor(), dblPrice);

  421.         double dblASW = bond.aswFromPrice (valParams, mktParams, null, wi.date(), wi.factor(), dblPrice);

  422.         double dblOAS = bond.oasFromPrice (valParams, mktParams, null, wi.date(), wi.factor(), dblPrice);

  423.         double dblModDur = bond.modifiedDurationFromPrice (valParams, mktParams, null, wi.date(), wi.factor(), dblPrice);

  424.         double dblMacDur = bond.macaulayDurationFromPrice (valParams, mktParams, null, wi.date(), wi.factor(), dblPrice);

  425.         double dblYield01 = bond.yield01FromPrice (valParams, mktParams, null, wi.date(), wi.factor(), dblPrice);

  426.         double dblConvexity = bond.convexityFromPrice (valParams, mktParams, null, wi.date(), wi.factor(), dblPrice);

  427.         System.out.println ("Price          : " + FormatUtil.FormatDouble (dblPrice, 1, 3, 100.));

  428.         System.out.println ("Yield          : " + FormatUtil.FormatDouble (wi.yield(), 1, 3, 100.));

  429.         System.out.println ("Workout Date   : " + new JulianDate (wi.date()));

  430.         System.out.println ("Workout Factor : " + FormatUtil.FormatDouble (wi.factor(), 1, 2, 100.));

  431.         System.out.println ("\n--SPREAD AND YIELD CALCULATIONS--\n");

  432.         System.out.println ("TSY Spread : " + FormatUtil.FormatDouble (dblTSYSpread, 1, 0, 10000.));

  433.         System.out.println ("G Spread   : " + FormatUtil.FormatDouble (dblGSpread, 1, 0, 10000.));

  434.         System.out.println ("I Spread   : " + FormatUtil.FormatDouble (dblISpread, 1, 0, 10000.));

  435.         System.out.println ("Z Spread   : " + FormatUtil.FormatDouble (dblZSpread, 1, 0, 10000.));

  436.         System.out.println ("ASW        : " + FormatUtil.FormatDouble (dblASW, 1, 0, 10000.));

  437.         System.out.println ("OAS        : " + FormatUtil.FormatDouble (dblOAS, 1, 0, 10000.));

  438.         System.out.println ("\n--RISK--\n");

  439.         System.out.println ("Modified Duration : " + FormatUtil.FormatDouble (dblModDur, 1, 2, 10000.));

  440.         System.out.println ("Macaulay Duration : " + FormatUtil.FormatDouble (dblMacDur, 1, 2, 1.));

  441.         System.out.println ("Risk              : " + FormatUtil.FormatDouble (dblYield01 * 10000., 1, 2, 1.));

  442.         System.out.println ("Convexity         : " + FormatUtil.FormatDouble (dblConvexity, 1, 2, 1000000.));

  443.         System.out.println ("DV01              : " + FormatUtil.FormatDouble (dblYield01 * dblNotional, 1, 0, 1.));

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

  445.         System.out.println ("Face      : " + FormatUtil.FormatDouble (dblNotional, 1, 0, 1.));

  446.         System.out.println ("Principal : " + FormatUtil.FormatDouble (dblPrice * dblNotional, 1, 2, 1.));

  447.         System.out.println ("Accrued   : " + FormatUtil.FormatDouble (dblAccrued * dblNotional, 1, 2, 1.));

  448.         System.out.println ("Total     : " + FormatUtil.FormatDouble ((dblPrice + dblAccrued) * dblNotional, 1, 2, 1.));

  449.         System.out.println ("\nCashflow\n--------");

  450.         for (CompositePeriod p : bond.couponPeriods())
  451.             System.out.println (
  452.                 DateUtil.YYYYMMDD (p.startDate()) + FIELD_SEPARATOR +
  453.                 DateUtil.YYYYMMDD (p.endDate()) + FIELD_SEPARATOR +
  454.                 DateUtil.YYYYMMDD (p.payDate()) + FIELD_SEPARATOR +
  455.                 FormatUtil.FormatDouble (p.couponDCF(), 1, 4, 1.) + FIELD_SEPARATOR +
  456.                 FormatUtil.FormatDouble (dc.df (p.payDate()), 1, 4, 1.) + FIELD_SEPARATOR
  457.             );
  458.     }

  459.     public static final void main (
  460.         final String astrArgs[])
  461.         throws Exception
  462.     {
  463.         // String strConfig = "c:\\Lakshmi\\BondAnal\\Config.xml";

  464.         EnvManager.InitEnv (
  465.             "",
  466.             true
  467.         );

  468.         BondPricerSample();

  469.         EnvManager.TerminateEnv();
  470.     }
  471. }