CoreCashFlowMeasures.java

  1. package org.drip.sample.bond;

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

  5. import org.drip.analytics.cashflow.*;
  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.*;
  11. import org.drip.param.definition.*;
  12. import org.drip.param.market.CurveSurfaceQuoteContainer;
  13. import org.drip.param.pricer.CreditPricerParams;
  14. import org.drip.param.quote.*;
  15. import org.drip.param.valuation.*;
  16. import org.drip.product.params.*;
  17. import org.drip.product.rates.*;
  18. import org.drip.product.definition.*;
  19. import org.drip.product.govvie.TreasuryComponent;
  20. import org.drip.param.creator.*;
  21. import org.drip.product.creator.*;
  22. import org.drip.service.env.EnvManager;
  23. import org.drip.service.template.TreasuryBuilder;
  24. import org.drip.state.creator.*;
  25. import org.drip.state.credit.CreditCurve;
  26. import org.drip.state.discount.MergedDiscountForwardCurve;
  27. import org.drip.state.govvie.GovvieCurve;
  28. import org.drip.state.identifier.ForwardLabel;

  29. /*
  30.  * -*- mode: java; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
  31.  */

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

  108. /**
  109.  * <i>CoreCashFlowMeasures</i> contains a demo of the Bond Core Measures and the Cash Flow Sample. It
  110.  * generates the Core and the RV measures for essentially the same bond (with identical cash flows)
  111.  * constructed in three different ways:
  112.  *
  113.  * <br><br>
  114.  *  <ul>
  115.  *      <li>
  116.  *          As a fixed rate bond.
  117.  *      </li>
  118.  *      <li>
  119.  *          As a floater.
  120.  *      </li>
  121.  *      <li>
  122.  *          As a bond constructed from a set of custom coupon and principal flows.
  123.  *      </li>
  124.  *  </ul>
  125.  * <br><br>
  126.  *
  127.  * It shows these measures reconcile where they should.
  128.  *  
  129.  * <br><br>
  130.  *  <ul>
  131.  *      <li><b>Module </b> = <a href = "https://github.com/lakshmiDRIP/DROP/tree/master/ProductCore.md">Product Core Module</a></li>
  132.  *      <li><b>Library</b> = <a href = "https://github.com/lakshmiDRIP/DROP/tree/master/FixedIncomeAnalyticsLibrary.md">Fixed Income Analytics</a></li>
  133.  *      <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>
  134.  *      <li><b>Package</b> = <a href = "https://github.com/lakshmiDRIP/DROP/tree/master/src/main/java/org/drip/sample/bond/README.md">Bullet, EOS Bond Metrics + Curve</a></li>
  135.  *  </ul>
  136.  * <br><br>
  137.  *
  138.  * @author Lakshmi Krishnamurthy
  139.  */

  140. public class CoreCashFlowMeasures {
  141.     private static final String FIELD_SEPARATOR = "    ";

  142.     private static final FixFloatComponent OTCIRS (
  143.         final JulianDate dtSpot,
  144.         final String strCurrency,
  145.         final String strMaturityTenor,
  146.         final double dblCoupon)
  147.     {
  148.         FixedFloatSwapConvention ffConv = IBORFixedFloatContainer.ConventionFromJurisdiction (
  149.             strCurrency,
  150.             "ALL",
  151.             strMaturityTenor,
  152.             "MAIN"
  153.         );

  154.         return ffConv.createFixFloatComponent (
  155.             dtSpot,
  156.             strMaturityTenor,
  157.             dblCoupon,
  158.             0.,
  159.             1.
  160.         );
  161.     }

  162.     /*
  163.      * Sample demonstrating building of rates curve from cash/future/swaps
  164.      *
  165.      *      USE WITH CARE: This sample ignores errors and does not handle exceptions.
  166.      */

  167.     private static MergedDiscountForwardCurve BuildRatesCurveFromInstruments (
  168.         final JulianDate dtStart,
  169.         final String[] astrCashTenor,
  170.         final double[] adblCashRate,
  171.         final String[] astrIRSTenor,
  172.         final double[] adblIRSRate,
  173.         final double dblBump,
  174.         final String strCurrency)
  175.         throws Exception
  176.     {
  177.         int iNumDCInstruments = astrCashTenor.length + adblIRSRate.length;
  178.         int aiDate[] = new int[iNumDCInstruments];
  179.         double adblRate[] = new double[iNumDCInstruments];
  180.         String astrCalibMeasure[] = new String[iNumDCInstruments];
  181.         double adblCompCalibValue[] = new double[iNumDCInstruments];
  182.         CalibratableComponent aCompCalib[] = new CalibratableComponent[iNumDCInstruments];

  183.         // Cash Calibration

  184.         JulianDate dtCashEffective = dtStart.addBusDays (1, strCurrency);

  185.         for (int i = 0; i < astrCashTenor.length; ++i) {
  186.             astrCalibMeasure[i] = "Rate";
  187.             adblRate[i] = java.lang.Double.NaN;
  188.             adblCompCalibValue[i] = adblCashRate[i] + dblBump;

  189.             aCompCalib[i] = SingleStreamComponentBuilder.Deposit (
  190.                 dtCashEffective,
  191.                 new JulianDate (aiDate[i] = dtCashEffective.addTenor (astrCashTenor[i]).julian()),
  192.                 ForwardLabel.Create (
  193.                     strCurrency,
  194.                     astrCashTenor[i]
  195.                 )
  196.             );
  197.         }

  198.         // IRS Calibration

  199.         JulianDate dtIRSEffective = dtStart.addBusDays (
  200.             2,
  201.             strCurrency
  202.         );

  203.         for (int i = 0; i < astrIRSTenor.length; ++i) {
  204.             astrCalibMeasure[i + astrCashTenor.length] = "SwapRate";
  205.             adblRate[i + astrCashTenor.length] = java.lang.Double.NaN;
  206.             adblCompCalibValue[i + astrCashTenor.length] = adblIRSRate[i] + dblBump;

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

  208.             aCompCalib[i + astrCashTenor.length] = OTCIRS (
  209.                 dtIRSEffective,
  210.                 strCurrency,
  211.                 astrIRSTenor[i],
  212.                 0.
  213.             );
  214.         }

  215.         /*
  216.          * Build the IR curve from the components, their calibration measures, and their calibration quotes.
  217.          */

  218.         return ScenarioDiscountCurveBuilder.NonlinearBuild (
  219.             dtStart,
  220.             strCurrency,
  221.             aCompCalib,
  222.             adblCompCalibValue,
  223.             astrCalibMeasure,
  224.             null
  225.         );
  226.     }

  227.     /*
  228.      * Sample demonstrating creation of simple fixed coupon treasury bond
  229.      *
  230.      *      USE WITH CARE: This sample ignores errors and does not handle exceptions.
  231.      */

  232.     private static final TreasuryComponent Treasury (
  233.         final String strName,
  234.         final double dblCoupon,
  235.         final JulianDate dt,
  236.         final String strTenor)
  237.         throws Exception
  238.     {
  239.         return TreasuryBuilder.FromCode (
  240.             "UST",
  241.             dt,
  242.             dt.addTenor (strTenor),
  243.             dblCoupon
  244.         );
  245.     }

  246.     /*
  247.      * Sample demonstrating creation of a set of the on-the-run treasury bonds
  248.      *
  249.      *      USE WITH CARE: This sample ignores errors and does not handle exceptions.
  250.      */

  251.     private static final TreasuryComponent[] OTRTreasurySet (
  252.         final JulianDate dt,
  253.         final String[] astrTenor,
  254.         final double[] adblCoupon)
  255.         throws Exception
  256.     {
  257.         TreasuryComponent aTreasury[] = new TreasuryComponent[astrTenor.length];

  258.         for (int i = 0; i < astrTenor.length; ++i)
  259.             aTreasury[i] = Treasury (
  260.                 "TSY" + astrTenor[i] + "ON",
  261.                 adblCoupon[i],
  262.                 dt,
  263.                 astrTenor[i]
  264.             );

  265.         return aTreasury;
  266.     }

  267.     private static final void AccumulateBondMarketQuote (
  268.         final CurveSurfaceQuoteContainer csqc,
  269.         final String[] astrOnTheRunCode,
  270.         final double[] adblYield)
  271.         throws Exception
  272.     {
  273.         for (int i = 0; i < astrOnTheRunCode.length; ++i) {
  274.             ProductMultiMeasure pmmq = new ProductMultiMeasure();

  275.             pmmq.addQuote (
  276.                 "Yield",
  277.                 new MultiSided (
  278.                     "mid",
  279.                     adblYield[i]
  280.                 ),
  281.                 true
  282.             );

  283.             csqc.setProductQuote (
  284.                 astrOnTheRunCode[i],
  285.                 pmmq
  286.             );
  287.         }
  288.     }

  289.     /*
  290.      * Sample demonstrating building of the treasury discount curve based off the on-the run instruments and their yields
  291.      *
  292.      *      USE WITH CARE: This sample ignores errors and does not handle exceptions.
  293.      */

  294.     private static final GovvieCurve BuildOnTheRunGovvieCurve (
  295.         final JulianDate dt,
  296.         final TreasuryComponent[] aTreasury,
  297.         final double[] adblYield)
  298.         throws Exception
  299.     {
  300.         return ScenarioGovvieCurveBuilder.CubicPolyShapePreserver (
  301.             "UST",
  302.             "UST",
  303.             aTreasury[0].currency(),
  304.             dt.julian(),
  305.             aTreasury,
  306.             adblYield,
  307.             "Yield"
  308.         );
  309.     }

  310.     /*
  311.      * Sample demonstrating creation of discount curve
  312.      *
  313.      *      USE WITH CARE: This sample ignores errors and does not handle exceptions.
  314.      */

  315.     private static final MergedDiscountForwardCurve MakeDiscountCurve (
  316.         final JulianDate dtCurve)
  317.         throws Exception
  318.     {
  319.         String[] astrCashTenor = new String[] {};
  320.         double[] adblCashRate = new double[] {};
  321.         String[] astrIRSTenor = new String[] {   "1Y",    "2Y",    "3Y",    "4Y",    "5Y",    "6Y",    "7Y",
  322.                "8Y",    "9Y",   "10Y",   "11Y",   "12Y",   "15Y",   "20Y",   "25Y",   "30Y",   "40Y",   "50Y"};
  323.         double[] adblIRSRate = new double[]  {0.00367, 0.00533, 0.00843, 0.01238, 0.01609, 0.01926, 0.02191,
  324.             0.02406, 0.02588, 0.02741, 0.02870, 0.02982, 0.03208, 0.03372, 0.03445, 0.03484, 0.03501, 0.03484};

  325.         return BuildRatesCurveFromInstruments (
  326.             dtCurve,
  327.             astrCashTenor,
  328.             adblCashRate,
  329.             astrIRSTenor,
  330.             adblIRSRate,
  331.             0.,
  332.             "USD"
  333.         );
  334.     }

  335.     /*
  336.      * Sample demonstrating creation of the principal factor schedule from date and factor array
  337.      *
  338.      *      USE WITH CARE: This sample ignores errors and does not handle exceptions.
  339.      */

  340.     private static final Array2D MakeFSPrincipal()
  341.         throws Exception
  342.     {
  343.         double[] aiDate = new double[5];
  344.         double[] adblFactor = new double[] {1., 1.0, 1.0, 1.0, 1.0};
  345.         // double[] adblFactor = new double[] {1., 0.9, 0.8, 0.7, 0.6};

  346.         JulianDate dtEOSStart = DateUtil.CreateFromYMD (
  347.             2018,
  348.             9,
  349.             11
  350.         ).addDays (2);

  351.         for (int i = 0; i < 5; ++i)
  352.             aiDate[i] = dtEOSStart.addYears (i + 2).julian();

  353.         return Array2D.FromArray (
  354.             aiDate,
  355.             adblFactor
  356.         );
  357.     }

  358.     /*
  359.      * Sample demonstrating creation of the coupon factor schedule from date and factor array
  360.      *
  361.      *      USE WITH CARE: This sample ignores errors and does not handle exceptions.
  362.      */

  363.     private static final Array2D MakeFSCoupon()
  364.         throws Exception
  365.     {
  366.         double[] aiDate = new double[5];
  367.         double[] adblFactor = new double[] {1., 1.0, 1.0, 1.0, 1.0};
  368.         // double[] adblFactor = new double[] {1., 0.9, 0.8, 0.7, 0.6};

  369.         JulianDate dtEOSStart = DateUtil.CreateFromYMD (
  370.             2018,
  371.             9,
  372.             11
  373.         ).addDays (2);

  374.         for (int i = 0; i < 5; ++i)
  375.             aiDate[i] = dtEOSStart.addYears (i + 2).julian();

  376.         return Array2D.FromArray (
  377.             aiDate,
  378.             adblFactor
  379.         );
  380.     }

  381.     /*
  382.      * Sample creates a custom named bond from the bond type and parameters
  383.      *
  384.      *      USE WITH CARE: This sample ignores errors and does not handle exceptions.
  385.      */

  386.     private static final Bond CreateCustomBond (
  387.         final String strName,
  388.         final String strCreditCurve,
  389.         final int iBondType)
  390.         throws Exception
  391.     {
  392.         BondProduct bond = null;
  393.         boolean bEOSOn = false;
  394.         boolean bEOSAmerican = false;

  395.         if (BondBuilder.BOND_TYPE_SIMPLE_FLOATER == iBondType)
  396.             bond = BondBuilder.CreateSimpleFloater ( // Simple Floating Rate Bond
  397.                 strName,        // Name
  398.                 "USD",          // Currency
  399.                 "USD-6M", // Rate Index
  400.                 strCreditCurve, // Credit Curve
  401.                 0.01,           // Floating Spread
  402.                 2,              // Coupon Frequency
  403.                 "30/360",       // Day Count
  404.                 DateUtil.CreateFromYMD (
  405.                     2008,
  406.                     9,
  407.                     21
  408.                 ), // Effective
  409.                 DateUtil.CreateFromYMD (
  410.                     2023,
  411.                     9,
  412.                     20
  413.                 ),  // Maturity
  414.                 MakeFSPrincipal(),      // Principal Schedule
  415.                 MakeFSCoupon()      // Coupon Schedule
  416.             );
  417.         else if (BondBuilder.BOND_TYPE_SIMPLE_FIXED == iBondType)
  418.             bond = BondBuilder.CreateSimpleFixed (  // Simple Fixed Rate Bond
  419.                 strName,        // Name
  420.                 "USD",          // Currency
  421.                 strCreditCurve, // Credit Curve
  422.                 0.05,           // Bond Coupon
  423.                 2,              // Coupon Frequency
  424.                 "30/360",       // Day Count
  425.                 DateUtil.CreateFromYMD (
  426.                     2008,
  427.                     9,
  428.                     21
  429.                 ), // Effective
  430.                 DateUtil.CreateFromYMD (
  431.                     2023,
  432.                     9,
  433.                     20
  434.                 ),  // Maturity
  435.                 MakeFSPrincipal(),      // Principal Schedule
  436.                 MakeFSCoupon()      // Coupon Schedule
  437.             );
  438.         else if (BondBuilder.BOND_TYPE_SIMPLE_FROM_CF == iBondType) {   // Bond from custom coupon and principal flows
  439.             final int NUM_CF_ENTRIES = 30;
  440.             double[] adblCouponAmount = new double[NUM_CF_ENTRIES];
  441.             double[] adblPrincipalAmount = new double[NUM_CF_ENTRIES];
  442.             JulianDate[] adt = new JulianDate[NUM_CF_ENTRIES];

  443.             JulianDate dtEffective = DateUtil.CreateFromYMD (
  444.                 2008,
  445.                 9,
  446.                 20
  447.             );

  448.             for (int i = 0; i < NUM_CF_ENTRIES; ++i) {
  449.                 adt[i] = dtEffective.addMonths (6 * (i + 1));

  450.                 adblCouponAmount[i] = 0.05;
  451.                 adblPrincipalAmount[i] = 1.0;
  452.             }

  453.             bond = BondBuilder.CreateBondFromCF (
  454.                 strName,                // Name
  455.                 dtEffective,            // Effective
  456.                 "USD",                  // Currency
  457.                 strCreditCurve,         // Credit Curve
  458.                 "30/360",               // Day Count
  459.                 1.,                     // Initial Notional
  460.                 0.05,                   // Coupon Rate
  461.                 2,                      // Frequency
  462.                 adt,                    // Array of dates
  463.                 adblCouponAmount,       // Array of coupon amount
  464.                 adblPrincipalAmount,    // Array of principal amount
  465.                 false                   // Principal is an outstanding notional
  466.             );
  467.         }

  468.         /*
  469.          * Bonds with options embedded
  470.          */

  471.         if (bEOSOn) {
  472.             int[] aiDate = new int[5];
  473.             double[] adblPutFactor = new double[5];
  474.             double[] adblCallFactor = new double[5];
  475.             EmbeddedOptionSchedule eosPut = null;
  476.             EmbeddedOptionSchedule eosCall = null;

  477.             JulianDate dtEOSStart = DateUtil.CreateFromYMD (
  478.                 2018,
  479.                 9,
  480.                 11
  481.             ).addDays (2);

  482.             for (int i = 0; i < 5; ++i) {
  483.                 adblPutFactor[i] = 0.9;
  484.                 adblCallFactor[i] = 1.0;

  485.                 aiDate[i] = dtEOSStart.addYears (i + 2).julian();
  486.             }

  487.             if (bEOSAmerican) {     // Creation of the American call and put schedule
  488.                 eosCall = EmbeddedOptionSchedule.FromAmerican (
  489.                     DateUtil.CreateFromYMD (
  490.                         2018,
  491.                         9,
  492.                         11
  493.                     ).julian() + 1,
  494.                     aiDate,
  495.                     adblCallFactor,
  496.                     false,
  497.                     30,
  498.                     false,
  499.                     Double.NaN,
  500.                     "",
  501.                     Double.NaN
  502.                 );

  503.                 eosPut = EmbeddedOptionSchedule.FromAmerican (
  504.                     DateUtil.CreateFromYMD (
  505.                         2018,
  506.                         9,
  507.                         11
  508.                     ).julian(),
  509.                     aiDate,
  510.                     adblPutFactor,
  511.                     true,
  512.                     30,
  513.                     false,
  514.                     Double.NaN,
  515.                     "",
  516.                     Double.NaN
  517.                 );
  518.             } else {        // Creation of the European call and put schedule
  519.                 eosCall = new EmbeddedOptionSchedule (
  520.                     aiDate,
  521.                     adblCallFactor,
  522.                     false,
  523.                     30,
  524.                     false,
  525.                     Double.NaN,
  526.                     "",
  527.                     Double.NaN
  528.                 );

  529.                 eosPut = new EmbeddedOptionSchedule (
  530.                     aiDate,
  531.                     adblPutFactor,
  532.                     true,
  533.                     30,
  534.                     false,
  535.                     Double.NaN,
  536.                     "",
  537.                     Double.NaN
  538.                 );
  539.             }

  540.             bond.setEmbeddedCallSchedule (eosCall);

  541.             bond.setEmbeddedPutSchedule (eosPut);
  542.         }

  543.         return (Bond) bond;
  544.     }

  545.     /*
  546.      * Sample demonstrating the creation/usage of the custom bond API
  547.      *
  548.      *      USE WITH CARE: This sample ignores errors and does not handle exceptions.
  549.      */

  550.     private static final void CustomBondAPISample()
  551.         throws Exception
  552.     {
  553.         Bond[] aBond = new Bond[3];
  554.         String strCreditCurve = "CC";

  555.         /*
  556.          * Creates a simple fixed coupon bond and adds it to the FI cache as a named object
  557.          */

  558.         aBond[0] = CreateCustomBond (
  559.             "CustomFixed",
  560.             strCreditCurve,
  561.             BondBuilder.BOND_TYPE_SIMPLE_FIXED
  562.         );

  563.         /*
  564.          * Creates a simple floater and adds it to the FI cache as a named object
  565.          */

  566.         aBond[1] = CreateCustomBond (
  567.             "CustomFRN",
  568.             strCreditCurve,
  569.             BondBuilder.BOND_TYPE_SIMPLE_FLOATER
  570.         );

  571.         /*
  572.          * Creates a custom bond from arbitrary cash flows and adds it to the FI cache as a named object
  573.          */

  574.         aBond[2] = CreateCustomBond (
  575.             "CustomBondFromCF",
  576.             strCreditCurve,
  577.             BondBuilder.BOND_TYPE_SIMPLE_FROM_CF
  578.         );

  579.         /*
  580.          * Base Discount Curve
  581.          */

  582.         MergedDiscountForwardCurve dc = MakeDiscountCurve (
  583.             DateUtil.CreateFromYMD (
  584.                 2018,
  585.                 9,
  586.                 11
  587.             )
  588.         );

  589.         String[] astrTSYTenor = new String[] {
  590.             "1Y", "2Y", "3Y", "5Y", "7Y", "10Y", "30Y"
  591.         };
  592.         final double[] adblTSYCoupon = new double[] {
  593.             0.0000, 0.00375, 0.00500, 0.0100, 0.01375, 0.01375, 0.02875
  594.         };
  595.         double[] adblTSYYield = new double[] {
  596.             0.00160, 0.00397, 0.00696, 0.01421, 0.01955, 0.02529, 0.03568
  597.         };

  598.         TreasuryComponent[] aTSYBond = OTRTreasurySet (
  599.             DateUtil.CreateFromYMD (
  600.                 2018,
  601.                 9,
  602.                 11
  603.             ),
  604.             astrTSYTenor,
  605.             adblTSYCoupon
  606.         );

  607.         /*
  608.          * Create the on-the-run treasury discount curve
  609.          */

  610.         GovvieCurve gc = BuildOnTheRunGovvieCurve (
  611.             DateUtil.CreateFromYMD (
  612.                 2018,
  613.                 9,
  614.                 11
  615.             ),
  616.             aTSYBond,
  617.             adblTSYYield
  618.         );

  619.         /*
  620.          * Credit Curve
  621.          */

  622.         CreditCurve cc = ScenarioCreditCurveBuilder.FlatHazard (
  623.             DateUtil.CreateFromYMD (
  624.                 2018,
  625.                 9,
  626.                 11
  627.             ).julian(),
  628.             strCreditCurve,
  629.             "USD",
  630.             0.01,
  631.             0.4
  632.         );

  633.         for (int i = 0; i < aBond.length; ++i) {
  634.             System.out.println ("\nBOND #" + i + "; " + aBond[i].name());

  635.             System.out.println ("--------------------------");

  636.             System.out.println ("--------------------------");

  637.             System.out.println ("\n\tAcc Start     Acc End     Pay Date      Cpn DCF       Pay01       Surv01");

  638.             System.out.println ("\t---------    ---------    ---------    ---------    ---------    --------");

  639.             /*
  640.              * Generates and displays the coupon period details for the bonds
  641.              */

  642.             for (CompositePeriod p : aBond[i].couponPeriods())
  643.                 System.out.println (
  644.                     DateUtil.YYYYMMDD (p.startDate()) + FIELD_SEPARATOR +
  645.                     DateUtil.YYYYMMDD (p.endDate()) + FIELD_SEPARATOR +
  646.                     DateUtil.YYYYMMDD (p.payDate()) + FIELD_SEPARATOR +
  647.                     FormatUtil.FormatDouble (p.couponDCF(), 1, 4, 1.) + FIELD_SEPARATOR +
  648.                     FormatUtil.FormatDouble (dc.df (p.payDate()), 1, 4, 1.) + FIELD_SEPARATOR +
  649.                     FormatUtil.FormatDouble (cc.survival (p.payDate()), 1, 4, 1.)
  650.                 );

  651.             /*
  652.              * Create the bond's component market parameters from the market inputs
  653.              */

  654.             CurveSurfaceQuoteContainer mktParams = MarketParamsBuilder.Create (
  655.                 dc,     // Discount curve
  656.                 gc,     // TSY Discount Curve (Includes Optional EDSF if available, or BILLS etc)
  657.                 cc,     // Credit Curve
  658.                 null,   // TSY quotes
  659.                 null,   // BOND ID
  660.                 null,   // Bond market quote
  661.                 Helper.CreateFixingsObject (
  662.                     aBond[i],
  663.                     DateUtil.CreateFromYMD (
  664.                         2018,
  665.                         9,
  666.                         11
  667.                     ),
  668.                     0.04    // Fixings
  669.                 )
  670.             );

  671.             AccumulateBondMarketQuote (
  672.                 mktParams,
  673.                 new String[] {
  674.                     "01YON",
  675.                     "02YON",
  676.                     "03YON",
  677.                     "05YON",
  678.                     "07YON",
  679.                     "10YON",
  680.                     "30YON"
  681.                 },
  682.                 adblTSYYield
  683.             );

  684.             /*
  685.              * Construct Valuation Parameters
  686.              */

  687.             ValuationParams valParams = ValuationParams.Spot (
  688.                 DateUtil.CreateFromYMD (
  689.                     2018,
  690.                     9,
  691.                     11
  692.                 ),
  693.                 0,
  694.                 "",
  695.                 Convention.DATE_ROLL_ACTUAL
  696.             );

  697.             ProductQuote cquote = QuoteBuilder.CreateProductQuote();

  698.             Quote q = QuoteBuilder.CreateQuote (
  699.                 "mid",
  700.                 0.05,
  701.                 Double.NaN
  702.             );

  703.             cquote.addQuote (
  704.                 "Yield",
  705.                 q,
  706.                 true
  707.             );

  708.             mktParams.setProductQuote (
  709.                 aBond[i].name(),
  710.                 cquote
  711.             );

  712.             System.out.println ("\n\tPrice From Yield: " +
  713.                 FormatUtil.FormatDouble (
  714.                     aBond[i].priceFromYield (
  715.                         valParams,
  716.                         mktParams,
  717.                         null,
  718.                         0.03
  719.                     ), 1, 3, 100.));

  720.             double dblPrice = aBond[i].priceFromYield (
  721.                 valParams,
  722.                 mktParams,
  723.                 null,
  724.                 0.03
  725.             );

  726.             WorkoutInfo wi = aBond[i].exerciseYieldFromPrice (
  727.                 valParams,
  728.                 mktParams,
  729.                 null,
  730.                 dblPrice
  731.             );

  732.             System.out.println ("\tWorkout Date: " + DateUtil.YYYYMMDD (wi.date()));

  733.             System.out.println ("\tWorkout Factor: " + wi.factor());

  734.             System.out.println ("\tWorkout Yield: " + FormatUtil.FormatDouble (wi.yield(), 1, 2, 100.));

  735.             System.out.println (
  736.                 "\tWorkout Yield From Price: " +
  737.                 FormatUtil.FormatDouble (
  738.                     aBond[i].yieldFromPrice (
  739.                         valParams,
  740.                         mktParams,
  741.                         null,
  742.                         wi.date(),
  743.                         wi.factor(),
  744.                         1.
  745.                     ), 1, 2, 100.));

  746.             if (!aBond[i].isFloater()) {
  747.                 System.out.println (
  748.                     "\tZ Spread From Price: " +
  749.                     FormatUtil.FormatDouble (
  750.                         aBond[i].zSpreadFromPrice (
  751.                             valParams,
  752.                             mktParams,
  753.                             null,
  754.                             wi.date(),
  755.                             wi.factor(),
  756.                             1.
  757.                         ), 1, 0, 10000.));

  758.                 System.out.println (
  759.                     "\tOAS From Price: " +
  760.                         FormatUtil.FormatDouble (
  761.                         aBond[i].oasFromPrice (
  762.                             valParams,
  763.                             mktParams,
  764.                             null,
  765.                             wi.date(),
  766.                             wi.factor(),
  767.                             1.
  768.                         ), 1, 0, 10000.));
  769.             } else;
  770.                 System.out.println (
  771.                     "\tDiscount Margin From Price: " +
  772.                         FormatUtil.FormatDouble (
  773.                         aBond[i].discountMarginFromPrice (
  774.                             valParams,
  775.                             mktParams,
  776.                             null,
  777.                             wi.date(),
  778.                             wi.factor(),
  779.                             1.
  780.                         ), 1, 0, 10000.));

  781.             System.out.println (
  782.                 "\tI Spread From Price: " +
  783.                 FormatUtil.FormatDouble (
  784.                     aBond[i].iSpreadFromPrice (
  785.                         valParams,
  786.                         mktParams,
  787.                         null,
  788.                         wi.date(),
  789.                         wi.factor(),
  790.                         1.
  791.                     ), 1, 0, 10000.));

  792.             double dblTreasurySpread = Double.NaN;

  793.             System.out.println (
  794.                 "\tTSY Spread From Price: " +
  795.                 FormatUtil.FormatDouble (
  796.                     dblTreasurySpread = aBond[i].tsySpreadFromPrice (
  797.                         valParams,
  798.                         mktParams,
  799.                         null,
  800.                         wi.date(),
  801.                         wi.factor(),
  802.                         1.
  803.                     ), 1, 0, 10000.));

  804.             System.out.println (
  805.                 "\tASW From Price: " +
  806.                 FormatUtil.FormatDouble (
  807.                     aBond[i].aswFromPrice (
  808.                         valParams,
  809.                         mktParams,
  810.                         null,
  811.                         wi.date(),
  812.                         wi.factor(),
  813.                         1.
  814.                     ), 1, 0, 10000.));

  815.             System.out.println (
  816.                 "\tCredit Basis From Price: " +
  817.                 FormatUtil.FormatDouble (
  818.                     aBond[i].creditBasisFromPrice (
  819.                         valParams,
  820.                         mktParams,
  821.                         null,
  822.                         wi.date(),
  823.                         wi.factor(),
  824.                         1.
  825.                     ), 1, 0, 10000.));

  826.             System.out.println (
  827.                 "\tPrice From TSY Spread: " +
  828.                 FormatUtil.FormatDouble (
  829.                     aBond[i].priceFromTSYSpread (
  830.                         valParams,
  831.                         mktParams,
  832.                         null,
  833.                         wi.date(),
  834.                         wi.factor(),
  835.                         dblTreasurySpread
  836.                     ), 1, 0, 100.));

  837.             System.out.println (
  838.                 "\tYield From TSY Spread: " +
  839.                 FormatUtil.FormatDouble (
  840.                     aBond[i].yieldFromTSYSpread (
  841.                         valParams,
  842.                         mktParams,
  843.                         null,
  844.                         wi.date(),
  845.                         wi.factor(),
  846.                         dblTreasurySpread
  847.                     ), 1, 0, 100.));

  848.             System.out.println (
  849.                 "\tASW From TSY Spread: " +
  850.                 FormatUtil.FormatDouble (
  851.                     aBond[i].aswFromTSYSpread (
  852.                         valParams,
  853.                         mktParams,
  854.                         null,
  855.                         wi.date(),
  856.                         wi.factor(),
  857.                         dblTreasurySpread
  858.                     ), 1, 0, 10000.));

  859.             System.out.println (
  860.                 "\tCredit Basis From TSY Spread: " +
  861.                 FormatUtil.FormatDouble (
  862.                     aBond[i].creditBasisFromTSYSpread (
  863.                         valParams,
  864.                         mktParams,
  865.                         null,
  866.                         wi.date(),
  867.                         wi.factor(),
  868.                         dblTreasurySpread
  869.                     ), 1, 0, 10000.));

  870.             /* System.out.println ("\tPECS From TSY Spread: " + FormatUtil.FormatDouble
  871.                 (aBond[i].pecsFromTSYSpread (valParams, mktParams, null, 0.0188), 1, 0, 10000.)); */

  872.             System.out.println (
  873.                 "\tTheoretical Price: " +
  874.                 FormatUtil.FormatDouble (
  875.                     aBond[i].priceFromCreditBasis (
  876.                         valParams,
  877.                         mktParams,
  878.                         null,
  879.                         wi.date(),
  880.                         wi.factor(),
  881.                         0.
  882.                     ), 1, 2, 100.));
  883.         }
  884.     }

  885.     /*
  886.      * API demonstrating how to calibrate a CDS curve from CDS and bond quotes
  887.      *
  888.      *      USE WITH CARE: This sample ignores errors and does not handle exceptions.
  889.      */

  890.     private static void BondCDSCurveCalibration()
  891.         throws Exception
  892.     {
  893.         /*
  894.          * Bond calibration instrument
  895.          */

  896.         Bond bond = BondBuilder.CreateSimpleFixed (
  897.             "CCCalibBond",
  898.             "USD",
  899.             "CC",
  900.             0.05,
  901.             2,
  902.             "30/360",
  903.             DateUtil.CreateFromYMD (
  904.                 2008,
  905.                 9,
  906.                 21
  907.             ),
  908.             DateUtil.CreateFromYMD (
  909.                 2023,
  910.                 9,
  911.                 20
  912.             ),
  913.             null,
  914.             null
  915.         );

  916.         /*
  917.          * Discount Curve
  918.          */

  919.         MergedDiscountForwardCurve dc = ScenarioDiscountCurveBuilder.ExponentiallyCompoundedFlatRate (
  920.             DateUtil.CreateFromYMD (
  921.                 2018,
  922.                 9,
  923.                 11
  924.             ),
  925.             "USD",
  926.             0.04
  927.         );

  928.         /*
  929.          * Credit Curve
  930.          */

  931.         CreditCurve cc = ScenarioCreditCurveBuilder.FlatHazard (
  932.             DateUtil.CreateFromYMD (
  933.                 2018,
  934.                 9,
  935.                 11
  936.             ).julian(),
  937.             "CC",
  938.             "USD",
  939.             0.01,
  940.             0.4
  941.         );

  942.         /*
  943.          * Component Market Parameters Container
  944.          */

  945.         CurveSurfaceQuoteContainer mktParams =  MarketParamsBuilder.Create (
  946.             dc,
  947.             null,
  948.             null,
  949.             cc,
  950.             null,
  951.             null,
  952.             null,
  953.             null
  954.         );

  955.         /*
  956.          * Valuation Parameters
  957.          */

  958.         ValuationParams valParams = ValuationParams.Spot (
  959.             DateUtil.CreateFromYMD (
  960.                 2018,
  961.                 9,
  962.                 11
  963.             ),
  964.             0,
  965.             "USD",
  966.             Convention.DATE_ROLL_ACTUAL
  967.         );

  968.         /*
  969.          * Theoretical Price
  970.          */

  971.         double dblTheoreticalPrice = bond.priceFromCreditBasis (
  972.             valParams,
  973.             mktParams,
  974.             null,
  975.             bond.maturityDate().julian(),
  976.             1.,
  977.             0.01
  978.         );


  979.         System.out.println ("Credit Price From DC and CC: " + dblTheoreticalPrice);

  980.         /*
  981.          * CDS calibration instrument
  982.          */

  983.         CreditDefaultSwap cds = CDSBuilder.CreateCDS (
  984.             DateUtil.CreateFromYMD (
  985.                 2018,
  986.                 9,
  987.                 11
  988.             ),
  989.             DateUtil.CreateFromYMD (
  990.                 2018,
  991.                 9,
  992.                 11
  993.             ).addTenor ("5Y"),
  994.             0.1,
  995.             "USD",
  996.             0.40,
  997.             "CC",
  998.             "USD",
  999.             true
  1000.         );

  1001.         /*
  1002.          * Set up the calibration instruments
  1003.          */

  1004.         CalibratableComponent[] aCalibInst = new CalibratableComponent[] {
  1005.             cds,
  1006.             bond
  1007.         };

  1008.         /*
  1009.          * Set up the calibration measures
  1010.          */

  1011.         String[] astrCalibMeasure = new String[] {
  1012.             "FairPremium",
  1013.             "FairPrice"
  1014.         };

  1015.         /*
  1016.          * Set up the calibration quotes
  1017.          */

  1018.         double[] adblQuotes = new double[] {
  1019.             100.,
  1020.             dblTheoreticalPrice
  1021.         };

  1022.         /*
  1023.          * Setup the curve scenario calibrator/generator and build the credit curve
  1024.          */

  1025.         CreditCurve ccCalib = ScenarioCreditCurveBuilder.Custom (
  1026.             "CC",                   // Name
  1027.             DateUtil.CreateFromYMD (
  1028.                 2018,
  1029.                 9,
  1030.                 11
  1031.             ),                      // Date
  1032.             aCalibInst,             // Calibration instruments
  1033.             dc,                     // Discount Curve
  1034.             adblQuotes,             // Component Quotes
  1035.             astrCalibMeasure,       // Calibration Measures
  1036.             0.40,                   // Recovery
  1037.             false                   // Calibration is not flat
  1038.         );

  1039.         /*
  1040.          * Calculate the survival probability, and recover the input quotes
  1041.          */

  1042.         System.out.println (
  1043.             "Surv (2021, 1, 14): " +
  1044.             ccCalib.survival (
  1045.                 DateUtil.CreateFromYMD (
  1046.                     2021,
  1047.                     1,
  1048.                     14
  1049.                 )
  1050.             )
  1051.         );

  1052.         /*
  1053.          * Calibrated Component Market Parameters Container
  1054.          */

  1055.         CurveSurfaceQuoteContainer mktParamsCalib = MarketParamsBuilder.Create (
  1056.             dc,
  1057.             null,
  1058.             null,
  1059.             ccCalib,
  1060.             null,
  1061.             null,
  1062.             null,
  1063.             null
  1064.         );

  1065.         /*
  1066.          * Verify the CDS fair premium using the calibrated credit curve
  1067.          */

  1068.         System.out.println (
  1069.             cds.primaryCode() + " => " + cds.measureValue (
  1070.                 valParams,
  1071.                 CreditPricerParams.Standard(),
  1072.                 mktParamsCalib,
  1073.                 null,
  1074.                 "FairPremium"
  1075.             )
  1076.         );

  1077.         /*
  1078.          * Verify the Bond fair price using the calibrated credit curve
  1079.          */

  1080.         System.out.println (
  1081.             bond.primaryCode() + " => " + bond.priceFromCreditBasis (
  1082.                 valParams,
  1083.                 mktParamsCalib,
  1084.                 null,
  1085.                 bond.maturityDate().julian(),
  1086.                 1.,
  1087.                 0.
  1088.             )
  1089.         );
  1090.     }

  1091.     public static final void main (
  1092.         final String astrArgs[])
  1093.         throws Exception
  1094.     {
  1095.         // String strConfig = "c:\\Lakshmi\\BondAnal\\Config.xml";

  1096.         EnvManager.InitEnv (
  1097.             "",
  1098.             true
  1099.         );

  1100.         CustomBondAPISample();

  1101.         BondCDSCurveCalibration();

  1102.         EnvManager.TerminateEnv();
  1103.     }
  1104. }