All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.opengamma.strata.report.cashflow.CashFlowReportRunner Maven / Gradle / Ivy

There is a newer version: 2.12.53
Show newest version
/**
 * Copyright (C) 2015 - present by OpenGamma Inc. and the OpenGamma group of companies
 * 
 * Please see distribution for license.
 */
package com.opengamma.strata.report.cashflow;

import static com.opengamma.strata.collect.Guavate.toImmutableList;
import static com.opengamma.strata.collect.Guavate.toImmutableSet;

import java.time.Instant;
import java.time.LocalDate;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.Set;
import java.util.stream.Collectors;

import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableTable;
import com.google.common.collect.Maps;
import com.opengamma.strata.calc.Column;
import com.opengamma.strata.collect.Messages;
import com.opengamma.strata.collect.result.Result;
import com.opengamma.strata.market.explain.ExplainKey;
import com.opengamma.strata.market.explain.ExplainMap;
import com.opengamma.strata.measure.Measures;
import com.opengamma.strata.report.Report;
import com.opengamma.strata.report.ReportCalculationResults;
import com.opengamma.strata.report.ReportRequirements;
import com.opengamma.strata.report.ReportRunner;

/**
 * Report runner for cash flow reports.
 */
public class CashFlowReportRunner
    implements ReportRunner {

  // TODO - when the cashflow report INI file supports specific columns, the following maps should
  // be represented by a built-in report template INI file.

  /**
   * The single shared instance of this report runner.
   */
  public static final CashFlowReportRunner INSTANCE = new CashFlowReportRunner();

  private static final ExplainKey INTERIM_AMOUNT_KEY = ExplainKey.of("InterimAmount");

  private static final Map, String> HEADER_MAP = ImmutableMap.of(
      ExplainKey.ENTRY_TYPE, "Flow Type",
      ExplainKey.ENTRY_INDEX, "Leg Number",
      ExplainKey.FORECAST_VALUE, "Flow Amount");

  private static final List> COLUMN_ORDER = ImmutableList.>builder()
      .add(ExplainKey.ENTRY_TYPE)
      .add(ExplainKey.ENTRY_INDEX)
      .add(ExplainKey.LEG_TYPE)
      .add(ExplainKey.PAY_RECEIVE)
      .add(ExplainKey.PAYMENT_CURRENCY)
      .add(ExplainKey.NOTIONAL)
      .add(ExplainKey.TRADE_NOTIONAL)
      .add(ExplainKey.UNADJUSTED_START_DATE)
      .add(ExplainKey.UNADJUSTED_END_DATE)
      .add(ExplainKey.START_DATE)
      .add(ExplainKey.END_DATE)
      .add(ExplainKey.FIXED_RATE)
      .add(ExplainKey.INDEX)
      .add(ExplainKey.FIXING_DATE)
      .add(ExplainKey.INDEX_VALUE)
      .add(ExplainKey.GEARING)
      .add(ExplainKey.SPREAD)
      .add(ExplainKey.WEIGHT)
      .add(ExplainKey.COMBINED_RATE)
      .add(ExplainKey.PAY_OFF_RATE)
      .add(ExplainKey.UNADJUSTED_PAYMENT_DATE)
      .add(ExplainKey.PAYMENT_DATE)
      .add(ExplainKey.ACCRUAL_DAYS)
      .add(ExplainKey.ACCRUAL_DAY_COUNT)
      .add(ExplainKey.ACCRUAL_YEAR_FRACTION)
      .add(ExplainKey.COMPOUNDING)
      .add(ExplainKey.UNIT_AMOUNT)
      .add(INTERIM_AMOUNT_KEY)
      .add(ExplainKey.FORECAST_VALUE)
      .add(ExplainKey.DISCOUNT_FACTOR)
      .add(ExplainKey.PRESENT_VALUE)
      .build();

  private static final List> INHERITED_KEYS = ImmutableList.>builder()
      .add(ExplainKey.ENTRY_INDEX)
      .add(ExplainKey.LEG_TYPE)
      .add(ExplainKey.PAY_RECEIVE)
      .build();

  // restricted constructor
  private CashFlowReportRunner() {
  }

  //-------------------------------------------------------------------------
  @Override
  public ReportRequirements requirements(CashFlowReportTemplate reportTemplate) {
    return ReportRequirements.of(Column.of(Measures.EXPLAIN_PRESENT_VALUE));
  }

  @Override
  public Report runReport(ReportCalculationResults calculationResults, CashFlowReportTemplate reportTemplate) {

    int tradeCount = calculationResults.getCalculationResults().getRowCount();
    if (tradeCount == 0) {
      throw new IllegalArgumentException("Calculation results is empty");
    }
    if (tradeCount > 1) {
      throw new IllegalArgumentException(
          Messages.format("Unable to show cashflow report for {} trades at once. " +
              "Please filter the portfolio to a single trade.", tradeCount));
    }

    int columnIdx = calculationResults.getColumns().indexOf(Column.of(Measures.EXPLAIN_PRESENT_VALUE));
    if (columnIdx == -1) {
      throw new IllegalArgumentException(
          Messages.format("Unable to find column for required measure '{}' in calculation results",
              Measures.EXPLAIN_PRESENT_VALUE));
    }

    Result result = calculationResults.getCalculationResults().get(0, columnIdx);
    if (result.isFailure()) {
      throw new IllegalArgumentException(
          Messages.format("Failure result found for required measure '{}': {}",
              Measures.EXPLAIN_PRESENT_VALUE, result.getFailure().getMessage()));
    }
    ExplainMap explainMap = (ExplainMap) result.getValue();

    return runReport(explainMap, calculationResults.getValuationDate());
  }

  private Report runReport(ExplainMap explainMap, LocalDate valuationDate) {
    List flatMap = flatten(explainMap);

    List> keys = getKeys(flatMap);
    List headers = keys.stream().map(this::mapHeader).collect(toImmutableList());
    ImmutableTable data = getData(flatMap, keys);

    return CashFlowReport.builder()
        .runInstant(Instant.now())
        .valuationDate(valuationDate)
        .columnKeys(keys)
        .columnHeaders(headers)
        .data(data)
        .build();
  }

  private List flatten(ExplainMap explainMap) {
    List flattenedMap = new ArrayList<>();
    flatten(explainMap, false, ImmutableMap.of(), Maps.newHashMap(), 0, flattenedMap);
    return flattenedMap;
  }

  @SuppressWarnings("unchecked")
  private void flatten(
      ExplainMap explainMap,
      boolean parentVisible,
      Map, Object> parentRow,
      Map, Object> currentRow,
      int level,
      List accumulator) {

    boolean hasParentFlow = currentRow.containsKey(ExplainKey.FORECAST_VALUE);
    boolean isFlow = explainMap.get(ExplainKey.PAYMENT_DATE).isPresent();
    boolean visible = parentVisible || isFlow;

    Set>> nestedListKeys = explainMap.getMap().keySet().stream()
        .filter(k -> List.class.isAssignableFrom(explainMap.get(k).get().getClass()))
        .map(k -> (ExplainKey>) k)
        .collect(toImmutableSet());

    // Populate the base data
    for (Map.Entry, Object> entry : explainMap.getMap().entrySet()) {
      ExplainKey key = entry.getKey();

      if (nestedListKeys.contains(key)) {
        continue;
      }
      if (key.equals(ExplainKey.FORECAST_VALUE)) {
        if (hasParentFlow) {
          // Collapsed rows, so flow must be the same as we already have
          continue;
        } else if (isFlow) {
          // This is first child flow row, so flow is equal to, and replaces, calculated amount
          currentRow.remove(INTERIM_AMOUNT_KEY);
        }
      }
      ExplainKey mappedKey = mapKey(key, isFlow);
      Object mappedValue = mapValue(mappedKey, entry.getValue(), level);
      if (isFlow) {
        currentRow.put(mappedKey, mappedValue);
      } else {
        currentRow.putIfAbsent(mappedKey, mappedValue);
      }
    }

    // Repeat the inherited entries from the parent row if this row hasn't overridden them
    INHERITED_KEYS.stream()
        .filter(parentRow::containsKey)
        .forEach(inheritedKey -> currentRow.putIfAbsent(inheritedKey, parentRow.get(inheritedKey)));

    if (nestedListKeys.size() > 0) {
      List nestedListEntries = nestedListKeys.stream()
          .flatMap(k -> explainMap.get(k).get().stream())
          .sorted(this::compareNestedEntries)
          .collect(Collectors.toList());

      if (nestedListEntries.size() == 1) {
        // Soak it up into this row
        flatten(nestedListEntries.get(0), visible, currentRow, currentRow, level, accumulator);
      } else {
        // Add child rows
        for (ExplainMap nestedListEntry : nestedListEntries) {
          flatten(nestedListEntry, visible, currentRow, Maps.newHashMap(), level + 1, accumulator);
        }
        // Add parent row after child rows (parent flows are a result of the children)
        if (visible) {
          accumulator.add(ExplainMap.of(currentRow));
        }
      }
    } else {
      if (visible) {
        accumulator.add(ExplainMap.of(currentRow));
      }
    }
  }

  private int compareNestedEntries(ExplainMap m1, ExplainMap m2) {
    Optional paymentDate1 = m1.get(ExplainKey.PAYMENT_DATE);
    Optional paymentDate2 = m2.get(ExplainKey.PAYMENT_DATE);
    if (paymentDate1.isPresent() && paymentDate1.isPresent()) {
      return paymentDate1.get().compareTo(paymentDate2.get());
    }
    if (!paymentDate2.isPresent()) {
      return paymentDate1.isPresent() ? 1 : 0;
    }
    return -1;
  }

  private ExplainKey mapKey(ExplainKey key, boolean isFlow) {
    if (!isFlow && key.equals(ExplainKey.FORECAST_VALUE)) {
      return INTERIM_AMOUNT_KEY;
    }
    return key;
  }

  private Object mapValue(ExplainKey key, Object value, int level) {
    if (ExplainKey.ENTRY_TYPE.equals(key) && level > 0) {
      return humanizeUpperCamelCase((String) value);
    }
    return value;
  }

  private String mapHeader(ExplainKey key) {
    String header = HEADER_MAP.get(key);
    if (header != null) {
      return header;
    }
    return humanizeUpperCamelCase(key.getName());
  }

  private String humanizeUpperCamelCase(String str) {
    StringBuilder buf = new StringBuilder(str.length() + 4);
    int lastIndex = 0;
    for (int i = 2; i < str.length(); i++) {
      char cur = str.charAt(i);
      char last = str.charAt(i - 1);
      if (Character.getType(last) == Character.UPPERCASE_LETTER && Character.getType(cur) == Character.LOWERCASE_LETTER) {
        buf.append(str.substring(lastIndex, i - 1)).append(' ');
        lastIndex = i - 1;
      }
    }
    buf.append(str.substring(lastIndex));
    return buf.toString();
  }

  private List> getKeys(List explainMap) {
    return explainMap.stream()
        .flatMap(m -> m.getMap().keySet().stream())
        .distinct()
        .sorted((k1, k2) -> COLUMN_ORDER.indexOf(k1) - COLUMN_ORDER.indexOf(k2))
        .collect(Collectors.toList());
  }

  private ImmutableTable getData(List flatMap, List> keys) {
    ImmutableTable.Builder builder = ImmutableTable.builder();

    for (int rowIdx = 0; rowIdx < flatMap.size(); rowIdx++) {
      ExplainMap rowMap = flatMap.get(rowIdx);

      for (int colIdx = 0; colIdx < keys.size(); colIdx++) {
        builder.put(rowIdx, colIdx, rowMap.get(keys.get(colIdx)));
      }
    }
    return builder.build();
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy