com.opengamma.strata.loader.csv.FxSingleTradeCsvPlugin Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of strata-loader Show documentation
Show all versions of strata-loader Show documentation
Loaders from standard and Strata-specific data formats
/*
* Copyright (C) 2017 - present by OpenGamma Inc. and the OpenGamma group of companies
*
* Please see distribution for license.
*/
package com.opengamma.strata.loader.csv;
import static com.opengamma.strata.loader.csv.CsvLoaderColumns.BUY_SELL_FIELD;
import static com.opengamma.strata.loader.csv.CsvLoaderColumns.CONVENTION_FIELD;
import static com.opengamma.strata.loader.csv.CsvLoaderColumns.CURRENCY_FIELD;
import static com.opengamma.strata.loader.csv.CsvLoaderColumns.FX_RATE_FIELD;
import static com.opengamma.strata.loader.csv.CsvLoaderColumns.LEG_1_CURRENCY_FIELD;
import static com.opengamma.strata.loader.csv.CsvLoaderColumns.LEG_1_DIRECTION_FIELD;
import static com.opengamma.strata.loader.csv.CsvLoaderColumns.LEG_1_NOTIONAL_FIELD;
import static com.opengamma.strata.loader.csv.CsvLoaderColumns.LEG_1_PAYMENT_DATE_FIELD;
import static com.opengamma.strata.loader.csv.CsvLoaderColumns.LEG_2_CURRENCY_FIELD;
import static com.opengamma.strata.loader.csv.CsvLoaderColumns.LEG_2_DIRECTION_FIELD;
import static com.opengamma.strata.loader.csv.CsvLoaderColumns.LEG_2_NOTIONAL_FIELD;
import static com.opengamma.strata.loader.csv.CsvLoaderColumns.LEG_2_PAYMENT_DATE_FIELD;
import static com.opengamma.strata.loader.csv.CsvLoaderColumns.NOTIONAL_FIELD;
import static com.opengamma.strata.loader.csv.CsvLoaderColumns.PAYMENT_DATE_CAL_FIELD;
import static com.opengamma.strata.loader.csv.CsvLoaderColumns.PAYMENT_DATE_CNV_FIELD;
import static com.opengamma.strata.loader.csv.CsvLoaderColumns.PAYMENT_DATE_FIELD;
import static com.opengamma.strata.loader.csv.CsvLoaderColumns.TRADE_TYPE_FIELD;
import java.time.LocalDate;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Optional;
import java.util.Set;
import com.google.common.collect.ImmutableSet;
import com.opengamma.strata.basics.currency.CurrencyAmount;
import com.opengamma.strata.basics.currency.CurrencyPair;
import com.opengamma.strata.basics.currency.FxRate;
import com.opengamma.strata.basics.currency.Payment;
import com.opengamma.strata.basics.date.BusinessDayAdjustment;
import com.opengamma.strata.collect.io.CsvOutput;
import com.opengamma.strata.collect.io.CsvOutput.CsvRowOutputWithHeaders;
import com.opengamma.strata.collect.io.CsvRow;
import com.opengamma.strata.collect.result.ParseFailureException;
import com.opengamma.strata.loader.LoaderUtils;
import com.opengamma.strata.product.AttributeType;
import com.opengamma.strata.product.Trade;
import com.opengamma.strata.product.TradeInfo;
import com.opengamma.strata.product.common.BuySell;
import com.opengamma.strata.product.common.PayReceive;
import com.opengamma.strata.product.fx.FxSingle;
import com.opengamma.strata.product.fx.FxSingleTrade;
/**
* Handles the CSV file format for FX Single trades.
*/
class FxSingleTradeCsvPlugin implements TradeCsvParserPlugin, TradeCsvWriterPlugin {
/**
* The singleton instance of the plugin.
*/
public static final FxSingleTradeCsvPlugin INSTANCE = new FxSingleTradeCsvPlugin();
/** The headers. */
private static final ImmutableSet HEADERS = ImmutableSet.of(
LEG_1_DIRECTION_FIELD,
LEG_1_PAYMENT_DATE_FIELD,
LEG_1_CURRENCY_FIELD,
LEG_1_NOTIONAL_FIELD,
LEG_2_DIRECTION_FIELD,
LEG_2_PAYMENT_DATE_FIELD,
LEG_2_CURRENCY_FIELD,
LEG_2_NOTIONAL_FIELD,
PAYMENT_DATE_CNV_FIELD,
PAYMENT_DATE_CAL_FIELD);
//-------------------------------------------------------------------------
@Override
public Set tradeTypeNames() {
return ImmutableSet.of("FX", "FXSINGLE", "FX SINGLE");
}
@Override
public Optional parseTrade(
Class> requiredJavaType,
CsvRow baseRow,
List additionalRows,
TradeInfo info,
TradeCsvInfoResolver resolver) {
if (requiredJavaType.isAssignableFrom(FxSingleTrade.class)) {
return Optional.of(resolver.parseFxSingleTrade(baseRow, info));
}
return Optional.empty();
}
@Override
public String getName() {
return FxSingleTrade.class.getSimpleName();
}
@Override
public Set> supportedTradeTypes() {
return ImmutableSet.of(FxSingleTrade.class);
}
//-------------------------------------------------------------------------
/**
* Parses the data from a CSV row.
*
* @param row the CSV row object
* @param info the trade info object
* @param resolver the resolver used to parse additional information
* @return the parsed trade
*/
static FxSingleTrade parse(CsvRow row, TradeInfo info, TradeCsvInfoResolver resolver) {
FxSingleTrade trade = parseRow(row, info);
return resolver.completeTrade(row, trade);
}
// parses the trade
static FxSingleTrade parseRow(CsvRow row, TradeInfo info) {
if (row.findValue(CONVENTION_FIELD).isPresent()) {
return parseConvention(row, info);
} else {
return parseFull(row, info);
}
}
// convention-based
// ideally we'd use the trade date plus "period to start" to get the spot/payment date
// but we don't have all the data and it gets complicated in places like TRY, RUB and AED
private static FxSingleTrade parseConvention(CsvRow row, TradeInfo info) {
CurrencyPair pair = row.getValue(CONVENTION_FIELD, CurrencyPair::parse);
BuySell buySell = row.getValue(BUY_SELL_FIELD, LoaderUtils::parseBuySell);
CurrencyAmount amount = buySell.normalize(CsvLoaderUtils.parseCurrencyAmount(row, CURRENCY_FIELD, NOTIONAL_FIELD));
double fxRate = row.getValue(FX_RATE_FIELD, LoaderUtils::parseDouble);
LocalDate paymentDate = row.getValue(PAYMENT_DATE_FIELD, LoaderUtils::parseDate);
Optional paymentAdj = parsePaymentDateAdjustment(row);
FxSingle fx = paymentAdj
.map(adj -> FxSingle.of(amount, FxRate.of(pair, fxRate), paymentDate, adj))
.orElseGet(() -> FxSingle.of(amount, FxRate.of(pair, fxRate), paymentDate));
return FxSingleTrade.of(info.withAttribute(AttributeType.BUY_SELL, buySell), fx);
}
// parse full definition
private static FxSingleTrade parseFull(CsvRow row, TradeInfo info) {
Optional buySellOpt = row.findValue(BUY_SELL_FIELD).map(LoaderUtils::parseBuySell);
FxSingle fx = parseFxSingle(row, "");
TradeInfo updatedInfo = buySellOpt.map(buySell -> info.withAttribute(AttributeType.BUY_SELL, buySell)).orElse(info);
return FxSingleTrade.of(updatedInfo, fx);
}
// parse an FxSingle
static FxSingle parseFxSingle(CsvRow row, String prefix) {
CurrencyAmount amount1 = CsvLoaderUtils.parseCurrencyAmountWithDirection(
row, prefix + LEG_1_CURRENCY_FIELD, prefix + LEG_1_NOTIONAL_FIELD, prefix + LEG_1_DIRECTION_FIELD);
LocalDate paymentDate1 = row.findValue(prefix + LEG_1_PAYMENT_DATE_FIELD, LoaderUtils::parseDate)
.orElseGet(() -> row.getValue(prefix + PAYMENT_DATE_FIELD, LoaderUtils::parseDate));
CurrencyAmount amount2 = CsvLoaderUtils.parseCurrencyAmountWithDirection(
row, prefix + LEG_2_CURRENCY_FIELD, prefix + LEG_2_NOTIONAL_FIELD, prefix + LEG_2_DIRECTION_FIELD);
LocalDate paymentDate2 = row.findValue(prefix + LEG_2_PAYMENT_DATE_FIELD, LoaderUtils::parseDate)
.orElseGet(() -> row.getValue(prefix + PAYMENT_DATE_FIELD, LoaderUtils::parseDate));
Optional paymentAdj = parsePaymentDateAdjustment(row);
if (amount1.isPositive() == amount2.isPositive()) {
throw new ParseFailureException(
"FxSingle legs must not have the same direction: {value}, {value2}",
row.getValue(prefix + LEG_1_DIRECTION_FIELD),
row.getValue(prefix + LEG_2_DIRECTION_FIELD));
}
Payment payment1 = Payment.of(amount1, paymentDate1);
Payment payment2 = Payment.of(amount2, paymentDate2);
return paymentAdj
.map(adj -> FxSingle.of(payment1, payment2, adj))
.orElseGet(() -> FxSingle.of(payment1, payment2));
}
// parses the payment date adjustment, which consists of two linked optional fields
static Optional parsePaymentDateAdjustment(CsvRow row) {
return CsvLoaderUtils.parseBusinessDayAdjustment(row, PAYMENT_DATE_CNV_FIELD, PAYMENT_DATE_CAL_FIELD)
.filter(adj -> !adj.equals(BusinessDayAdjustment.NONE));
}
//-------------------------------------------------------------------------
@Override
public Set headers(List trades) {
if (trades.stream()
.anyMatch(trade -> trade.getInfo().findAttribute(AttributeType.BUY_SELL).isPresent())) {
LinkedHashSet headers = new LinkedHashSet<>();
headers.add(BUY_SELL_FIELD);
headers.addAll(HEADERS);
return headers;
}
return HEADERS;
}
@Override
public void writeCsv(CsvRowOutputWithHeaders csv, FxSingleTrade trade) {
csv.writeCell(TRADE_TYPE_FIELD, "FxSingle");
trade.getInfo().findAttribute(AttributeType.BUY_SELL).ifPresent(buySell -> csv.writeCell(BUY_SELL_FIELD, buySell));
writeFxSingle(csv, "", trade.getProduct());
csv.writeNewLine();
}
void writeFxSingle(CsvOutput.CsvRowOutputWithHeaders csv, String prefix, FxSingle product) {
Payment basePayment = product.getBaseCurrencyPayment();
csv.writeCell(prefix + LEG_1_DIRECTION_FIELD, PayReceive.ofSignedAmount(basePayment.getAmount()));
csv.writeCell(prefix + LEG_1_CURRENCY_FIELD, basePayment.getCurrency());
csv.writeCell(prefix + LEG_1_NOTIONAL_FIELD, basePayment.getAmount());
csv.writeCell(prefix + LEG_1_PAYMENT_DATE_FIELD, basePayment.getDate());
Payment counterPayment = product.getCounterCurrencyPayment();
csv.writeCell(prefix + LEG_2_DIRECTION_FIELD, PayReceive.ofSignedAmount(counterPayment.getAmount()));
csv.writeCell(prefix + LEG_2_CURRENCY_FIELD, counterPayment.getCurrency());
csv.writeCell(prefix + LEG_2_NOTIONAL_FIELD, counterPayment.getAmount());
csv.writeCell(prefix + LEG_2_PAYMENT_DATE_FIELD, counterPayment.getDate());
product.getPaymentDateAdjustment().ifPresent(bda -> {
csv.writeCell(PAYMENT_DATE_CAL_FIELD, bda.getCalendar());
csv.writeCell(PAYMENT_DATE_CNV_FIELD, bda.getConvention());
});
}
//-------------------------------------------------------------------------
// Restricted constructor.
private FxSingleTradeCsvPlugin() {
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy