/** * Copyright (C) 2011 - present by OpenGamma Inc. and the OpenGamma group of companies * * Please see distribution for license. */ package com.opengamma.analytics.financial.instrument.bond; import static org.testng.AssertJUnit.assertEquals; import static org.testng.AssertJUnit.assertTrue; import org.testng.annotations.Test; import org.threeten.bp.Period; import org.threeten.bp.ZonedDateTime; import com.opengamma.analytics.financial.instrument.payment.CouponFixedDefinition; import com.opengamma.analytics.financial.interestrate.bond.definition.BondFixedSecurity; import com.opengamma.analytics.financial.interestrate.bond.definition.BondFixedTransaction; import com.opengamma.analytics.financial.interestrate.bond.provider.BondSecurityDiscountingMethod; import com.opengamma.analytics.financial.schedule.ScheduleCalculator; import com.opengamma.financial.convention.businessday.BusinessDayConvention; import com.opengamma.financial.convention.businessday.BusinessDayConventions; import com.opengamma.financial.convention.calendar.Calendar; import com.opengamma.financial.convention.calendar.MondayToFridayCalendar; import com.opengamma.financial.convention.daycount.AccruedInterestCalculator; import com.opengamma.financial.convention.daycount.ActualActualICMA; import com.opengamma.financial.convention.daycount.DayCount; import com.opengamma.financial.convention.daycount.DayCounts; import com.opengamma.financial.convention.yield.SimpleYieldConvention; import com.opengamma.financial.convention.yield.YieldConvention; import com.opengamma.financial.convention.yield.YieldConventionFactory; import com.opengamma.util.money.Currency; import com.opengamma.util.test.TestGroup; import com.opengamma.util.time.DateUtils; /** * Test. */ @Test(groups = TestGroup.UNIT) public class BondFixedTransactionDefinitionTest { //Issue: Semi-annual 2Y private static final Currency CUR = Currency.EUR; private static final Period PAYMENT_TENOR = Period.ofMonths(6); private static final Calendar CALENDAR = new MondayToFridayCalendar("A"); private static final String ISSUER_NAME = "Issuer"; private static final DayCount DAY_COUNT = DayCounts.ACT_ACT_ICMA; private static final BusinessDayConvention BUSINESS_DAY = BusinessDayConventions.FOLLOWING; private static final boolean IS_EOM = false; private static final Period BOND_TENOR = Period.ofYears(2); private static final int SETTLEMENT_DAYS = 2; private static final ZonedDateTime START_ACCRUAL_DATE = DateUtils.getUTCDate(2011, 7, 13); private static final ZonedDateTime MATURITY_DATE = START_ACCRUAL_DATE.plus(BOND_TENOR); private static final double RATE = 0.0325; private static final YieldConvention YIELD_CONVENTION = YieldConventionFactory.INSTANCE.getYieldConvention("STREET CONVENTION"); private static final BondFixedSecurityDefinition BOND_SECURITY_DEFINITION = BondFixedSecurityDefinition.from(CUR, MATURITY_DATE, START_ACCRUAL_DATE, PAYMENT_TENOR, RATE, SETTLEMENT_DAYS, CALENDAR, DAY_COUNT, BUSINESS_DAY, YIELD_CONVENTION, IS_EOM, ISSUER_NAME); // Transaction private static final double PRICE_DIRTY = 0.90; private static final double YIELD = 0.05; private static final double PRICE_CLEAN = 0.95; private static final ZonedDateTime SETTLEMENT_DATE = DateUtils.getUTCDate(2011, 8, 18); private static final double QUANTITY = 100000000; //100m private static final BondFixedTransactionDefinition BOND_TRANSACTION_DEFINITION = new BondFixedTransactionDefinition(BOND_SECURITY_DEFINITION, QUANTITY, SETTLEMENT_DATE, PRICE_DIRTY); // to derivatives: common private static final String CREDIT_CURVE_NAME = "Credit"; private static final String REPO_CURVE_NAME = "Repo"; private static final String[] CURVES_NAME = {CREDIT_CURVE_NAME, REPO_CURVE_NAME }; private static final ZonedDateTime REFERENCE_DATE_1 = DateUtils.getUTCDate(2011, 8, 17); private static final BondSecurityDiscountingMethod METHOD_BOND_FIXED = BondSecurityDiscountingMethod.getInstance(); private static final double TOLERANCE_PRICE = 1.0E-8; @Test(expectedExceptions = IllegalArgumentException.class) public void testNullUnderlying() { new BondFixedTransactionDefinition(null, QUANTITY, SETTLEMENT_DATE, PRICE_DIRTY); } @Test(expectedExceptions = IllegalArgumentException.class) public void testNullUnderlyingYield() { BondFixedTransactionDefinition.fromYield(null, QUANTITY, SETTLEMENT_DATE, PRICE_DIRTY); } @Test(expectedExceptions = IllegalArgumentException.class) public void testNullSettle() { new BondFixedTransactionDefinition(BOND_SECURITY_DEFINITION, QUANTITY, null, PRICE_DIRTY); } @Test(expectedExceptions = IllegalArgumentException.class) public void testNullSettleYield() { BondFixedTransactionDefinition.fromYield(BOND_SECURITY_DEFINITION, QUANTITY, null, YIELD); } @Test public void testGetters() { assertEquals(PRICE_DIRTY, BOND_TRANSACTION_DEFINITION.getPrice()); assertEquals(QUANTITY, BOND_TRANSACTION_DEFINITION.getQuantity()); assertEquals(SETTLEMENT_DATE, BOND_TRANSACTION_DEFINITION.getSettlementDate()); assertEquals(BOND_SECURITY_DEFINITION, BOND_TRANSACTION_DEFINITION.getUnderlyingBond()); final double expectedAccrued = 0.195652174 * RATE / 2; //36 days out of 184 in Actual/Actual ICMA. assertEquals(expectedAccrued, BOND_TRANSACTION_DEFINITION.getAccruedInterestAtSettlement(), 1E-6); assertEquals(DateUtils.getUTCDate(2011, 7, 13), BOND_TRANSACTION_DEFINITION.getPreviousAccrualDate()); assertEquals(DateUtils.getUTCDate(2012, 1, 13), BOND_TRANSACTION_DEFINITION.getNextAccrualDate()); } @Test public void ofYield() { BondFixedTransactionDefinition bondOfYieldTransactionDefinition = BondFixedTransactionDefinition.fromYield(BOND_SECURITY_DEFINITION, QUANTITY, SETTLEMENT_DATE, YIELD); BondFixedSecurity bondOfYieldSecurity = bondOfYieldTransactionDefinition.getUnderlyingBond().toDerivative(REFERENCE_DATE_1, SETTLEMENT_DATE); double cleanPrice = METHOD_BOND_FIXED.cleanPriceFromYield(bondOfYieldSecurity, YIELD); assertEquals("Bond transaction: ofYield", cleanPrice, bondOfYieldTransactionDefinition.getPrice(), TOLERANCE_PRICE); } @Test public void toDerivatives() { final BondFixedSecurity bondSecurityStandard = BOND_SECURITY_DEFINITION.toDerivative(REFERENCE_DATE_1); final BondFixedTransaction bondTransaction = BOND_TRANSACTION_DEFINITION.toDerivative(REFERENCE_DATE_1); assertEquals("Bond transaction: toDerivative", bondSecurityStandard, bondTransaction.getBondStandard()); final BondFixedSecurity bondSecurityPurchase = BOND_SECURITY_DEFINITION.toDerivative(REFERENCE_DATE_1, SETTLEMENT_DATE); assertEquals("Bond transaction: toDerivative", bondSecurityPurchase.getAccruedInterest(), bondTransaction.getBondTransaction().getAccruedInterest()); assertEquals("Bond transaction: toDerivative", bondSecurityPurchase.getCouponPerYear(), bondTransaction.getBondTransaction().getCouponPerYear()); assertEquals("Bond transaction: toDerivative", bondSecurityPurchase.getYieldConvention(), bondTransaction.getBondTransaction().getYieldConvention()); } // MDLZ 5 3/8 12/11/14 - ISIN: XS0417033007 - Check the long first coupon and accrued private static final Currency GBP = Currency.GBP; private static final ActualActualICMA DAY_COUNT_ACTACTICMA = new ActualActualICMA(); private static final BusinessDayConvention FOLLOWING = BusinessDayConventions.FOLLOWING; private static final String ISSUER_MON = " MONDELEZ INTERNATIONAL"; private static final YieldConvention US_STREET = SimpleYieldConvention.US_STREET; private static final int SETTLEMENT_DAYS_MON = 3; private static final Period PAYMENT_TENOR_MON = Period.ofMonths(6); // private static final int COUPON_PER_YEAR_MON = 2; private static final ZonedDateTime BOND_MATURITY_MON = DateUtils.getUTCDate(2014, 12, 11); private static final ZonedDateTime BOND_START_MON = DateUtils.getUTCDate(2009, 3, 11); private static final ZonedDateTime BOND_FIRSTCPN_MON = DateUtils.getUTCDate(2009, 12, 11); private static final double RATE_MON = 0.05375; private static final BondFixedSecurityDefinition BOND_MON_SECURITY_DEFINITION = BondFixedSecurityDefinition.from(GBP, BOND_START_MON, BOND_FIRSTCPN_MON, BOND_MATURITY_MON, PAYMENT_TENOR_MON, RATE_MON, SETTLEMENT_DAYS_MON, CALENDAR, DAY_COUNT_ACTACTICMA, FOLLOWING, US_STREET, IS_EOM, ISSUER_MON); private static final double TOLERANCE_ACCRUED = 1.0E-2; @Test public void longFirstCouponMon() { final double notional = 1000000; final ZonedDateTime settle090410 = DateUtils.getUTCDate(2009, 04, 10); final double accrual090410 = 4429.95; BondFixedSecurity bond090410 = BOND_MON_SECURITY_DEFINITION.toDerivative(settle090410, settle090410); assertEquals("Bond transaction: accrued", accrual090410, bond090410.getAccruedInterest() * notional, TOLERANCE_ACCRUED); final ZonedDateTime settle091010 = DateUtils.getUTCDate(2009, 10, 10); final double accrual091010 = 31354.97; BondFixedSecurity bond091010 = BOND_MON_SECURITY_DEFINITION.toDerivative(settle091010, settle091010); assertEquals("Bond transaction: accrued", accrual091010, bond091010.getAccruedInterest() * notional, TOLERANCE_ACCRUED); final ZonedDateTime settle101010 = DateUtils.getUTCDate(2010, 10, 10); final double accrual101010 = 17769.81; BondFixedSecurity bond101010 = BOND_MON_SECURITY_DEFINITION.toDerivative(settle101010, settle101010); assertEquals("Bond transaction: accrued", accrual101010, bond101010.getAccruedInterest() * notional, TOLERANCE_ACCRUED); } private static final BondFixedSecurityDefinition BOND_UKT_500_20140907 = BondDataSetsGbp.bondUKT5_20140907(); /** Test the constructor with settlement date in the ex-coupon period. */ @Test public void settlementExCoupon() { ZonedDateTime settleDate = DateUtils.getUTCDate(2011, 9, 2); BondFixedTransactionDefinition transaction = new BondFixedTransactionDefinition(BOND_UKT_500_20140907, QUANTITY, settleDate, PRICE_DIRTY); int cpnIndex = transaction.getCouponIndex(); CouponFixedDefinition cpn = BOND_UKT_500_20140907.getCoupons().getNthPayment(cpnIndex); double cpnAccrued = cpn.getAmount() / cpn.getNotional(); final int nbCoupon = BOND_UKT_500_20140907.getCoupons().getNumberOfPayments(); double accruedUndajusted = AccruedInterestCalculator.getAccruedInterest(BOND_UKT_500_20140907.getDayCount(), cpnIndex, nbCoupon, transaction.getPreviousAccrualDate(), settleDate, transaction.getNextAccrualDate(), cpn.getRate(), BOND_UKT_500_20140907.getCouponPerYear(), BOND_UKT_500_20140907.isEOM()); double accruedAtSettle = accruedUndajusted - cpnAccrued; double accruedComputed = transaction.getAccruedInterestAtSettlement(); assertTrue("BondFixedTransactionDefinition: ex-coupon accrued negative", accruedComputed < 0.0); assertEquals("BondFixedTransactionDefinition", accruedAtSettle, accruedComputed, TOLERANCE_ACCRUED); } /** Test the constructor with date around coupon. */ @Test public void settlementUKTAllDate() { ZonedDateTime settleDate = DateUtils.getUTCDate(2011, 8, 1); int nbSettle = 40; double[] accruedSettle = new double[nbSettle]; for (int loopsettle = 0; loopsettle < nbSettle; loopsettle++) { settleDate = ScheduleCalculator.getAdjustedDate(settleDate, 1, CALENDAR); BondFixedTransactionDefinition transaction = new BondFixedTransactionDefinition(BOND_UKT_500_20140907, QUANTITY, settleDate, PRICE_DIRTY); accruedSettle[loopsettle] = transaction.getAccruedInterestAtSettlement(); } int indexJump = 19; for (int loopref = 0; loopref < nbSettle - 1; loopref++) { if (loopref != indexJump) { assertTrue("Bond Fixed Security Definition to derivative - " + loopref, accruedSettle[loopref + 1] - accruedSettle[loopref] < 0.05 / 360 * 3); // 3 days of accrued } } @SuppressWarnings("unused") int t = 0; } }